From 3e7684efac78518ac7316a303f3bfc482fcc97a8 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Thu, 5 Jan 2012 14:37:22 +1000 Subject: [PATCH] Some profiling & optimisations. Implemented Session Events Compression, Backup edf File compression for ResMed, preference changed around a bit, and new options for the compression and backup stuff. And more efficient Weighted Percentile calculations --- Graphs/gSummaryChart.cpp | 6 +- SleepLib/calcs.cpp | 1 - SleepLib/common.cpp | 5 + SleepLib/common.h | 19 + SleepLib/day.cpp | 119 ++- SleepLib/day.h | 1 + SleepLib/event.cpp | 84 +- SleepLib/event.h | 8 + SleepLib/loader_plugins/cms50_loader.cpp | 8 +- SleepLib/loader_plugins/intellipap_loader.cpp | 6 +- SleepLib/loader_plugins/prs1_loader.cpp | 8 +- SleepLib/loader_plugins/resmed_loader.cpp | 260 ++++-- SleepLib/machine_common.h | 1 + SleepLib/machine_loader.cpp | 39 + SleepLib/machine_loader.h | 3 + SleepLib/profiles.cpp | 277 +++--- SleepLib/profiles.h | 492 +++++------ SleepLib/session.cpp | 225 +++-- daily.cpp | 19 +- mainwindow.cpp | 8 +- newprofile.cpp | 4 +- overview.cpp | 4 +- overview.h | 2 +- overview.ui | 22 +- preferencesdialog.cpp | 32 +- preferencesdialog.h | 2 + preferencesdialog.ui | 812 +++++++++--------- 27 files changed, 1457 insertions(+), 1010 deletions(-) diff --git a/Graphs/gSummaryChart.cpp b/Graphs/gSummaryChart.cpp index d64e6737..850b5384 100644 --- a/Graphs/gSummaryChart.cpp +++ b/Graphs/gSummaryChart.cpp @@ -427,6 +427,7 @@ void SummaryChart::paint(gGraph & w,int left, int top, int width, int height) if (!m_hours.contains(zd)) goto jumpnext; //continue; + hours=m_hours[zd]; int x1=px; @@ -473,12 +474,13 @@ void SummaryChart::paint(gGraph & w,int left, int top, int width, int height) outlines->add(x1,py,x2,py,x2,py,x2,py-h); } // if (bar //py-=h; - totalvalues[0]+=tmp; + totalvalues[0]+=hours*tmp; } - totalcounts[0]++; + totalcounts[0]+=hours; totalvalues[1]+=j; totalcounts[1]++; total_val+=hours; + total_hours+=hours; total_days++; } else { if (!d.value().contains(0)) goto jumpnext; diff --git a/SleepLib/calcs.cpp b/SleepLib/calcs.cpp index 2261c82b..4f990a14 100644 --- a/SleepLib/calcs.cpp +++ b/SleepLib/calcs.cpp @@ -409,7 +409,6 @@ int calcAHIGraph(Session *session) const qint64 window_size=3600000L; const qint64 window_step=30000; // 30 second windows - if (session->machine()->GetType()!=MT_CPAP) return 0; if (session->eventlist.contains(CPAP_AHI)) return 0; // abort if already there diff --git a/SleepLib/common.cpp b/SleepLib/common.cpp index f80ee1fc..7ea4ac91 100644 --- a/SleepLib/common.cpp +++ b/SleepLib/common.cpp @@ -36,3 +36,8 @@ QString weightString(float kg, UnitSystem us) } return("Bad UnitSystem"); } + +bool operator <(const ValueCount & a, const ValueCount & b) +{ + return a.value < b.value; +} diff --git a/SleepLib/common.h b/SleepLib/common.h index a072de50..c6eb8e11 100644 --- a/SleepLib/common.h +++ b/SleepLib/common.h @@ -6,11 +6,29 @@ enum UnitSystem { US_Undefined, US_Metric, US_Archiac }; +typedef float EventDataType; + +struct ValueCount { + ValueCount() { value=0; count=0; p=0; } + ValueCount(const ValueCount & copy) { + value=copy.value; + count=copy.count; + p=copy.p; + } + EventDataType value; + int count; + double p; +}; + +// Primarily sort by value +bool operator <(const ValueCount & a, const ValueCount & b); + const float ounce_convert=28.3495231; // grams const float pound_convert=ounce_convert*16; QString weightString(float kg, UnitSystem us=US_Undefined); + const QString STR_UNIT_CM=QObject::tr("cm"); const QString STR_UNIT_INCH=QObject::tr("\""); const QString STR_UNIT_FOOT=QObject::tr("ft"); @@ -40,6 +58,7 @@ const QString STR_PROP_SubModel="SubModel"; const QString STR_PROP_Serial="Serial"; const QString STR_PROP_DataVersion="DataVersion"; const QString STR_PROP_Path="Path"; +const QString STR_PROP_BackupPath="BackupPath"; const QString STR_PROP_LastImported="LastImported"; const QString STR_MACH_ResMed="ResMed"; diff --git a/SleepLib/day.cpp b/SleepLib/day.cpp index 034593bf..adc9b4b1 100644 --- a/SleepLib/day.cpp +++ b/SleepLib/day.cpp @@ -6,6 +6,7 @@ #include "day.h" #include "profiles.h" +#include #include Day::Day(Machine *m) @@ -149,60 +150,110 @@ EventDataType Day::settings_wavg(ChannelID code) } - - - - EventDataType Day::percentile(ChannelID code,EventDataType percentile) { - // Cache this calculation - //if (percentile>=1) return 0; // probably better to crash and burn. +// QHash >::iterator pi; +// pi=perc_cache.find(code); +// if (pi!=perc_cache.end()) { +// QHash & hsh=pi.value(); +// QHash::iterator hi=hsh.find( +// if (hi!=pi.value().end()) { +// return hi.value(); +// } +// } + // Cache this calculation? QVector::iterator s; - // Don't assume sessions are in order. - QVector ar; + QMap wmap; + + int SN=0; + + // First Calculate count of all events for (s=sessions.begin();s!=sessions.end();s++) { if (!(*s)->enabled()) continue; Session & sess=*(*s); QHash > ::iterator ei=sess.m_valuesummary.find(code); if (ei==sess.m_valuesummary.end()) continue; + EventDataType gain=sess.m_gain[code]; + EventDataType weight,value; for (QHash::iterator i=ei.value().begin();i!=ei.value().end();i++) { - for (int j=0;j::iterator first=ar.begin(); - QVector::iterator last=ar.end(); - QVector::iterator middle = first + int((last-first) * percentile); - std::nth_element(first,middle,last); - EventDataType val=*middle; + QVector valcnt; -// qSort(ar); -// int p=EventDataType(size)*percentile; -// float p2=EventDataType(size)*percentile; -// float diff=p2-p; -// EventDataType val=ar[p]; -// if (diff>0) { -// int s=p+1; -// if (s>size-1) s=size-1; -// EventDataType v2=ar[s]; -// EventDataType v3=v2-val; -// if (v3>0) { -// val+=v3*diff; -// } + // Build sorted list of value/counts + for (QMap::iterator n=wmap.begin();n!=wmap.end();n++) { + ValueCount vc; + vc.value=n.key(); + vc.count=n.value(); + vc.p=0; + valcnt.push_back(vc); + } + // sort by weight, then value + qSort(valcnt); + + //double SN=100.0/double(N); // 100% / overall sum + double p=100.0*percentile; + + double nth=double(SN)*percentile; // index of the position in the unweighted set would be + double nthi=floor(nth); + + int sum1=0,sum2=0; + int w1,w2=0; + EventDataType v1,v2; + + int N=valcnt.size(); + int k=0; + + for (k=0;k < N;k++) { + v1=valcnt[k].value; + w1=valcnt[k].count; + sum1+=w1; + + if (sum1 > nthi) { + return v1; + } + if (sum1 == nthi){ + break; // boundary condition + } + } + if (k>=N) + return v1; + + v2=valcnt[k+1].value; + w2=valcnt[k+1].count; + sum2=sum1+w2; + // value lies between v1 and v2 + + double px=100.0/double(SN); // Percentile represented by one full value + + // calculate percentile ranks + double p1=px * (double(sum1)-(double(w1)/2.0)); + double p2=px * (double(sum2)-(double(w2)/2.0)); + + // calculate linear interpolation + double v=v1 + ((p-p1)/(p2-p1)) * (v2-v1); + + return v; + + +// p1.....p.............p2 +// 37 55 70 -// } - return val; } EventDataType Day::p90(ChannelID code) diff --git a/SleepLib/day.h b/SleepLib/day.h index 62402df0..99a69d47 100644 --- a/SleepLib/day.h +++ b/SleepLib/day.h @@ -155,6 +155,7 @@ public: protected: //! \brief A Vector containing all sessions for this day QVector sessions; + QHash > perc_cache; qint64 d_first,d_last; private: bool d_firstsession; diff --git a/SleepLib/event.cpp b/SleepLib/event.cpp index 11e4f477..5f985086 100644 --- a/SleepLib/event.cpp +++ b/SleepLib/event.cpp @@ -22,6 +22,8 @@ EventList::EventList(EventListType et,EventDataType gain, EventDataType offset, m_update_minmax=false; } + m_data.reserve(2048); + // Reserve a few to increase performace?? } EventList::~EventList() @@ -112,17 +114,33 @@ void EventList::AddWaveform(qint64 start, qint16 * data, int recs, qint64 durati //double rate=duration/recs; //realloc buffers. + int r=m_count; m_count+=recs; - m_data.reserve(m_count); + m_data.resize(m_count); + EventStoreType *edata=m_data.data(); + + EventStoreType raw; EventDataType val; - for (int i=0;ival) m_min=val; if (m_maxval) m_min=val; if (m_maxval) m_min=val; if (m_max & getTime() { return m_time; } + + // Don't mess with these without considering the consequences + void rawDataResize(quint32 i) { m_data.resize(i); m_count=i; } + void rawData2Resize(quint32 i) { m_data2.resize(i); m_count=i; } + void rawTimeResize(quint32 i) { m_time.resize(i); m_count=i; } + EventStoreType * rawData() { return m_data.data(); } + EventStoreType * rawData2() { return m_data2.data(); } + quint32 * rawTime() { return m_time.data(); } protected: //! \brief The time storage vector, in 32bits delta format, added as offsets to m_first diff --git a/SleepLib/loader_plugins/cms50_loader.cpp b/SleepLib/loader_plugins/cms50_loader.cpp index 925ddb47..8e1c1a5b 100644 --- a/SleepLib/loader_plugins/cms50_loader.cpp +++ b/SleepLib/loader_plugins/cms50_loader.cpp @@ -324,11 +324,11 @@ Machine *CMS50Loader::CreateMachine(Profile *profile) m->SetClass(cms50_class_name); m->properties[STR_PROP_Brand]="Contec"; m->properties[STR_PROP_Model]="CMS50X"; - QString a; - a.sprintf("%i",cms50_data_version); - m->properties[STR_PROP_DataVersion]=a; + m->properties[STR_PROP_DataVersion]=QString::number(cms50_data_version); + profile->AddMachine(m); - m->properties[STR_PROP_Path]="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+m->hexid()+"/"; + QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+m->hexid()+"/"; + m->properties[STR_PROP_Path]=path; return m; } diff --git a/SleepLib/loader_plugins/intellipap_loader.cpp b/SleepLib/loader_plugins/intellipap_loader.cpp index 69598448..76e7bc0c 100644 --- a/SleepLib/loader_plugins/intellipap_loader.cpp +++ b/SleepLib/loader_plugins/intellipap_loader.cpp @@ -399,7 +399,11 @@ Machine *IntellipapLoader::CreateMachine(QString serial,Profile *profile) profile->AddMachine(m); m->properties[STR_PROP_Serial]=serial; - m->properties[STR_PROP_Path]="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/"; + m->properties[STR_PROP_DataVersion]=QString::number(intellipap_data_version); + + QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/"; + m->properties[STR_PROP_Path]=path; + m->properties[STR_PROP_BackupPath]=path+"Backup/"; return m; } diff --git a/SleepLib/loader_plugins/prs1_loader.cpp b/SleepLib/loader_plugins/prs1_loader.cpp index a77cf1d9..20b6b267 100644 --- a/SleepLib/loader_plugins/prs1_loader.cpp +++ b/SleepLib/loader_plugins/prs1_loader.cpp @@ -152,7 +152,11 @@ Machine *PRS1Loader::CreateMachine(QString serial,Profile *profile) profile->AddMachine(m); m->properties[STR_PROP_Serial]=serial; - m->properties[STR_PROP_Path]="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/";; + m->properties[STR_PROP_DataVersion]=QString::number(prs1_data_version); + + QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/"; + m->properties[STR_PROP_Path]=path; + m->properties[STR_PROP_BackupPath]=path+"Backup/"; return m; } @@ -419,7 +423,7 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile) } - m->properties[STR_PROP_DataVersion]=QString().sprintf("%i",prs1_data_version); + m->properties[STR_PROP_DataVersion]=QString::number(prs1_data_version); m->properties[STR_PROP_LastImported]=QDateTime::currentDateTime().toString(Qt::ISODate); m->Save(); // Save any new sessions to disk in our format if (qprogress) qprogress->setValue(100); diff --git a/SleepLib/loader_plugins/resmed_loader.cpp b/SleepLib/loader_plugins/resmed_loader.cpp index 1ea738bc..fa159732 100644 --- a/SleepLib/loader_plugins/resmed_loader.cpp +++ b/SleepLib/loader_plugins/resmed_loader.cpp @@ -198,21 +198,45 @@ bool EDFParser::Parse() } bool EDFParser::Open(QString name) { - QFile f(name); - if (!f.open(QIODevice::ReadOnly)) return false; - if (!f.isReadable()) return false; - filename=name; - filesize=f.size(); - datasize=filesize-EDFHeaderSize; - if (datasize<0) return false; //Urk.. This needs fixing for VC++, as it doesn't have packed attribute type.. - f.read((char *)&header,EDFHeaderSize); - //qDebug() << "Opening " << name; - buffer=new char [datasize]; - f.read(buffer,datasize); - f.close(); + if (name.endsWith(".gz")) { + filename=name.mid(0,-3); + QFile fi(name); + fi.open(QFile::ReadOnly); + fi.seek(fi.size()-4); + unsigned char ch[4]; + fi.read((char *)ch,4); + filesize=ch[0] | (ch [1] << 8) | (ch[2] << 16) | (ch[3] << 24); + datasize=filesize-EDFHeaderSize; + if (datasize<0) return false; + qDebug() << "Size of" << name << "uncompressed=" << filesize; + gzFile f=gzopen(name.toAscii(),"rb"); + if (!f) { + qDebug() << "EDFParser::Open() Couldn't open file" << name; + return false; + } + gzread(f,(char *)&header,EDFHeaderSize); + buffer=new char [datasize]; + gzbuffer(f,65536*2); + gzread(f,buffer,datasize); + gzclose(f); + } else { + QFile f(name); + if (!f.open(QIODevice::ReadOnly)) + return false; + filename=name; + filesize=f.size(); + datasize=filesize-EDFHeaderSize; + if (datasize<0) return false; + + f.read((char *)&header,EDFHeaderSize); + //qDebug() << "Opening " << name; + buffer=new char [datasize]; + f.read(buffer,datasize); + f.close(); + } pos=0; return true; } @@ -248,10 +272,11 @@ Machine *ResmedLoader::CreateMachine(QString serial,Profile *profile) m->properties[STR_PROP_Serial]=serial; m->properties[STR_PROP_Brand]=STR_MACH_ResMed; - QString a; - a.sprintf("%i",resmed_data_version); - m->properties[STR_PROP_DataVersion]=a; - m->properties[STR_PROP_Path]="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/"; + m->properties[STR_PROP_DataVersion]=QString::number(resmed_data_version); + + QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/"; + m->properties[STR_PROP_Path]=path; + m->properties[STR_PROP_BackupPath]=path+"Backup/"; return m; @@ -268,6 +293,7 @@ int ResmedLoader::Open(QString & path,Profile *profile) const QString ext_TGT="tgt"; const QString ext_CRC="crc"; const QString ext_EDF="edf"; + const QString ext_gz=".gz"; QString serial; // Serial number QString key,value; @@ -330,9 +356,13 @@ int ResmedLoader::Open(QString & path,Profile *profile) // Early check for STR.edf file, so we can early exit before creating faulty machine record. QString strpath=path+strfile+ext_EDF; // STR.edf file f.setFileName(strpath); - if (!f.exists()) { - qDebug() << "Missing STR.edf file"; - return 0; + if (!f.exists()) { // No STR.edf.. Do we have a STR.edf.gz? + strpath+=ext_gz; + f.setFileName(strpath); + if (!f.exists()) { + qDebug() << "Missing STR.edf file"; + return 0; + } } /////////////////////////////////////////////////////////////////////////////////// @@ -340,6 +370,17 @@ int ResmedLoader::Open(QString & path,Profile *profile) /////////////////////////////////////////////////////////////////////////////////// Machine *m=CreateMachine(serial,profile); + bool create_backups=PROFILE.session->backupCardData(); + bool compress_backups=PROFILE.session->compressBackupData(); + + QString backup_path=PROFILE.Get(m->properties[STR_PROP_BackupPath]); + if (backup_path.isEmpty()) + backup_path=PROFILE.Get(m->properties[STR_PROP_Path])+"Backup/"; + + if (path==backup_path) { + create_backups=false; + } + /////////////////////////////////////////////////////////////////////////////////// // Parse the idmap into machine objects properties, (overwriting any old values) /////////////////////////////////////////////////////////////////////////////////// @@ -369,24 +410,36 @@ int ResmedLoader::Open(QString & path,Profile *profile) // Creating early as we need the object QDir dir(newpath); + /////////////////////////////////////////////////////////////////////////////////// // Create the backup folder for storing a copy of everything in.. + // (Unless we are importing from this backup folder) /////////////////////////////////////////////////////////////////////////////////// - QString backup_path=PROFILE.Get(m->properties[STR_PROP_Path])+"Backup/"; - if (!dir.exists(backup_path)) { - if (!dir.mkpath(backup_path+datalog)) { - qDebug() << "Could not create S9 backup directory :-/"; + if (create_backups) { + if (!dir.exists(backup_path)) { + if (!dir.mkpath(backup_path+datalog)) { + qDebug() << "Could not create S9 backup directory :-/"; + } } + + // Copy Identification files to backup folder + QFile::copy(path+idfile+ext_TGT,backup_path+idfile+ext_TGT); + QFile::copy(path+idfile+ext_CRC,backup_path+idfile+ext_CRC); + + + //copy STR files to backup folder + if (strpath.endsWith(ext_gz)) // Already compressed. + QFile::copy(strpath,backup_path+strfile+ext_EDF+ext_gz); + else { // Compress STR file to backup folder + compress_backups ? + compressFile(strpath,backup_path+strfile+ext_EDF) + : + QFile::copy(strpath,backup_path+strfile+ext_EDF); + } + + QFile::copy(path+"STR.crc",backup_path+"STR.crc"); } - // Copy Identification files to backup folder - QFile::copy(path+idfile+ext_TGT,backup_path+idfile+ext_TGT); - QFile::copy(path+idfile+ext_CRC,backup_path+idfile+ext_CRC); - - // Copy STR files to backup folder - QFile::copy(strpath,backup_path+strfile+ext_EDF); - QFile::copy(path+strfile+ext_CRC,backup_path+strfile+ext_CRC); - /////////////////////////////////////////////////////////////////////////////////// // Process the actual STR.edf data /////////////////////////////////////////////////////////////////////////////////// @@ -475,14 +528,14 @@ int ResmedLoader::Open(QString & path,Profile *profile) dir.setSorting(QDir::Name); QFileInfoList flist=dir.entryInfoList(); - QString ext,rest,datestr;//,s,codestr; + QString datestr; SessionID sessionid; QDateTime date; int size=flist.size(); sessfiles.clear(); - // For each file in filelist... + // For each file in flist... for (int i=0;isetValue((float(i+1)/float(size)*10.0)); @@ -536,6 +593,7 @@ int ResmedLoader::Open(QString & path,Profile *profile) ///////////////////////////////////////////////////////////////////////////// // Scan over file list and knock out of dayused list ///////////////////////////////////////////////////////////////////////////// + int dn; for (QMap >::iterator si=sessfiles.begin();si!=sessfiles.end();si++) { sessionid=si.key(); @@ -544,17 +602,18 @@ int ResmedLoader::Open(QString & path,Profile *profile) if (edn<0) edn=0; // Find real day number from str.edf mask on/off data. - int dn=-1; + dn=-1; for (int j=edn;j=st) { + et=strlast.at(j); if (sessionid<(et+300)) { dn=j; break; } } } + // If found, mark day off so STR.edf summary data isn't used instead of the real thing. if (dn>=0) { dayused[dn]=0; } @@ -565,16 +624,21 @@ int ResmedLoader::Open(QString & path,Profile *profile) ///////////////////////////////////////////////////////////////////////////// // For all days not in session lists, (to get at days without data records) ///////////////////////////////////////////////////////////////////////////// - for (int dn=0;dnreally_set_first(qint64(st)*1000L); sess->really_set_last(qint64(et)*1000L); sess->SetChanged(true); m->AddSession(sess,profile); - } // Add the actual data to the last session EventDataType tmp,dur; @@ -693,12 +756,14 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->settings[CPAP_PressureMax]=pressure; //sess->setMax(CPAP_Pressure,pressure); } - } } EventDataType valmed=0,valmax=0,val95=0; + ///////////////////////////////////////////////////////////////////// + // Leak Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("Leak Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_Leak]=sig->gain*60.0; @@ -714,6 +779,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->m_valuesummary[CPAP_Leak][valmax]=4; } + ///////////////////////////////////////////////////////////////////// + // Minute Ventilation Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("Min Vent Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_MinuteVent]=sig->gain; @@ -728,6 +796,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->setMax(CPAP_MinuteVent,valmax*sig->gain); sess->m_valuesummary[CPAP_MinuteVent][valmax]=4; } + ///////////////////////////////////////////////////////////////////// + // Respiratory Rate Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("RR Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_RespRate]=sig->gain; @@ -743,6 +814,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->m_valuesummary[CPAP_RespRate][valmax]=4; } + ///////////////////////////////////////////////////////////////////// + // Tidal Volume Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("Tid Vol Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_TidalVolume]=sig->gain*1000.0; @@ -758,6 +832,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->m_valuesummary[CPAP_TidalVolume][valmax]=4; } + ///////////////////////////////////////////////////////////////////// + // Target Minute Ventilation Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("Targ Vent Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_TgMV]=sig->gain; @@ -774,6 +851,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) } + ///////////////////////////////////////////////////////////////////// + // I:E Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("I:E Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_IE]=sig->gain; @@ -789,6 +869,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->m_valuesummary[CPAP_IE][valmax]=4; } + ///////////////////////////////////////////////////////////////////// + // Mask Pressure Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("Mask Pres Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_Pressure]=sig->gain; @@ -804,6 +887,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->m_valuesummary[CPAP_Pressure][valmax]=4; } + ///////////////////////////////////////////////////////////////////// + // Inspiratory Pressure (IPAP) Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("Insp Pres Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_IPAP]=sig->gain; @@ -818,6 +904,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->setMax(CPAP_IPAP,valmax*sig->gain); sess->m_valuesummary[CPAP_IPAP][valmax]=4; } + ///////////////////////////////////////////////////////////////////// + // Expiratory Pressure (EPAP) Summary + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("Exp Pres Med"))) { valmed=sig->data[dn]; sess->m_gain[CPAP_EPAP]=sig->gain; @@ -833,33 +922,39 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->m_valuesummary[CPAP_EPAP][valmax]=4; } + ///////////////////////////////////////////////////////////////////// + // Duration and Event Indices + ///////////////////////////////////////////////////////////////////// if ((sig=stredf.lookupName("Mask Dur"))) { dur=sig->data[dn]*sig->gain; + dur/=60.0f; // convert to hours. } - if ((sig=stredf.lookupName("OAI"))) { + if ((sig=stredf.lookupName("OAI"))) { // Obstructive Apnea Index tmp=sig->data[dn]*sig->gain; sess->setCph(CPAP_Obstructive,tmp); - sess->setCount(CPAP_Obstructive,tmp*(dur/60.0)); + sess->setCount(CPAP_Obstructive,tmp*dur); // Converting from indice to counts.. } - if ((sig=stredf.lookupName("HI"))) { + if ((sig=stredf.lookupName("HI"))) { // Hypopnea Index tmp=sig->data[dn]*sig->gain; sess->setCph(CPAP_Hypopnea,tmp); - sess->setCount(CPAP_Hypopnea,tmp*(dur/60.0)); + sess->setCount(CPAP_Hypopnea,tmp*dur); } - if ((sig=stredf.lookupName("UAI"))) { + if ((sig=stredf.lookupName("UAI"))) { // Unspecified Apnea Index tmp=sig->data[dn]*sig->gain; sess->setCph(CPAP_Apnea,tmp); - sess->setCount(CPAP_Apnea,tmp*(dur/60.0)); + sess->setCount(CPAP_Apnea,tmp*dur); } - if ((sig=stredf.lookupName("CAI"))) { + if ((sig=stredf.lookupName("CAI"))) { // "Central" Apnea Index tmp=sig->data[dn]*sig->gain; sess->setCph(CPAP_ClearAirway,tmp); - sess->setCount(CPAP_ClearAirway,tmp*(dur/60.0)); + sess->setCount(CPAP_ClearAirway,tmp*dur); } } } + bool gz; + backup_path+=datalog+"/"; ///////////////////////////////////////////////////////////////////////////// // Scan through new file list and import sessions @@ -873,18 +968,31 @@ int ResmedLoader::Open(QString & path,Profile *profile) // Create the session sess=new Session(m,sessionid); + QString filename,fullpath,backupfile,backfile, crcfile; // Process EDF File List for (int i=0;isetValue(10.0+(float(++cnt)/float(size)*90.0)); QApplication::processEvents(); diff --git a/SleepLib/machine_common.h b/SleepLib/machine_common.h index 9c690dea..dfd3ccf9 100644 --- a/SleepLib/machine_common.h +++ b/SleepLib/machine_common.h @@ -28,6 +28,7 @@ class BoundsError {}; class OldDBVersion {}; const quint32 magic=0xC73216AB; // Magic number for Sleepyhead Data Files.. Don't touch! +const quint32 compmagic=0xC73216AC; // Magic number for Compressed Sleepyhead Data Files.. Don't touch! //const int max_number_event_fields=10; // This should probably move somewhere else diff --git a/SleepLib/machine_loader.cpp b/SleepLib/machine_loader.cpp index aee0a447..ca9b4fca 100644 --- a/SleepLib/machine_loader.cpp +++ b/SleepLib/machine_loader.cpp @@ -44,6 +44,45 @@ MachineLoader::~MachineLoader() delete *m; } } + +bool MachineLoader::compressFile(QString inpath, QString outpath) +{ + if (outpath.isEmpty()) + outpath=inpath+".gz"; + else if (!outpath.endsWith(".gz")) { + outpath+=".gz"; + } + + QFile f(inpath); + if (!f.exists(inpath)) { + qDebug() << "compressFile()" << inpath << "does not exist"; + return false; + } + qint64 size=f.size(); + if (!f.open(QFile::ReadOnly)) { + qDebug() << "compressFile() Couldn't open" << inpath; + return false; + } + char * buf=new char [size]; + if (!f.read(buf,size)) { + delete buf; + qDebug() << "compressFile() Couldn't read all of" << inpath; + return false; + } + f.close(); + gzFile gz=gzopen(outpath.toAscii(),"wb"); + gzbuffer(gz,65536*2); + if (!gz) { + qDebug() << "compressFile() Couldn't open" << outpath <<"for writing"; + delete buf; + return false; + } + gzwrite(gz,buf,size); + gzclose(gz); + delete buf; + return true; +} + /*const QString machine_profile_name="MachineList.xml"; void MachineLoader::LoadMachineList() diff --git a/SleepLib/machine_loader.h b/SleepLib/machine_loader.h index 564e68c4..17293984 100644 --- a/SleepLib/machine_loader.h +++ b/SleepLib/machine_loader.h @@ -11,6 +11,7 @@ License: GPL #define MACHINE_LOADER_H #include "profiles.h" #include "machine.h" +#include "zlib.h" /*! \class MachineLoader \brief Base class to derive a new Machine importer from @@ -33,6 +34,8 @@ public: //! \brief Override to returns the class name of this MachineLoader virtual const QString & ClassName()=0; + bool compressFile(QString inpath, QString outpath=""); + /* MachineLoader(Profile *profile,QString & classname); diff --git a/SleepLib/profiles.cpp b/SleepLib/profiles.cpp index 5b5ecf77..d387e3b0 100644 --- a/SleepLib/profiles.cpp +++ b/SleepLib/profiles.cpp @@ -13,6 +13,7 @@ License: GPL #include #include #include +#include #include "preferences.h" #include "profiles.h" @@ -27,6 +28,8 @@ Preferences *p_pref; Preferences *p_layout; Profile * p_profile; + + Profile::Profile() :Preferences(),is_first_day(true) { @@ -428,7 +431,7 @@ Profile *Create(QString name) prof->user->setUserName(name); //prof->Set("Realname",realname); //if (!password.isEmpty()) prof.user->setPassword(password); - prof->Set(STR_GEN_DataFolder,QString("{home}/Profiles/{")+QString(UI_STR_UserName)+QString("}")); + prof->Set(STR_GEN_DataFolder,QString("{home}/Profiles/{")+QString(STR_UI_UserName)+QString("}")); Machine *m=new Machine(prof,0); m->SetClass("Journal"); @@ -504,80 +507,6 @@ void Scan() } // namespace Profiles -// DoctorInfo Strings -const char * DI_STR_Name="DoctorName"; -const char * DI_STR_Phone="DoctorPhone"; -const char * DI_STR_Email="DoctorEmail"; -const char * DI_STR_Practice="DoctorPractice"; -const char * DI_STR_Address="DoctorAddress"; -const char * DI_STR_PatientID="DoctorPatientID"; - -// UserInfo Strings -const char * UI_STR_DOB="DOB"; -const char * UI_STR_FirstName="FirstName"; -const char * UI_STR_LastName="LastName"; -const char * UI_STR_UserName="UserName"; -const char * UI_STR_Password="Password"; -const char * UI_STR_Address="Address"; -const char * UI_STR_Phone="Phone"; -const char * UI_STR_EmailAddress="EmailAddress"; -const char * UI_STR_Country="Country"; -const char * UI_STR_Height="Height"; -const char * UI_STR_Gender="Gender"; -const char * UI_STR_TimeZone="TimeZone"; -const char * UI_STR_Language="Language"; -const char * UI_STR_DST="DST"; - -// OxiSettings Strings -const char * OS_STR_EnableOximetry="EnableOximetry"; -const char * OS_STR_SyncOximetry="SyncOximetry"; -const char * OS_STR_OximeterType="OximeterType"; -const char * OS_STR_OxiDiscardThreshold="OxiDiscardThreshold"; -const char * OS_STR_SPO2DropDuration="SPO2DropDuration"; -const char * OS_STR_SPO2DropPercentage="SPO2DropPercentage"; -const char * OS_STR_PulseChangeDuration="PulseChangeDuration"; -const char * OS_STR_PulseChangeBPM="PulseChangeBPM"; - -// CPAPSettings Strings -const char * CS_STR_ComplianceHours="ComplianceHours"; -const char * CS_STR_ShowCompliance="ShowCompliance"; -const char * CS_STR_ShowLeaksMode="ShowLeaksMode"; -const char * CS_STR_MaskStartDate="MaskStartDate"; -const char * CS_STR_MaskDescription="MaskDescription"; -const char * CS_STR_MaskType="MaskType"; -const char * CS_STR_PrescribedMode="CPAPPrescribedMode"; -const char * CS_STR_PrescribedMinPressure="CPAPPrescribedMinPressure"; -const char * CS_STR_PrescribedMaxPressure="CPAPPrescribedMaxPressure"; -const char * CS_STR_UntreatedAHI="UntreatedAHI"; -const char * CS_STR_Notes="CPAPNotes"; -const char * CS_STR_DateDiagnosed="DateDiagnosed"; - -// ImportSettings Strings -const char * IS_STR_DaySplitTime="DaySplitTime"; -const char * IS_STR_CacheSessions="MemoryHog"; -const char * IS_STR_CombineCloseSessions="CombineCloserSessions"; -const char * IS_STR_IgnoreShorterSessions="IgnoreShorterSessions"; -const char * IS_STR_Multithreading="EnableMultithreading"; -const char * IS_STR_TrashDayCache="TrashDayCache"; -const char * IS_STR_ShowSerialNumbers="ShowSerialNumbers"; - -// AppearanceSettings Strings -const char * AS_STR_GraphHeight="GraphHeight"; -const char * AS_STR_AntiAliasing="UseAntiAliasing"; -const char * AS_STR_GraphSnapshots="EnableGraphSnapshots"; -const char * AS_STR_Animations="AnimationsAndTransitions"; -const char * AS_STR_SquareWave="SquareWavePlots"; -const char * AS_STR_OverlayType="OverlayType"; - -// UserSettings Strings -const char * US_STR_UnitSystem="UnitSystem"; -const char * US_STR_EventWindowSize="EventWindowSize"; -const char * US_STR_SkipEmptyDays="SkipEmptyDays"; -const char * US_STR_RebuildCache="RebuildCache"; -const char * US_STR_ShowDebug="ShowDebug"; -const char * US_STR_LinkGroups="LinkGroups"; -const char * US_STR_CalculateRDI="CalculateRDI"; - int Profile::countDays(MachineType mt, QDate start, QDate end) { if (!start.isValid()) @@ -778,17 +707,14 @@ EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, Mac QDate date=start; - // This is one messy function.. It requires all data to be loaded.. :( - - QHash summary; - QHash::iterator sumi; - QVector array; + QMap wmap; QHash >::iterator vsi; EventDataType val,gain; bool setgain=false; + EventDataType weight,value; - //double val=0,tmp,hours=0; + int SN=0; do { Day * day=GetGoodDay(date,mt); if (day) { @@ -800,81 +726,86 @@ EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, Mac Session *sess=*s; gain=sess->m_gain[code]; if (!gain) gain=1; - setgain=true; vsi=sess->m_valuesummary.find(code); if (vsi==sess->m_valuesummary.end()) continue; QHash & vsum=vsi.value(); for (QHash::iterator k=vsum.begin();k!=vsum.end();k++) { - for (int z=0;z >::iterator el=(*s)->eventlist.find(code); - if (el==(*s)->eventlist.end()) continue; - for (int j=0;j::iterator k=summary.begin();k!=summary.end();k++) { -// for (int i=0;i valcnt; - /*if (array.size()==0) return 0; - qSort(array); + // Build sorted list of value/counts + for (QMap::iterator n=wmap.begin();n!=wmap.end();n++) { + ValueCount vc; + vc.value=n.key(); + vc.count=n.value(); + vc.p=0; + valcnt.push_back(vc); + } + // sort by weight, then value + qSort(valcnt); + //double SN=100.0/double(N); // 100% / overall sum + double p=100.0*percent; - int idx=array.size()*percent; - if (idx>array.size()-1) idx=array.size()-1; - return array[idx]; */ + double nth=double(SN)*percent; // index of the position in the unweighted set would be + double nthi=floor(nth); - QVector::iterator first=array.begin(); - QVector::iterator last=array.end(); - QVector::iterator middle = first + int((last-first) * percent); - std::nth_element(first,middle,last); - val=*middle; + int sum1=0,sum2=0; + int w1,w2=0; + EventDataType v1,v2; - return val; -// int size=array.size(); -// if (!size) -// return 0; -// size--; -// qSort(array); -// int p=EventDataType(size)*percent; -// float p2=EventDataType(size)*percent; -// float diff=p2-p; -// val=array[p]; -// if (diff>0) { -// int s=p+1; -// if (s>size-1) s=size-1; -// EventDataType v2=array[s]; -// EventDataType v3=v2-val; -// if (v3>0) { -// val+=v3*diff; -// } + int N=valcnt.size(); + int k=0; -// } + for (k=0;k < N;k++) { + v1=valcnt[k].value; + w1=valcnt[k].count; + sum1+=w1; -// return val; + if (sum1 > nthi) { + return v1; + } + if (sum1 == nthi){ + break; // boundary condition + } + } + if (k>=N) + return v1; + v2=valcnt[k+1].value; + w2=valcnt[k+1].count; + sum2=sum1+w2; + // value lies between v1 and v2 + + double px=100.0/double(SN); // Percentile represented by one full value + + // calculate percentile ranks + double p1=px * (double(sum1)-(double(w1)/2.0)); + double p2=px * (double(sum2)-(double(w2)/2.0)); + + // calculate linear interpolation + double v=v1 + ((p-p1)/(p2-p1)) * (v2-v1); + + // p1.....p.............p2 + // 37 55 70 + + return v; } QDate Profile::FirstDay(MachineType mt) @@ -929,3 +860,81 @@ QDate Profile::LastGoodDay(MachineType mt) } while (d>=f); return f; //m_first; } + + +// DoctorInfo Strings +const char * STR_DI_Name="DoctorName"; +const char * STR_DI_Phone="DoctorPhone"; +const char * STR_DI_Email="DoctorEmail"; +const char * STR_DI_Practice="DoctorPractice"; +const char * STR_DI_Address="DoctorAddress"; +const char * STR_DI_PatientID="DoctorPatientID"; + +// UserInfo Strings +const char * STR_UI_DOB="DOB"; +const char * STR_UI_FirstName="FirstName"; +const char * STR_UI_LastName="LastName"; +const char * STR_UI_UserName="UserName"; +const char * STR_UI_Password="Password"; +const char * STR_UI_Address="Address"; +const char * STR_UI_Phone="Phone"; +const char * STR_UI_EmailAddress="EmailAddress"; +const char * STR_UI_Country="Country"; +const char * STR_UI_Height="Height"; +const char * STR_UI_Gender="Gender"; +const char * STR_UI_TimeZone="TimeZone"; +const char * STR_UI_Language="Language"; +const char * STR_UI_DST="DST"; + +// OxiSettings Strings +const char * STR_OS_EnableOximetry="EnableOximetry"; +const char * STR_OS_SyncOximetry="SyncOximetry"; +const char * STR_OS_OximeterType="OximeterType"; +const char * STR_OS_OxiDiscardThreshold="OxiDiscardThreshold"; +const char * STR_OS_SPO2DropDuration="SPO2DropDuration"; +const char * STR_OS_SPO2DropPercentage="SPO2DropPercentage"; +const char * STR_OS_PulseChangeDuration="PulseChangeDuration"; +const char * STR_OS_PulseChangeBPM="PulseChangeBPM"; + +// CPAPSettings Strings +const char * STR_CS_ComplianceHours="ComplianceHours"; +const char * STR_CS_ShowCompliance="ShowCompliance"; +const char * STR_CS_ShowLeaksMode="ShowLeaksMode"; +const char * STR_CS_MaskStartDate="MaskStartDate"; +const char * STR_CS_MaskDescription="MaskDescription"; +const char * STR_CS_MaskType="MaskType"; +const char * STR_CS_PrescribedMode="CPAPPrescribedMode"; +const char * STR_CS_PrescribedMinPressure="CPAPPrescribedMinPressure"; +const char * STR_CS_PrescribedMaxPressure="CPAPPrescribedMaxPressure"; +const char * STR_CS_UntreatedAHI="UntreatedAHI"; +const char * STR_CS_Notes="CPAPNotes"; +const char * STR_CS_DateDiagnosed="DateDiagnosed"; + +// ImportSettings Strings +const char * STR_IS_DaySplitTime="DaySplitTime"; +const char * STR_IS_CacheSessions="MemoryHog"; +const char * STR_IS_CombineCloseSessions="CombineCloserSessions"; +const char * STR_IS_IgnoreShorterSessions="IgnoreShorterSessions"; +const char * STR_IS_Multithreading="EnableMultithreading"; +const char * STR_IS_BackupCardData="BackupCardData"; +const char * STR_IS_CompressBackupData="CompressBackupData"; +const char * STR_IS_CompressSessionData="CompressSessionData"; + +// AppearanceSettings Strings +const char * STR_AS_GraphHeight="GraphHeight"; +const char * STR_AS_AntiAliasing="UseAntiAliasing"; +const char * STR_AS_GraphSnapshots="EnableGraphSnapshots"; +const char * STR_AS_Animations="AnimationsAndTransitions"; +const char * STR_AS_SquareWave="SquareWavePlots"; +const char * STR_AS_OverlayType="OverlayType"; + +// UserSettings Strings +const char * STR_US_UnitSystem="UnitSystem"; +const char * STR_US_EventWindowSize="EventWindowSize"; +const char * STR_US_SkipEmptyDays="SkipEmptyDays"; +const char * STR_US_RebuildCache="RebuildCache"; +const char * STR_US_ShowDebug="ShowDebug"; +const char * STR_US_LinkGroups="LinkGroups"; +const char * STR_US_CalculateRDI="CalculateRDI"; +const char * STR_US_ShowSerialNumbers="ShowSerialNumbers"; + diff --git a/SleepLib/profiles.h b/SleepLib/profiles.h index 0524eeb9..a4979d73 100644 --- a/SleepLib/profiles.h +++ b/SleepLib/profiles.h @@ -151,59 +151,59 @@ extern Profile * p_profile; #define LAYOUT (*p_layout) #define PROFILE (*p_profile) -extern const char * DI_STR_Name; -extern const char * DI_STR_Phone; -extern const char * DI_STR_Email; -extern const char * DI_STR_Practice; -extern const char * DI_STR_Address; -extern const char * DI_STR_PatientID; +extern const char * STR_DI_Name; +extern const char * STR_DI_Phone; +extern const char * STR_DI_Email; +extern const char * STR_DI_Practice; +extern const char * STR_DI_Address; +extern const char * STR_DI_PatientID; class DoctorInfo { public: DoctorInfo(Profile *p) : m_profile(p) { - if (!m_profile->contains(DI_STR_Name)) (*m_profile)[DI_STR_Name]=QString(); - if (!m_profile->contains(DI_STR_Phone)) (*m_profile)[DI_STR_Phone]=QString(); - if (!m_profile->contains(DI_STR_Email)) (*m_profile)[DI_STR_Email]=QString(); - if (!m_profile->contains(DI_STR_Practice)) (*m_profile)[DI_STR_Practice]=QString(); - if (!m_profile->contains(DI_STR_Address)) (*m_profile)[DI_STR_Address]=QString(); - if (!m_profile->contains(DI_STR_PatientID)) (*m_profile)[DI_STR_PatientID]=QString(); + if (!m_profile->contains(STR_DI_Name)) (*m_profile)[STR_DI_Name]=QString(); + if (!m_profile->contains(STR_DI_Phone)) (*m_profile)[STR_DI_Phone]=QString(); + if (!m_profile->contains(STR_DI_Email)) (*m_profile)[STR_DI_Email]=QString(); + if (!m_profile->contains(STR_DI_Practice)) (*m_profile)[STR_DI_Practice]=QString(); + if (!m_profile->contains(STR_DI_Address)) (*m_profile)[STR_DI_Address]=QString(); + if (!m_profile->contains(STR_DI_PatientID)) (*m_profile)[STR_DI_PatientID]=QString(); } ~DoctorInfo() {} void setProfile(Profile *p) { m_profile=p; } - const QString name() { return (*m_profile)[DI_STR_Name].toString(); } - const QString phone() { return (*m_profile)[DI_STR_Phone].toString(); } - const QString email() { return (*m_profile)[DI_STR_Email].toString(); } - const QString practiceName() { return (*m_profile)[DI_STR_Practice].toString(); } - const QString address() { return (*m_profile)[DI_STR_Address].toString(); } - const QString patientID() { return (*m_profile)[DI_STR_PatientID].toString(); } + const QString name() { return (*m_profile)[STR_DI_Name].toString(); } + const QString phone() { return (*m_profile)[STR_DI_Phone].toString(); } + const QString email() { return (*m_profile)[STR_DI_Email].toString(); } + const QString practiceName() { return (*m_profile)[STR_DI_Practice].toString(); } + const QString address() { return (*m_profile)[STR_DI_Address].toString(); } + const QString patientID() { return (*m_profile)[STR_DI_PatientID].toString(); } - void setName(QString name) { (*m_profile)[DI_STR_Name]=name; } - void setPhone(QString phone) { (*m_profile)[DI_STR_Phone]=phone; } - void setEmail(QString phone) { (*m_profile)[DI_STR_Email]=phone; } - void setPracticeName(QString practice) { (*m_profile)[DI_STR_Practice]=practice; } - void setAddress(QString address) { (*m_profile)[DI_STR_Address]=address; } - void setPatientID(QString pid) { (*m_profile)[DI_STR_PatientID]=pid; } + void setName(QString name) { (*m_profile)[STR_DI_Name]=name; } + void setPhone(QString phone) { (*m_profile)[STR_DI_Phone]=phone; } + void setEmail(QString phone) { (*m_profile)[STR_DI_Email]=phone; } + void setPracticeName(QString practice) { (*m_profile)[STR_DI_Practice]=practice; } + void setAddress(QString address) { (*m_profile)[STR_DI_Address]=address; } + void setPatientID(QString pid) { (*m_profile)[STR_DI_PatientID]=pid; } Profile *m_profile; }; -extern const char * UI_STR_DOB; -extern const char * UI_STR_FirstName; -extern const char * UI_STR_LastName; -extern const char * UI_STR_UserName; -extern const char * UI_STR_Password; -extern const char * UI_STR_Address; -extern const char * UI_STR_Phone; -extern const char * UI_STR_EmailAddress; -extern const char * UI_STR_Country; -extern const char * UI_STR_Height; -extern const char * UI_STR_Gender; -extern const char * UI_STR_TimeZone; -extern const char * UI_STR_Language; -extern const char * UI_STR_DST; +extern const char * STR_UI_DOB; +extern const char * STR_UI_FirstName; +extern const char * STR_UI_LastName; +extern const char * STR_UI_UserName; +extern const char * STR_UI_Password; +extern const char * STR_UI_Address; +extern const char * STR_UI_Phone; +extern const char * STR_UI_EmailAddress; +extern const char * STR_UI_Country; +extern const char * STR_UI_Height; +extern const char * STR_UI_Gender; +extern const char * STR_UI_TimeZone; +extern const char * STR_UI_Language; +extern const char * STR_UI_DST; /*! \class UserInfo \brief Profile Options relating to the User Information @@ -214,78 +214,78 @@ public: //! \brief Create UserInfo object given Profile *p, and initialize the defaults UserInfo(Profile *p) : m_profile(p) { - if (!m_profile->contains(UI_STR_DOB)) (*m_profile)[UI_STR_DOB]=QDate(1970,1,1); - if (!m_profile->contains(UI_STR_FirstName)) (*m_profile)[UI_STR_FirstName]=QString(); - if (!m_profile->contains(UI_STR_LastName)) (*m_profile)[UI_STR_LastName]=QString(); - if (!m_profile->contains(UI_STR_UserName)) (*m_profile)[UI_STR_UserName]=QString(); - if (!m_profile->contains(UI_STR_Password)) (*m_profile)[UI_STR_Password]=QString(); - if (!m_profile->contains(UI_STR_Address)) (*m_profile)[UI_STR_Address]=QString(); - if (!m_profile->contains(UI_STR_Phone)) (*m_profile)[UI_STR_Phone]=QString(); - if (!m_profile->contains(UI_STR_EmailAddress)) (*m_profile)[UI_STR_EmailAddress]=QString(); - if (!m_profile->contains(UI_STR_Country)) (*m_profile)[UI_STR_Country]=QString(); - if (!m_profile->contains(UI_STR_Height)) (*m_profile)[UI_STR_Height]=0.0; - if (!m_profile->contains(UI_STR_Gender)) (*m_profile)[UI_STR_Gender]=(int)GenderNotSpecified; - if (!m_profile->contains(UI_STR_TimeZone)) (*m_profile)[UI_STR_TimeZone]=QString(); - if (!m_profile->contains(UI_STR_Language)) (*m_profile)[UI_STR_Language]="English"; - if (!m_profile->contains(UI_STR_DST)) (*m_profile)[UI_STR_DST]=false; + if (!m_profile->contains(STR_UI_DOB)) (*m_profile)[STR_UI_DOB]=QDate(1970,1,1); + if (!m_profile->contains(STR_UI_FirstName)) (*m_profile)[STR_UI_FirstName]=QString(); + if (!m_profile->contains(STR_UI_LastName)) (*m_profile)[STR_UI_LastName]=QString(); + if (!m_profile->contains(STR_UI_UserName)) (*m_profile)[STR_UI_UserName]=QString(); + if (!m_profile->contains(STR_UI_Password)) (*m_profile)[STR_UI_Password]=QString(); + if (!m_profile->contains(STR_UI_Address)) (*m_profile)[STR_UI_Address]=QString(); + if (!m_profile->contains(STR_UI_Phone)) (*m_profile)[STR_UI_Phone]=QString(); + if (!m_profile->contains(STR_UI_EmailAddress)) (*m_profile)[STR_UI_EmailAddress]=QString(); + if (!m_profile->contains(STR_UI_Country)) (*m_profile)[STR_UI_Country]=QString(); + if (!m_profile->contains(STR_UI_Height)) (*m_profile)[STR_UI_Height]=0.0; + if (!m_profile->contains(STR_UI_Gender)) (*m_profile)[STR_UI_Gender]=(int)GenderNotSpecified; + if (!m_profile->contains(STR_UI_TimeZone)) (*m_profile)[STR_UI_TimeZone]=QString(); + if (!m_profile->contains(STR_UI_Language)) (*m_profile)[STR_UI_Language]="English"; + if (!m_profile->contains(STR_UI_DST)) (*m_profile)[STR_UI_DST]=false; } ~UserInfo() {} void setProfile(Profile *p) { m_profile=p; } - QDate DOB() { return (*m_profile)[UI_STR_DOB].toDate(); } - const QString firstName() { return (*m_profile)[UI_STR_FirstName].toString(); } - const QString lastName() { return (*m_profile)[UI_STR_LastName].toString(); } - const QString userName() { return (*m_profile)[UI_STR_UserName].toString(); } - const QString address() { return (*m_profile)[UI_STR_Address].toString(); } - const QString phone() { return (*m_profile)[UI_STR_Phone].toString(); } - const QString email() { return (*m_profile)[UI_STR_EmailAddress].toString(); } - double height() { return (*m_profile)[UI_STR_Height].toDouble(); } - const QString country() { return (*m_profile)[UI_STR_Country].toString(); } - Gender gender() { return (Gender)(*m_profile)[UI_STR_Gender].toInt(); } - const QString timeZone() { return (*m_profile)[UI_STR_TimeZone].toString(); } - const QString language() { return (*m_profile)[UI_STR_Language].toString(); } - bool daylightSaving() { return (*m_profile)[UI_STR_DST].toBool(); } + QDate DOB() { return (*m_profile)[STR_UI_DOB].toDate(); } + const QString firstName() { return (*m_profile)[STR_UI_FirstName].toString(); } + const QString lastName() { return (*m_profile)[STR_UI_LastName].toString(); } + const QString userName() { return (*m_profile)[STR_UI_UserName].toString(); } + const QString address() { return (*m_profile)[STR_UI_Address].toString(); } + const QString phone() { return (*m_profile)[STR_UI_Phone].toString(); } + const QString email() { return (*m_profile)[STR_UI_EmailAddress].toString(); } + double height() { return (*m_profile)[STR_UI_Height].toDouble(); } + const QString country() { return (*m_profile)[STR_UI_Country].toString(); } + Gender gender() { return (Gender)(*m_profile)[STR_UI_Gender].toInt(); } + const QString timeZone() { return (*m_profile)[STR_UI_TimeZone].toString(); } + const QString language() { return (*m_profile)[STR_UI_Language].toString(); } + bool daylightSaving() { return (*m_profile)[STR_UI_DST].toBool(); } - void setDOB(QDate date) { (*m_profile)[UI_STR_DOB]=date; } - void setFirstName(QString name) { (*m_profile)[UI_STR_FirstName]=name; } - void setLastName(QString name) { (*m_profile)[UI_STR_LastName]=name; } - void setUserName(QString username) { (*m_profile)[UI_STR_UserName]=username; } - void setAddress(QString address) { (*m_profile)[UI_STR_Address]=address; } - void setPhone(QString phone) { (*m_profile)[UI_STR_Phone]=phone; } - void setEmail(QString email) { (*m_profile)[UI_STR_EmailAddress]=email; } - void setHeight(double height) { (*m_profile)[UI_STR_Height]=height; } - void setCountry(QString country) { (*m_profile)[UI_STR_Country]=country; } - void setGender(Gender g) { (*m_profile)[UI_STR_Gender]=(int)g; } - void setTimeZone(QString tz) { (*m_profile)[UI_STR_TimeZone]=tz; } - void setLanguage(QString language) { (*m_profile)[UI_STR_Language]=language; } - void setDaylightSaving(bool ds) { (*m_profile)[UI_STR_DST]=ds; } + void setDOB(QDate date) { (*m_profile)[STR_UI_DOB]=date; } + void setFirstName(QString name) { (*m_profile)[STR_UI_FirstName]=name; } + void setLastName(QString name) { (*m_profile)[STR_UI_LastName]=name; } + void setUserName(QString username) { (*m_profile)[STR_UI_UserName]=username; } + void setAddress(QString address) { (*m_profile)[STR_UI_Address]=address; } + void setPhone(QString phone) { (*m_profile)[STR_UI_Phone]=phone; } + void setEmail(QString email) { (*m_profile)[STR_UI_EmailAddress]=email; } + void setHeight(double height) { (*m_profile)[STR_UI_Height]=height; } + void setCountry(QString country) { (*m_profile)[STR_UI_Country]=country; } + void setGender(Gender g) { (*m_profile)[STR_UI_Gender]=(int)g; } + void setTimeZone(QString tz) { (*m_profile)[STR_UI_TimeZone]=tz; } + void setLanguage(QString language) { (*m_profile)[STR_UI_Language]=language; } + void setDaylightSaving(bool ds) { (*m_profile)[STR_UI_DST]=ds; } bool hasPassword() { - return !((*m_profile)[UI_STR_Password].toString().isEmpty()); + return !((*m_profile)[STR_UI_Password].toString().isEmpty()); } bool checkPassword(QString password) { QByteArray ba=password.toUtf8(); - return ((*m_profile)[UI_STR_Password].toString()==QString(QCryptographicHash::hash(ba,QCryptographicHash::Sha1).toHex())); + return ((*m_profile)[STR_UI_Password].toString()==QString(QCryptographicHash::hash(ba,QCryptographicHash::Sha1).toHex())); } void setPassword(QString password) { QByteArray ba=password.toUtf8(); // Hash me. - (*m_profile)[UI_STR_Password]=QString(QCryptographicHash::hash(ba,QCryptographicHash::Sha1).toHex()); + (*m_profile)[STR_UI_Password]=QString(QCryptographicHash::hash(ba,QCryptographicHash::Sha1).toHex()); } Profile *m_profile; }; -extern const char * OS_STR_EnableOximetry; -extern const char * OS_STR_SyncOximetry; -extern const char * OS_STR_OximeterType; -extern const char * OS_STR_OxiDiscardThreshold; -extern const char * OS_STR_SPO2DropDuration; -extern const char * OS_STR_SPO2DropPercentage; -extern const char * OS_STR_PulseChangeDuration; -extern const char * OS_STR_PulseChangeBPM; +extern const char * STR_OS_EnableOximetry; +extern const char * STR_OS_SyncOximetry; +extern const char * STR_OS_OximeterType; +extern const char * STR_OS_OxiDiscardThreshold; +extern const char * STR_OS_SPO2DropDuration; +extern const char * STR_OS_SPO2DropPercentage; +extern const char * STR_OS_PulseChangeDuration; +extern const char * STR_OS_PulseChangeBPM; /*! \class OxiSettings \brief Profile Options relating to the Oximetry settings @@ -296,52 +296,52 @@ public: //! \brief Create OxiSettings object given Profile *p, and initialize the defaults OxiSettings(Profile *p) :m_profile(p) { - if (!m_profile->contains(OS_STR_EnableOximetry)) (*m_profile)[OS_STR_EnableOximetry]=false; - if (!m_profile->contains(OS_STR_SyncOximetry)) (*m_profile)[OS_STR_SyncOximetry]=true; - if (!m_profile->contains(OS_STR_OximeterType)) (*m_profile)[OS_STR_OximeterType]="CMS50"; - if (!m_profile->contains(OS_STR_OxiDiscardThreshold)) (*m_profile)[OS_STR_OxiDiscardThreshold]=0.0; - if (!m_profile->contains(OS_STR_SPO2DropDuration)) (*m_profile)[OS_STR_SPO2DropDuration]=8.0; - if (!m_profile->contains(OS_STR_SPO2DropPercentage)) (*m_profile)[OS_STR_SPO2DropPercentage]=3.0; - if (!m_profile->contains(OS_STR_PulseChangeDuration)) (*m_profile)[OS_STR_PulseChangeDuration]=8.0; - if (!m_profile->contains(OS_STR_PulseChangeBPM)) (*m_profile)[OS_STR_PulseChangeBPM]=5.0; + if (!m_profile->contains(STR_OS_EnableOximetry)) (*m_profile)[STR_OS_EnableOximetry]=false; + if (!m_profile->contains(STR_OS_SyncOximetry)) (*m_profile)[STR_OS_SyncOximetry]=true; + if (!m_profile->contains(STR_OS_OximeterType)) (*m_profile)[STR_OS_OximeterType]="CMS50"; + if (!m_profile->contains(STR_OS_OxiDiscardThreshold)) (*m_profile)[STR_OS_OxiDiscardThreshold]=0.0; + if (!m_profile->contains(STR_OS_SPO2DropDuration)) (*m_profile)[STR_OS_SPO2DropDuration]=8.0; + if (!m_profile->contains(STR_OS_SPO2DropPercentage)) (*m_profile)[STR_OS_SPO2DropPercentage]=3.0; + if (!m_profile->contains(STR_OS_PulseChangeDuration)) (*m_profile)[STR_OS_PulseChangeDuration]=8.0; + if (!m_profile->contains(STR_OS_PulseChangeBPM)) (*m_profile)[STR_OS_PulseChangeBPM]=5.0; } ~OxiSettings() {} void setProfile(Profile *p) { m_profile=p; } - bool oximetryEnabled() { return (*m_profile)[OS_STR_EnableOximetry].toBool(); } - bool syncOximetry() { return (*m_profile)[OS_STR_SyncOximetry].toBool(); } - QString oximeterType() { return (*m_profile)[OS_STR_OximeterType].toString(); } - double oxiDiscardThreshold() { return (*m_profile)[OS_STR_OxiDiscardThreshold].toDouble(); } - double spO2DropDuration() { return (*m_profile)[OS_STR_SPO2DropDuration].toDouble(); } - double spO2DropPercentage() { return (*m_profile)[OS_STR_SPO2DropPercentage].toDouble(); } - double pulseChangeDuration() { return (*m_profile)[OS_STR_PulseChangeDuration].toDouble(); } - double pulseChangeBPM() { return (*m_profile)[OS_STR_PulseChangeBPM].toDouble(); } + bool oximetryEnabled() { return (*m_profile)[STR_OS_EnableOximetry].toBool(); } + bool syncOximetry() { return (*m_profile)[STR_OS_SyncOximetry].toBool(); } + QString oximeterType() { return (*m_profile)[STR_OS_OximeterType].toString(); } + double oxiDiscardThreshold() { return (*m_profile)[STR_OS_OxiDiscardThreshold].toDouble(); } + double spO2DropDuration() { return (*m_profile)[STR_OS_SPO2DropDuration].toDouble(); } + double spO2DropPercentage() { return (*m_profile)[STR_OS_SPO2DropPercentage].toDouble(); } + double pulseChangeDuration() { return (*m_profile)[STR_OS_PulseChangeDuration].toDouble(); } + double pulseChangeBPM() { return (*m_profile)[STR_OS_PulseChangeBPM].toDouble(); } - void setOximetryEnabled(bool enabled) { (*m_profile)[OS_STR_EnableOximetry]=enabled; } - void setSyncOximetry(bool synced) { (*m_profile)[OS_STR_SyncOximetry]=synced; } - void setOximeterType(QString oxitype) { (*m_profile)[OS_STR_OximeterType]=oxitype; } - void setOxiDiscardThreshold(double thresh) { (*m_profile)[OS_STR_OxiDiscardThreshold]=thresh; } - void setSpO2DropDuration(double duration) { (*m_profile)[OS_STR_SPO2DropDuration]=duration; } - void setSpO2DropPercentage(double percentage) { (*m_profile)[OS_STR_SPO2DropPercentage]=percentage; } - void setPulseChangeDuration(double duration) { (*m_profile)[OS_STR_PulseChangeDuration]=duration; } - void setPulseChangeBPM(double bpm) { (*m_profile)[OS_STR_PulseChangeBPM]=bpm; } + void setOximetryEnabled(bool enabled) { (*m_profile)[STR_OS_EnableOximetry]=enabled; } + void setSyncOximetry(bool synced) { (*m_profile)[STR_OS_SyncOximetry]=synced; } + void setOximeterType(QString oxitype) { (*m_profile)[STR_OS_OximeterType]=oxitype; } + void setOxiDiscardThreshold(double thresh) { (*m_profile)[STR_OS_OxiDiscardThreshold]=thresh; } + void setSpO2DropDuration(double duration) { (*m_profile)[STR_OS_SPO2DropDuration]=duration; } + void setSpO2DropPercentage(double percentage) { (*m_profile)[STR_OS_SPO2DropPercentage]=percentage; } + void setPulseChangeDuration(double duration) { (*m_profile)[STR_OS_PulseChangeDuration]=duration; } + void setPulseChangeBPM(double bpm) { (*m_profile)[STR_OS_PulseChangeBPM]=bpm; } Profile *m_profile; }; -extern const char * CS_STR_ComplianceHours; -extern const char * CS_STR_ShowCompliance; -extern const char * CS_STR_ShowLeaksMode; -extern const char * CS_STR_MaskStartDate; -extern const char * CS_STR_MaskDescription; -extern const char * CS_STR_MaskType; -extern const char * CS_STR_PrescribedMode; -extern const char * CS_STR_PrescribedMinPressure; -extern const char * CS_STR_PrescribedMaxPressure; -extern const char * CS_STR_UntreatedAHI; -extern const char * CS_STR_Notes; -extern const char * CS_STR_DateDiagnosed; +extern const char * STR_CS_ComplianceHours; +extern const char * STR_CS_ShowCompliance; +extern const char * STR_CS_ShowLeaksMode; +extern const char * STR_CS_MaskStartDate; +extern const char * STR_CS_MaskDescription; +extern const char * STR_CS_MaskType; +extern const char * STR_CS_PrescribedMode; +extern const char * STR_CS_PrescribedMinPressure; +extern const char * STR_CS_PrescribedMaxPressure; +extern const char * STR_CS_UntreatedAHI; +extern const char * STR_CS_Notes; +extern const char * STR_CS_DateDiagnosed; /*! \class CPAPSettings \brief Profile Options relating to the CPAP settings @@ -352,19 +352,19 @@ public: //! \brief Create CPAPSettings object given Profile *p, and initialize the defaults CPAPSettings(Profile *p) :m_profile(p) { - if (!m_profile->contains(CS_STR_ComplianceHours)) (*m_profile)[CS_STR_ComplianceHours]=4; - if (!m_profile->contains(CS_STR_ShowCompliance)) (*m_profile)[CS_STR_ShowCompliance]=true; - if (!m_profile->contains(CS_STR_ShowLeaksMode)) (*m_profile)[CS_STR_ShowLeaksMode]=0; + if (!m_profile->contains(STR_CS_ComplianceHours)) (*m_profile)[STR_CS_ComplianceHours]=4; + if (!m_profile->contains(STR_CS_ShowCompliance)) (*m_profile)[STR_CS_ShowCompliance]=true; + if (!m_profile->contains(STR_CS_ShowLeaksMode)) (*m_profile)[STR_CS_ShowLeaksMode]=0; // TODO: Check if this date is initiliazed yet - if (!m_profile->contains(CS_STR_MaskStartDate)) (*m_profile)[CS_STR_MaskStartDate]=QDate(); - if (!m_profile->contains(CS_STR_MaskDescription)) (*m_profile)[CS_STR_MaskDescription]=QString(); - if (!m_profile->contains(CS_STR_MaskType)) (*m_profile)[CS_STR_MaskType]=Mask_Unknown; - if (!m_profile->contains(CS_STR_PrescribedMode)) (*m_profile)[CS_STR_PrescribedMode]=MODE_UNKNOWN; - if (!m_profile->contains(CS_STR_PrescribedMinPressure)) (*m_profile)[CS_STR_PrescribedMinPressure]=0.0; - if (!m_profile->contains(CS_STR_PrescribedMaxPressure)) (*m_profile)[CS_STR_PrescribedMaxPressure]=0.0; - if (!m_profile->contains(CS_STR_UntreatedAHI)) (*m_profile)[CS_STR_UntreatedAHI]=0.0; - if (!m_profile->contains(CS_STR_Notes)) (*m_profile)[CS_STR_Notes]=QString(); - if (!m_profile->contains(CS_STR_DateDiagnosed)) (*m_profile)[CS_STR_DateDiagnosed]=QDate(); + if (!m_profile->contains(STR_CS_MaskStartDate)) (*m_profile)[STR_CS_MaskStartDate]=QDate(); + if (!m_profile->contains(STR_CS_MaskDescription)) (*m_profile)[STR_CS_MaskDescription]=QString(); + if (!m_profile->contains(STR_CS_MaskType)) (*m_profile)[STR_CS_MaskType]=Mask_Unknown; + if (!m_profile->contains(STR_CS_PrescribedMode)) (*m_profile)[STR_CS_PrescribedMode]=MODE_UNKNOWN; + if (!m_profile->contains(STR_CS_PrescribedMinPressure)) (*m_profile)[STR_CS_PrescribedMinPressure]=0.0; + if (!m_profile->contains(STR_CS_PrescribedMaxPressure)) (*m_profile)[STR_CS_PrescribedMaxPressure]=0.0; + if (!m_profile->contains(STR_CS_UntreatedAHI)) (*m_profile)[STR_CS_UntreatedAHI]=0.0; + if (!m_profile->contains(STR_CS_Notes)) (*m_profile)[STR_CS_Notes]=QString(); + if (!m_profile->contains(STR_CS_DateDiagnosed)) (*m_profile)[STR_CS_DateDiagnosed]=QDate(); } ~CPAPSettings() {} @@ -372,43 +372,44 @@ public: void setProfile(Profile *p) { m_profile=p; } //Getters - double complianceHours() { return (*m_profile)[CS_STR_ComplianceHours].toDouble(); } - bool showComplianceInfo() { return (*m_profile)[CS_STR_ShowCompliance].toBool(); } - int leakMode() { return (*m_profile)[CS_STR_ShowLeaksMode].toInt(); } - QDate maskStartDate() { return (*m_profile)[CS_STR_MaskStartDate].toDate(); } - QString maskDescription() { return (*m_profile)[CS_STR_MaskDescription].toString(); } - MaskType maskType() { return (MaskType)(*m_profile)[CS_STR_MaskType].toInt(); } - CPAPMode mode() { return CPAPMode((*m_profile)[CS_STR_PrescribedMode].toInt()); } - double minPressure() { return (*m_profile)[CS_STR_PrescribedMinPressure].toDouble(); } - double maxPressure() { return (*m_profile)[CS_STR_PrescribedMaxPressure].toDouble(); } - double untreatedAHI() { return (*m_profile)[CS_STR_UntreatedAHI].toDouble(); } - const QString notes() { return (*m_profile)[CS_STR_Notes].toString(); } - QDate dateDiagnosed() { return (*m_profile)[CS_STR_DateDiagnosed].toDate(); } + double complianceHours() { return (*m_profile)[STR_CS_ComplianceHours].toDouble(); } + bool showComplianceInfo() { return (*m_profile)[STR_CS_ShowCompliance].toBool(); } + int leakMode() { return (*m_profile)[STR_CS_ShowLeaksMode].toInt(); } + QDate maskStartDate() { return (*m_profile)[STR_CS_MaskStartDate].toDate(); } + QString maskDescription() { return (*m_profile)[STR_CS_MaskDescription].toString(); } + MaskType maskType() { return (MaskType)(*m_profile)[STR_CS_MaskType].toInt(); } + CPAPMode mode() { return CPAPMode((*m_profile)[STR_CS_PrescribedMode].toInt()); } + double minPressure() { return (*m_profile)[STR_CS_PrescribedMinPressure].toDouble(); } + double maxPressure() { return (*m_profile)[STR_CS_PrescribedMaxPressure].toDouble(); } + double untreatedAHI() { return (*m_profile)[STR_CS_UntreatedAHI].toDouble(); } + const QString notes() { return (*m_profile)[STR_CS_Notes].toString(); } + QDate dateDiagnosed() { return (*m_profile)[STR_CS_DateDiagnosed].toDate(); } //Setters - void setMode(CPAPMode mode) { (*m_profile)[CS_STR_PrescribedMode]=(int)mode; } - void setMinPressure(double pressure) { (*m_profile)[CS_STR_PrescribedMinPressure]=pressure; } - void setMaxPressure(double pressure) { (*m_profile)[CS_STR_PrescribedMaxPressure]=pressure; } - void setUntreatedAHI(double ahi) { (*m_profile)[CS_STR_UntreatedAHI]=ahi; } - void setNotes(QString notes) { (*m_profile)[CS_STR_Notes]=notes; } - void setDateDiagnosed(QDate date) { (*m_profile)[CS_STR_DateDiagnosed]=date; } - void setComplianceHours(double hours) { (*m_profile)[CS_STR_ComplianceHours]=hours; } - void setShowComplianceInfo(bool b) { (*m_profile)[CS_STR_ShowCompliance]=b; } - void setLeakMode(int leakmode) { (*m_profile)[CS_STR_ShowLeaksMode]=(int)leakmode; } - void setMaskStartDate(QDate date) { (*m_profile)[CS_STR_MaskStartDate]=date; } - void setMaskDescription(QString description) { (*m_profile)[CS_STR_MaskDescription]=description; } - void setMaskType(MaskType masktype) { (*m_profile)[CS_STR_MaskType]=(int)masktype; } + void setMode(CPAPMode mode) { (*m_profile)[STR_CS_PrescribedMode]=(int)mode; } + void setMinPressure(double pressure) { (*m_profile)[STR_CS_PrescribedMinPressure]=pressure; } + void setMaxPressure(double pressure) { (*m_profile)[STR_CS_PrescribedMaxPressure]=pressure; } + void setUntreatedAHI(double ahi) { (*m_profile)[STR_CS_UntreatedAHI]=ahi; } + void setNotes(QString notes) { (*m_profile)[STR_CS_Notes]=notes; } + void setDateDiagnosed(QDate date) { (*m_profile)[STR_CS_DateDiagnosed]=date; } + void setComplianceHours(double hours) { (*m_profile)[STR_CS_ComplianceHours]=hours; } + void setShowComplianceInfo(bool b) { (*m_profile)[STR_CS_ShowCompliance]=b; } + void setLeakMode(int leakmode) { (*m_profile)[STR_CS_ShowLeaksMode]=(int)leakmode; } + void setMaskStartDate(QDate date) { (*m_profile)[STR_CS_MaskStartDate]=date; } + void setMaskDescription(QString description) { (*m_profile)[STR_CS_MaskDescription]=description; } + void setMaskType(MaskType masktype) { (*m_profile)[STR_CS_MaskType]=(int)masktype; } Profile *m_profile; }; -extern const char * IS_STR_DaySplitTime; -extern const char * IS_STR_CacheSessions; -extern const char * IS_STR_CombineCloseSessions; -extern const char * IS_STR_IgnoreShorterSessions; -extern const char * IS_STR_Multithreading; -extern const char * IS_STR_TrashDayCache; -extern const char * IS_STR_ShowSerialNumbers; +extern const char * STR_IS_DaySplitTime; +extern const char * STR_IS_CacheSessions; +extern const char * STR_IS_CombineCloseSessions; +extern const char * STR_IS_IgnoreShorterSessions; +extern const char * STR_IS_Multithreading; +extern const char * STR_IS_BackupCardData; +extern const char * STR_IS_CompressBackupData; +extern const char * STR_IS_CompressSessionData; /*! \class ImportSettings \brief Profile Options relating to the Import process @@ -419,43 +420,46 @@ public: //! \brief Create ImportSettings object given Profile *p, and initialize the defaults SessionSettings(Profile *p) :m_profile(p) { - if (!m_profile->contains(IS_STR_DaySplitTime)) (*m_profile)[IS_STR_DaySplitTime]=QTime(12,0,0); - if (!m_profile->contains(IS_STR_CacheSessions)) (*m_profile)[IS_STR_CacheSessions]=false; - if (!m_profile->contains(IS_STR_CombineCloseSessions)) (*m_profile)[IS_STR_CombineCloseSessions]=240; - if (!m_profile->contains(IS_STR_IgnoreShorterSessions)) (*m_profile)[IS_STR_IgnoreShorterSessions]=5; - if (!m_profile->contains(IS_STR_Multithreading)) (*m_profile)[IS_STR_Multithreading]=QThread::idealThreadCount() > 1; - if (!m_profile->contains(IS_STR_TrashDayCache)) (*m_profile)[IS_STR_TrashDayCache]=false; // can't remember.. - if (!m_profile->contains(IS_STR_ShowSerialNumbers)) (*m_profile)[IS_STR_ShowSerialNumbers]=false; + if (!m_profile->contains(STR_IS_DaySplitTime)) (*m_profile)[STR_IS_DaySplitTime]=QTime(12,0,0); + if (!m_profile->contains(STR_IS_CacheSessions)) (*m_profile)[STR_IS_CacheSessions]=false; + if (!m_profile->contains(STR_IS_CombineCloseSessions)) (*m_profile)[STR_IS_CombineCloseSessions]=240; + if (!m_profile->contains(STR_IS_IgnoreShorterSessions)) (*m_profile)[STR_IS_IgnoreShorterSessions]=5; + if (!m_profile->contains(STR_IS_Multithreading)) (*m_profile)[STR_IS_Multithreading]=QThread::idealThreadCount() > 1; + if (!m_profile->contains(STR_IS_BackupCardData)) (*m_profile)[STR_IS_BackupCardData]=true; + if (!m_profile->contains(STR_IS_CompressBackupData)) (*m_profile)[STR_IS_CompressBackupData]=false; + if (!m_profile->contains(STR_IS_CompressSessionData)) (*m_profile)[STR_IS_CompressSessionData]=false; } ~SessionSettings() {} void setProfile(Profile *p) { m_profile=p; } - QTime daySplitTime() { return (*m_profile)[IS_STR_DaySplitTime].toTime(); } - bool cacheSessions() { return (*m_profile)[IS_STR_CacheSessions].toBool(); } - double combineCloseSessions() { return (*m_profile)[IS_STR_CombineCloseSessions].toDouble(); } - double ignoreShortSessions() { return (*m_profile)[IS_STR_IgnoreShorterSessions].toDouble(); } - bool multithreading() { return (*m_profile)[IS_STR_Multithreading].toBool(); } - bool trashDayCache() { return (*m_profile)[IS_STR_TrashDayCache].toBool(); } - bool showSerialNumbers() { return (*m_profile)[IS_STR_ShowSerialNumbers].toBool(); } + QTime daySplitTime() { return (*m_profile)[STR_IS_DaySplitTime].toTime(); } + bool cacheSessions() { return (*m_profile)[STR_IS_CacheSessions].toBool(); } + double combineCloseSessions() { return (*m_profile)[STR_IS_CombineCloseSessions].toDouble(); } + double ignoreShortSessions() { return (*m_profile)[STR_IS_IgnoreShorterSessions].toDouble(); } + bool multithreading() { return (*m_profile)[STR_IS_Multithreading].toBool(); } + bool compressSessionData() { return (*m_profile)[STR_IS_CompressSessionData].toBool(); } + bool compressBackupData() { return (*m_profile)[STR_IS_CompressBackupData].toBool(); } + bool backupCardData() { return (*m_profile)[STR_IS_BackupCardData].toBool(); } - void setDaySplitTime(QTime time) { (*m_profile)[IS_STR_DaySplitTime]=time; } - void setCacheSessions(bool c) { (*m_profile)[IS_STR_CacheSessions]=c; } - void setCombineCloseSessions(double val) { (*m_profile)[IS_STR_CombineCloseSessions]=val; } - void setIgnoreShortSessions(double val) { (*m_profile)[IS_STR_IgnoreShorterSessions]=val; } - void setMultithreading(bool enabled) { (*m_profile)[IS_STR_Multithreading]=enabled; } - void setTrashDayCache(bool trash) { (*m_profile)[IS_STR_TrashDayCache]=trash; } - void setShowSerialNumbers(bool trash) { (*m_profile)[IS_STR_ShowSerialNumbers]=trash; } + void setDaySplitTime(QTime time) { (*m_profile)[STR_IS_DaySplitTime]=time; } + void setCacheSessions(bool c) { (*m_profile)[STR_IS_CacheSessions]=c; } + void setCombineCloseSessions(double val) { (*m_profile)[STR_IS_CombineCloseSessions]=val; } + void setIgnoreShortSessions(double val) { (*m_profile)[STR_IS_IgnoreShorterSessions]=val; } + void setMultithreading(bool enabled) { (*m_profile)[STR_IS_Multithreading]=enabled; } + void setBackupCardData(bool enabled) { (*m_profile)[STR_IS_BackupCardData]=enabled; } + void setCompressBackupData(bool enabled) { (*m_profile)[STR_IS_CompressBackupData]=enabled; } + void setCompressSessionData(bool enabled) { (*m_profile)[STR_IS_CompressSessionData]=enabled; } Profile *m_profile; }; -extern const char * AS_STR_GraphHeight; -extern const char * AS_STR_AntiAliasing; -extern const char * AS_STR_GraphSnapshots; -extern const char * AS_STR_Animations; -extern const char * AS_STR_SquareWave; -extern const char * AS_STR_OverlayType; +extern const char * STR_AS_GraphHeight; +extern const char * STR_AS_AntiAliasing; +extern const char * STR_AS_GraphSnapshots; +extern const char * STR_AS_Animations; +extern const char * STR_AS_SquareWave; +extern const char * STR_AS_OverlayType; /*! \class AppearanceSettings \brief Profile Options relating to Visual Appearance @@ -466,54 +470,54 @@ public: //! \brief Create AppearanceSettings object given Profile *p, and initialize the defaults AppearanceSettings(Profile *p) :m_profile(p) { - if (!m_profile->contains(AS_STR_GraphHeight)) (*m_profile)[AS_STR_GraphHeight]=180.0; - if (!m_profile->contains(AS_STR_AntiAliasing)) (*m_profile)[AS_STR_AntiAliasing]=false; // i think it's ugly - if (!m_profile->contains(AS_STR_GraphSnapshots)) (*m_profile)[AS_STR_GraphSnapshots]=true; - if (!m_profile->contains(AS_STR_Animations)) (*m_profile)[AS_STR_Animations]=true; - if (!m_profile->contains(AS_STR_SquareWave)) (*m_profile)[AS_STR_SquareWave]=false; - if (!m_profile->contains(AS_STR_OverlayType)) (*m_profile)[AS_STR_OverlayType]=ODT_Bars; + if (!m_profile->contains(STR_AS_GraphHeight)) (*m_profile)[STR_AS_GraphHeight]=180.0; + if (!m_profile->contains(STR_AS_AntiAliasing)) (*m_profile)[STR_AS_AntiAliasing]=false; // i think it's ugly + if (!m_profile->contains(STR_AS_GraphSnapshots)) (*m_profile)[STR_AS_GraphSnapshots]=true; + if (!m_profile->contains(STR_AS_Animations)) (*m_profile)[STR_AS_Animations]=true; + if (!m_profile->contains(STR_AS_SquareWave)) (*m_profile)[STR_AS_SquareWave]=false; + if (!m_profile->contains(STR_AS_OverlayType)) (*m_profile)[STR_AS_OverlayType]=ODT_Bars; } ~AppearanceSettings() {} void setProfile(Profile *p) { m_profile=p; } //! \brief Returns the normal (unscaled) height of a graph - int graphHeight() { return (*m_profile)[AS_STR_GraphHeight].toInt(); } + int graphHeight() { return (*m_profile)[STR_AS_GraphHeight].toInt(); } //! \brief Returns true if AntiAliasing (the graphical smoothing method) is enabled - bool antiAliasing() { return (*m_profile)[AS_STR_AntiAliasing].toBool(); } + bool antiAliasing() { return (*m_profile)[STR_AS_AntiAliasing].toBool(); } //! \brief Returns true if renderPixmap function is in use, which takes snapshots of graphs - bool graphSnapshots() { return (*m_profile)[AS_STR_GraphSnapshots].toBool(); } + bool graphSnapshots() { return (*m_profile)[STR_AS_GraphSnapshots].toBool(); } //! \brief Returns true if Graphical animations & Transitions will be drawn - bool animations() { return (*m_profile)[AS_STR_Animations].toBool(); } + bool animations() { return (*m_profile)[STR_AS_Animations].toBool(); } //! \brief Returns true if Square Wave plots are preferred (where possible) - bool squareWavePlots() { return (*m_profile)[AS_STR_SquareWave].toBool(); } + bool squareWavePlots() { return (*m_profile)[STR_AS_SquareWave].toBool(); } //! \brief Returns the type of overlay flags (which are displayed over the Flow Waveform) - OverlayDisplayType overlayType() { return (OverlayDisplayType )(*m_profile)[AS_STR_OverlayType].toInt(); } + OverlayDisplayType overlayType() { return (OverlayDisplayType )(*m_profile)[STR_AS_OverlayType].toInt(); } //! \brief Set the normal (unscaled) height of a graph. - void setGraphHeight(int height) { (*m_profile)[AS_STR_GraphHeight]=height; } + void setGraphHeight(int height) { (*m_profile)[STR_AS_GraphHeight]=height; } //! \brief Set to true to turn on AntiAliasing (the graphical smoothing method) - void setAntiAliasing(bool aa) { (*m_profile)[AS_STR_AntiAliasing]=aa; } + void setAntiAliasing(bool aa) { (*m_profile)[STR_AS_AntiAliasing]=aa; } //! \brief Set to true if renderPixmap functions are in use, which takes snapshots of graphs. - void setGraphSnapshots(bool gs) { (*m_profile)[AS_STR_GraphSnapshots]=gs; } + void setGraphSnapshots(bool gs) { (*m_profile)[STR_AS_GraphSnapshots]=gs; } //! \brief Set to true if Graphical animations & Transitions will be drawn - void setAnimations(bool anim) { (*m_profile)[AS_STR_Animations]=anim; } + void setAnimations(bool anim) { (*m_profile)[STR_AS_Animations]=anim; } //! \brief Set whether or not to useSquare Wave plots (where possible) - void setSquareWavePlots(bool sw) { (*m_profile)[AS_STR_SquareWave]=sw; } + void setSquareWavePlots(bool sw) { (*m_profile)[STR_AS_SquareWave]=sw; } //! \brief Sets the type of overlay flags (which are displayed over the Flow Waveform) - void setOverlayType(OverlayDisplayType od) { (*m_profile)[AS_STR_OverlayType]=(int)od; } + void setOverlayType(OverlayDisplayType od) { (*m_profile)[STR_AS_OverlayType]=(int)od; } Profile *m_profile; }; - -extern const char * US_STR_UnitSystem; -extern const char * US_STR_EventWindowSize; -extern const char * US_STR_SkipEmptyDays; -extern const char * US_STR_RebuildCache; -extern const char * US_STR_ShowDebug; -extern const char * US_STR_LinkGroups; -extern const char * US_STR_CalculateRDI; +extern const char * STR_US_UnitSystem; +extern const char * STR_US_EventWindowSize; +extern const char * STR_US_SkipEmptyDays; +extern const char * STR_US_RebuildCache; +extern const char * STR_US_ShowDebug; +extern const char * STR_US_LinkGroups; +extern const char * STR_US_CalculateRDI; +extern const char * STR_US_ShowSerialNumbers; /*! \class UserSettings \brief Profile Options relating to General User Settings @@ -524,34 +528,36 @@ public: //! \brief Create UserSettings object given Profile *p, and initialize the defaults UserSettings(Profile *p) :m_profile(p) { - if (!m_profile->contains(US_STR_UnitSystem)) (*m_profile)[US_STR_UnitSystem]=US_Metric; - if (!m_profile->contains(US_STR_EventWindowSize)) (*m_profile)[US_STR_EventWindowSize]=4.0; - if (!m_profile->contains(US_STR_SkipEmptyDays)) (*m_profile)[US_STR_SkipEmptyDays]=true; - if (!m_profile->contains(US_STR_RebuildCache)) (*m_profile)[US_STR_RebuildCache]=false; // can't remember.. - if (!m_profile->contains(US_STR_ShowDebug)) (*m_profile)[US_STR_ShowDebug]=false; - if (!m_profile->contains(US_STR_LinkGroups)) (*m_profile)[US_STR_LinkGroups]=true; // can't remember.. - if (!m_profile->contains(US_STR_CalculateRDI)) (*m_profile)[US_STR_CalculateRDI]=false; + if (!m_profile->contains(STR_US_UnitSystem)) (*m_profile)[STR_US_UnitSystem]=US_Metric; + if (!m_profile->contains(STR_US_EventWindowSize)) (*m_profile)[STR_US_EventWindowSize]=4.0; + if (!m_profile->contains(STR_US_SkipEmptyDays)) (*m_profile)[STR_US_SkipEmptyDays]=true; + if (!m_profile->contains(STR_US_RebuildCache)) (*m_profile)[STR_US_RebuildCache]=false; // can't remember.. + if (!m_profile->contains(STR_US_ShowDebug)) (*m_profile)[STR_US_ShowDebug]=false; + if (!m_profile->contains(STR_US_LinkGroups)) (*m_profile)[STR_US_LinkGroups]=true; // can't remember.. + if (!m_profile->contains(STR_US_CalculateRDI)) (*m_profile)[STR_US_CalculateRDI]=false; + if (!m_profile->contains(STR_US_ShowSerialNumbers)) (*m_profile)[STR_US_ShowSerialNumbers]=false; } ~UserSettings() {} void setProfile(Profile *p) { m_profile=p; } - UnitSystem unitSystem() { return (UnitSystem)(*m_profile)[US_STR_UnitSystem].toInt(); } - double eventWindowSize() { return (*m_profile)[US_STR_EventWindowSize].toDouble(); } - bool skipEmptyDays() { return (*m_profile)[US_STR_SkipEmptyDays].toBool(); } - bool rebuildCache() { return (*m_profile)[US_STR_RebuildCache].toBool(); } - bool showDebug() { return (*m_profile)[US_STR_ShowDebug].toBool(); } - bool linkGroups() { return (*m_profile)[US_STR_LinkGroups].toBool(); } - bool calculateRDI() { return (*m_profile)[US_STR_CalculateRDI].toBool(); } - - void setUnitSystem(UnitSystem us) { (*m_profile)[US_STR_UnitSystem]=(int)us; } - void setEventWindowSize(double size) { (*m_profile)[US_STR_EventWindowSize]=size; } - void setSkipEmptyDays(bool skip) { (*m_profile)[US_STR_SkipEmptyDays]=skip; } - void setRebuildCache(bool rebuild) { (*m_profile)[US_STR_RebuildCache]=rebuild; } - void setShowDebug(bool b) { (*m_profile)[US_STR_ShowDebug]=b; } - void setLinkGroups(bool link) { (*m_profile)[US_STR_LinkGroups]=link; } - void setCalculateRDI(bool rdi) { (*m_profile)[US_STR_CalculateRDI]=rdi; } + UnitSystem unitSystem() { return (UnitSystem)(*m_profile)[STR_US_UnitSystem].toInt(); } + double eventWindowSize() { return (*m_profile)[STR_US_EventWindowSize].toDouble(); } + bool skipEmptyDays() { return (*m_profile)[STR_US_SkipEmptyDays].toBool(); } + bool rebuildCache() { return (*m_profile)[STR_US_RebuildCache].toBool(); } + bool showDebug() { return (*m_profile)[STR_US_ShowDebug].toBool(); } + bool linkGroups() { return (*m_profile)[STR_US_LinkGroups].toBool(); } + bool calculateRDI() { return (*m_profile)[STR_US_CalculateRDI].toBool(); } + bool showSerialNumbers() { return (*m_profile)[STR_US_ShowSerialNumbers].toBool(); } + void setUnitSystem(UnitSystem us) { (*m_profile)[STR_US_UnitSystem]=(int)us; } + void setEventWindowSize(double size) { (*m_profile)[STR_US_EventWindowSize]=size; } + void setSkipEmptyDays(bool skip) { (*m_profile)[STR_US_SkipEmptyDays]=skip; } + void setRebuildCache(bool rebuild) { (*m_profile)[STR_US_RebuildCache]=rebuild; } + void setShowDebug(bool b) { (*m_profile)[STR_US_ShowDebug]=b; } + void setLinkGroups(bool link) { (*m_profile)[STR_US_LinkGroups]=link; } + void setCalculateRDI(bool rdi) { (*m_profile)[STR_US_CalculateRDI]=rdi; } + void setShowSerialNumbers(bool enabled) { (*m_profile)[STR_US_ShowSerialNumbers]=enabled; } Profile *m_profile; }; diff --git a/SleepLib/session.cpp b/SleepLib/session.cpp index fd25edfa..0395391b 100644 --- a/SleepLib/session.cpp +++ b/SleepLib/session.cpp @@ -12,7 +12,9 @@ #include #include #include -#include + +#include "SleepLib/calcs.h" +#include "SleepLib/profiles.h" using namespace std; @@ -23,7 +25,7 @@ const quint16 filetype_data=1; // This is the uber important database version for SleepyHeads internal storage // Increment this after stuffing with Session's save & load code. const quint16 summary_version=11; -const quint16 events_version=8; +const quint16 events_version=9; Session::Session(Machine * m,SessionID session) { @@ -337,27 +339,42 @@ bool Session::LoadSummary(QString filename) return true; } +const quint16 compress_method=1; + bool Session::StoreEvents(QString filename) { QFile file(filename); file.open(QIODevice::WriteOnly); - QDataStream out(&file); + QByteArray headerbytes; + QDataStream header(&headerbytes,QIODevice::WriteOnly); + header.setVersion(QDataStream::Qt_4_6); + header.setByteOrder(QDataStream::LittleEndian); + + header << (quint32)magic; // New Magic Number + header << (quint16)events_version; // File Version + header << (quint16)filetype_data; // File type 1 == Event + header << (quint32)s_machine->id();// Machine Type + header << (quint32)s_session; // This session's ID + header << s_first; + header << s_last; + + quint16 compress=0; + + if (p_profile->session->compressSessionData()) + compress=compress_method; + + header << (quint16)compress; + + header << (quint16)s_machine->GetType();// Machine Type + + QByteArray databytes; + QDataStream out(&databytes,QIODevice::WriteOnly); out.setVersion(QDataStream::Qt_4_6); out.setByteOrder(QDataStream::LittleEndian); - out << (quint32)magic; // Magic Number - out << (quint16)events_version; // File Version - out << (quint16)filetype_data; // File type 1 == Event - out << (quint32)s_machine->id();// Machine ID - - out << (quint32)s_session; // This session's ID - out << s_first; - out << s_last; - out << (qint16)eventlist.size(); // Number of event categories - QHash >::iterator i; for (i=eventlist.begin(); i!=eventlist.end(); i++) { @@ -386,26 +403,59 @@ bool Session::StoreEvents(QString filename) for (int j=0;j0) { + data=qCompress(databytes); + } else { + data=databytes; + } + + file.write(headerbytes); + file.write(data); + file.close(); return true; } bool Session::LoadEvents(QString filename) { + quint32 t32,magicnum,machid,sessid; + quint16 t16,version,type,crc16,machtype,compmethod; + quint8 t8; + qint32 datasize; + if (filename.isEmpty()) { qDebug() << "Session::LoadEvents() Filename is empty"; return false; @@ -416,47 +466,67 @@ bool Session::LoadEvents(QString filename) qDebug() << "Couldn't open file" << filename; return false; } - QDataStream in(&file); - in.setVersion(QDataStream::Qt_4_6); - in.setByteOrder(QDataStream::LittleEndian); - quint32 t32; - quint16 t16; - quint8 t8; - //qint16 i16; - qint16 version; + QByteArray headerbytes=file.read(42); - in >> t32; // Magic Number - if (t32!=magic) { - qWarning() << "Wrong Magic number in " << filename; - return false; - } + QDataStream header(headerbytes); + header.setVersion(QDataStream::Qt_4_6); + header.setByteOrder(QDataStream::LittleEndian); - in >> version; // File Version + bool compressed=true; + header >> magicnum; // Magic Number (quint32) + header >> version; // Version (quint16) + header >> type; // File type (quint16) + header >> machid; // Machine ID (quint32) + header >> sessid; //(quint32) + header >> s_first; //(qint64) + header >> s_last; //(qint64) - if (version<6) { // prior to version 6 is too old to deal with - qDebug() << "Old File Version"; - //throw OldDBVersion(); - //qWarning() << "Old dbversion "<< t16 << "summary file.. Sorry, you need to purge and reimport"; - return false; - } - - in >> t16; // File Type - if (t16!=filetype_data) { + if (type!=filetype_data) { qDebug() << "Wrong File Type in " << filename; return false; } - qint32 ts32; - in >> ts32; // MachineID - if (ts32!=s_machine->id()) { - qWarning() << "Machine ID does not match in" << filename << " I will try to load anyway in case you know what your doing."; + if (magicnum!=magic) { + qWarning() << "Wrong Magic number in " << filename; + return false; } - in >> t32; // Sessionid - s_session=t32; - in >> s_first; - in >> s_last; + if (version<6) { // prior to version 6 is too old to deal with + qDebug() << "Old File Version, can't open file"; + return false; + } + + if (version<9) { + file.seek(32); + } else { + header >> compmethod; // Compression Method (quint16) + header >> machtype; // Machine Type (quint16) + header >> datasize; // Size of Uncompressed Data (quint32) + header >> crc16; // CRC16 of Uncompressed Data (quint16) + } + + QByteArray databytes,temp=file.readAll(); + file.close(); + if (compmethod>0) { + databytes=qUncompress(temp); + } else { + databytes=temp; + } + + if (databytes.size()!=datasize) { + qDebug() << "File" << filename << "has returned wrong datasize"; + return false; + } + quint16 crc=qChecksum(databytes.data(),databytes.size()); + if (crc!=crc16) { + qDebug() << "CRC Doesn't match in" << filename; + return false; + } + + QDataStream in(databytes); + in.setVersion(QDataStream::Qt_4_6); + in.setByteOrder(QDataStream::LittleEndian); qint16 mcsize; in >> mcsize; // number of Machine Code lists @@ -523,34 +593,39 @@ bool Session::LoadEvents(QString filename) size2=sizevec[i]; for (int j=0;j> t; - evec.m_data.push_back(t); - } + evec.m_data.resize(evec.m_count); + EventStoreType *ptr=evec.m_data.data(); + + in.readRawData((char *)ptr, evec.m_count << 1); +// for (quint32 c=0;c> t; +// *ptr++=t; +// } if (evec.hasSecondField()) { - evec.m_data2.reserve(evec.m_count); - for (quint32 c=0;c> t; - evec.m_data2.push_back(t); - } + evec.m_data2.resize(evec.m_count); + ptr=evec.m_data2.data(); + + in.readRawData((char *)ptr,evec.m_count << 1); +// for (quint32 c=0;c> t; +// *ptr++=t; +// } } //last=evec.first(); if (evec.type()!=EVL_Waveform) { - evec.m_time.reserve(evec.m_count); - for (quint32 c=0;c> x; - //last+=x; - evec.m_time.push_back(x); - //evec.m_time[c]=x; - } + evec.m_time.resize(evec.m_count); + quint32 * tptr=evec.m_time.data(); + in.readRawData((char *)tptr,evec.m_count << 2); +// for (quint32 c=0;c> x; +// *tptr++=x; +// } } } } - file.close(); if (versiongain(); m_gain[id]=gain; } - if (!((id==CPAP_FlowRate) || (id==CPAP_MaskPressure))) + if (!((id==CPAP_FlowRate) || (id==CPAP_MaskPressureHi) || (id==CPAP_RespEvent) || (id==CPAP_MaskPressure))) updateCountSummary(id); Min(id); @@ -627,7 +702,8 @@ void Session::UpdateSummaries() count(id); last(id); first(id); - if ((id==CPAP_FlowRate) || (id==CPAP_MaskPressure)) continue; + if (((id==CPAP_FlowRate) || (id==CPAP_MaskPressureHi) || (id==CPAP_RespEvent) || (id==CPAP_MaskPressure))) + continue; cph(id); sph(id); @@ -1183,6 +1259,7 @@ EventList * Session::AddEventList(ChannelID code, EventListType et,EventDataType //return NULL; } EventList * el=new EventList(et,gain,offset,min,max,rate,second_field); + eventlist[code].push_back(el); //s_machine->registerChannel(chan); return el; diff --git a/daily.cpp b/daily.cpp index 3688df44..bb09735e 100644 --- a/daily.cpp +++ b/daily.cpp @@ -423,15 +423,17 @@ void Daily::ReloadGraphs() { ui->splitter->setVisible(true); QDate d; + if (previous_date.isValid()) { d=previous_date; - Unload(d); - } //else +// Unload(d); + } d=PROFILE.LastDay(); if (!d.isValid()) { d=ui->calendar->selectedDate(); } on_calendar_currentPageChanged(d.year(),d.month()); + // this fires a signal which unloads the old and loads the new ui->calendar->setSelectedDate(d); //Load(d); } @@ -582,15 +584,21 @@ void Daily::LoadDate(QDate date) void Daily::on_calendar_selectionChanged() { - this->setCursor(Qt::BusyCursor); + QTime time; + time_t unload_time, load_time, other_time; + time.start(); + this->setCursor(Qt::BusyCursor); if (previous_date.isValid()) { // GraphView->fadeOut(); Unload(previous_date); } + unload_time=time.restart(); //bool fadedir=previous_date < ui->calendar->selectedDate(); ZombieMeterMoved=false; Load(ui->calendar->selectedDate()); + load_time=time.restart(); + //GraphView->fadeIn(fadedir); GraphView->redraw(); ui->calButton->setText(ui->calendar->selectedDate().toString(Qt::TextDate)); @@ -607,6 +615,9 @@ void Daily::on_calendar_selectionChanged() ui->weightSpinBox->setSuffix(STR_UNIT_KG); } this->setCursor(Qt::ArrowCursor); + other_time=time.restart(); + + qDebug() << "Page change time (in ms): Unload ="<"+cpap->machine->properties[STR_PROP_SubModel]; html+=""+cpap->machine->properties[STR_PROP_Brand]+"
"+cpap->machine->properties[STR_PROP_Model]+" "+cpap->machine->properties[STR_PROP_ModelNumber]+submodel+"\n"; - if (PROFILE.session->showSerialNumbers()) { + if (PROFILE.general->showSerialNumbers()) { html+=""+cpap->machine->properties[STR_PROP_Serial]+"\n"; } CPAPMode mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode); diff --git a/mainwindow.cpp b/mainwindow.cpp index e8aea035..3ec8a416 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -560,7 +560,7 @@ void MainWindow::on_summaryButton_clicked() if (mach.size()==0) { html+=""; - html+="

Please Import Some Data


SleepyHead is pretty much useless without it.
"; + html+="

Please Import Some Data


SleepyHead is pretty much useless without it.
First import can take a few minutes."; html+=htmlFooter(); ui->summaryView->setHtml(html); return; @@ -813,7 +813,6 @@ void MainWindow::on_summaryButton_clicked() QDate date=lastcpap; Day * day; bool lastchanged=false; - int cnt=0; QVector rxchange; do { day=PROFILE.GetGoodDay(date,MT_CPAP); @@ -1531,7 +1530,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) return; } - QString username=PROFILE.Get(QString("_{")+QString(UI_STR_UserName)+"}_"); + QString username=PROFILE.Get(QString("_{")+QString(STR_UI_UserName)+"}_"); bool print_bookmarks=false; if (name==STR_TR_Daily) { @@ -1823,7 +1822,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) if (name==STR_TR_Daily) { if (!print_bookmarks) { for (int i=0;isize();i++) { - gGraph *g=(*gv)[i]; + g=(*gv)[i]; if (g->isEmpty()) continue; if (!g->visible()) continue; @@ -2251,6 +2250,7 @@ void MainWindow::on_actionAll_Data_for_current_CPAP_machine_triggered() void MainWindow::keyPressEvent(QKeyEvent * event) { + Q_UNUSED(event) //qDebug() << "Keypress:" << event->key(); } diff --git a/newprofile.cpp b/newprofile.cpp index ecb887eb..a2b0120d 100644 --- a/newprofile.cpp +++ b/newprofile.cpp @@ -140,7 +140,7 @@ void NewProfile::on_nextButton_clicked() } } else { - prof.Erase(UI_STR_Password); + prof.Erase(STR_UI_Password); } profile->user->setGender((Gender)ui->genderCombo->currentIndex()); @@ -246,7 +246,7 @@ void NewProfile::edit(const QString name) ui->userNameEdit->setReadOnly(true); ui->firstNameEdit->setText(profile->user->firstName()); ui->lastNameEdit->setText(profile->user->lastName()); - if (profile->contains(UI_STR_Password) && !profile->p_preferences[UI_STR_Password].toString().isEmpty()) { + if (profile->contains(STR_UI_Password) && !profile->p_preferences[STR_UI_Password].toString().isEmpty()) { // leave the password box blank.. QString a="******"; ui->passwordEdit1->setText(a); diff --git a/overview.cpp b/overview.cpp index da6063de..dec29f06 100644 --- a/overview.cpp +++ b/overview.cpp @@ -430,7 +430,7 @@ void Overview::ResetGraphLayout() } -void Overview::on_printDailyButton_clicked() +/*void Overview::on_printDailyButton_clicked() { qint64 st,et; GraphView->GetXBounds(st,et); @@ -460,7 +460,7 @@ void Overview::on_printDailyButton_clicked() } else mainwin->Notify("If this was implemented yet, You'd be able to print multiple daily reports right now."); -} +}*/ void Overview::on_rangeCombo_activated(int index) { diff --git a/overview.h b/overview.h index 3eb7f118..0f404df4 100644 --- a/overview.h +++ b/overview.h @@ -94,7 +94,7 @@ private slots: //! \brief Resets view to currently shown start & end dates void on_toolButton_clicked(); - void on_printDailyButton_clicked(); + //void on_printDailyButton_clicked(); void on_rangeCombo_activated(int index); diff --git a/overview.ui b/overview.ui index 8c947343..878923d0 100644 --- a/overview.ui +++ b/overview.ui @@ -42,13 +42,13 @@ 4 - 9 + 4 0 - 9 + 4 0 @@ -62,6 +62,12 @@ + + + 0 + 0 + + Last Week @@ -118,6 +124,12 @@ + + + 0 + 0 + + true @@ -135,6 +147,12 @@ + + + 0 + 0 + + true diff --git a/preferencesdialog.cpp b/preferencesdialog.cpp index 511fb6cb..0158437d 100644 --- a/preferencesdialog.cpp +++ b/preferencesdialog.cpp @@ -168,6 +168,12 @@ PreferencesDialog::PreferencesDialog(QWidget *parent,Profile * _profile) : ui->complianceGroupbox->setChecked(profile->cpap->showComplianceInfo()); ui->complianceHours->setValue(profile->cpap->complianceHours()); + bool bcd=profile->session->backupCardData(); + ui->createSDBackups->setChecked(bcd); + ui->compressSDBackups->setEnabled(bcd); + ui->compressSDBackups->setChecked(profile->session->compressBackupData()); + ui->compressSessionData->setChecked(profile->session->compressSessionData()); + ui->graphHeight->setValue(profile->appearance->graphHeight()); if (!PREF.contains(STR_GEN_UpdatesAutoCheck)) PREF[STR_GEN_UpdatesAutoCheck]=true; @@ -288,14 +294,16 @@ void PreferencesDialog::Save() if ((profile->session->daySplitTime()!=ui->timeEdit->time()) || (profile->session->combineCloseSessions()!=ui->combineSlider->value()) || (profile->session->ignoreShortSessions()!=ui->IgnoreSlider->value())) { - profile->session->setTrashDayCache(true); needs_restart=true; - } else profile->session->setTrashDayCache(false); + } if (profile->general->calculateRDI() != ui->AddRERAtoAHI->isChecked()) { profile->general->setCalculateRDI(ui->AddRERAtoAHI->isChecked()); needs_restart=true; } + profile->session->setBackupCardData(ui->createSDBackups->isChecked()); + profile->session->setCompressBackupData(ui->compressSDBackups->isChecked()); + profile->session->setCompressSessionData(ui->compressSessionData->isChecked()); profile->session->setCombineCloseSessions(ui->combineSlider->value()); profile->session->setIgnoreShortSessions(ui->IgnoreSlider->value()); @@ -712,3 +720,23 @@ void PreferencesDialog::on_maskTypeCombo_activated(int index) } } } + +void PreferencesDialog::on_createSDBackups_toggled(bool checked) +{ + if (profile->session->backupCardData() && !checked) { + QList mach=PROFILE.GetMachines(MT_CPAP); + bool haveS9=false; + for (int i=0;iGetClass()==STR_MACH_ResMed) { + haveS9=true; + break; + } + } + if (haveS9 && QMessageBox::question(this,"This may not be a good idea","ResMed S9 machines routinely delete certain data from your SD card older than 7 and 30 days (depending on resolution). If you ever need to reimport this data again (whether in SleepyHead or ResScan) this data won't come back. If you need to conserve disk space, please remember to carry out manual backups. Are you sure you want to disable these backups?",QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) { + ui->createSDBackups->setChecked(true); + return; + } + } + if (!checked) ui->compressSDBackups->setChecked(false); + ui->compressSDBackups->setEnabled(checked); +} diff --git a/preferencesdialog.h b/preferencesdialog.h index a96d5627..c2cb16da 100644 --- a/preferencesdialog.h +++ b/preferencesdialog.h @@ -87,6 +87,8 @@ private slots: void on_maskTypeCombo_activated(int index); + void on_createSDBackups_toggled(bool checked); + private: //! \brief Populates the Graph Model view with data from the Daily, Overview & Oximetry gGraphView objects void resetGraphModel(); diff --git a/preferencesdialog.ui b/preferencesdialog.ui index c6cb2fc6..da9c3124 100644 --- a/preferencesdialog.ui +++ b/preferencesdialog.ui @@ -42,7 +42,7 @@ - 6 + 0 @@ -50,10 +50,10 @@ - 6 + 4 - 5 + 4 @@ -62,7 +62,7 @@ - 0 + 4 0 @@ -258,6 +258,16 @@ p, li { white-space: pre-wrap; } + + + + Keep session data in memory to speed up revisiting days. + + + Cache Session Data (uses more system memory) + + + @@ -277,198 +287,73 @@ p, li { white-space: pre-wrap; } - + - Custom Event Flagging + Session Storage Options - - true - - + - 0 - - 4 + + 9 + - 0 + 4 - 0 + 4 4 - - - - s - - - 1.000000000000000 - - - 10.000000000000000 - - - - - - - % - - - 10.000000000000000 - - - - - - - 0 - 0 - - + - Apnea + Create SD Card Backups during Import (especially important for ResMed users) - - + + - Duration - - - - - - - - 0 - 0 - - - - Flow Restriction - - - - - - - Qt::Horizontal - - - - 40 - 20 - - - - - - - - % - - - 40.000000000000000 - - - - - - - s - - - 10.000000000000000 + Compress Session Data (slower, but makes SleepyHead data smaller) - + - Hypopnea + Compress SD Card Backups (slower, but makes backups smaller) - - - - - - - AHI/Hour Graph Settings - - - true - - - - 0 - - - 4 - - - 0 - - - 0 - - - 4 - - - 0 - - - - - 0 - 0 - + + + + true + - Window Size + The following options affect the amount of disk space SleepyHead uses, and all have an effect on how long import takes. This will only really be noticeable on first import. + + + true - - - - minutes - - - 60 - - - - - + + - Reset to zero after each window + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans'; font-size:10pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Note: <span style=" font-style:italic;">Compression options don't automatically recompress already saved data.</span></p></body></html> - - - - Qt::Horizontal - - - - 40 - 20 - - - - @@ -487,112 +372,17 @@ p, li { white-space: pre-wrap; } - + - Other + &CPAP - + - 0 - - 4 - - 0 - - - 0 - - + 4 - - - - Import Locations - - - - 0 - - - 0 - - - 4 - - - 0 - - - 0 - - - - - QAbstractItemView::NoEditTriggers - - - - - - - - - - Add - - - - - - - Remove - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - - - - - - - - &Mask - - - - 0 - - - 2 - - - 2 - - - 0 - - - 2 - @@ -771,53 +561,258 @@ p, li { white-space: pre-wrap; } - - - + + + Qt::Vertical - - - 0 - - - 2 - - - 0 - - - 0 - - - - - QFrame::StyledPanel - - - Hmmm... Empty Space -Mask History could go here one day - -For now a reminder for PRS1 Users -The lower leak does not yet show unintentional -leaks. It currently showns the leak line -subtracted from the 0% percentile -(aka the minimum value) - -<--- None of this data is being used yet! - -To solve this properly requires some fairly -complex forumulae derived from the -Mask Leak Profiles -(Believe it or not, I am not a maths geek) - - - Qt::AlignCenter - - - - + + + + 4 + + + + + + 75 + true + + + + Use RDI instead of AHI (PRS1 only) + + + + + + + Show Compliance + + + true + + + + 4 + + + 4 + + + + + + 0 + 0 + + + + Regard days with under this usage as "incompliant". + + + hours + + + 1 + + + 8.000000000000000 + + + 4.000000000000000 + + + + + + + + 0 + 0 + + + + as over + + + + + + + of usage per night + + + + + + + + + + Custom Event Flagging + + + true + + + + + + Duration + + + + + + + + 0 + 0 + + + + Flow Restriction + + + + + + + + 0 + 0 + + + + Apnea + + + + + + + s + + + 1.000000000000000 + + + 10.000000000000000 + + + + + + + % + + + 10.000000000000000 + + + + + + + Qt::Horizontal + + + + 40 + 20 + + + + + + + + Hypopnea + + + + + + + s + + + 10.000000000000000 + + + + + + + % + + + 40.000000000000000 + + + + + + + + + + AHI/Hour Graph Settings + + + true + + + + + + + 0 + 0 + + + + Window + + + + + + + minutes + + + 60 + + + + + + + Zero Reset + + + + + + + + + + Qt::Vertical + + + + 20 + 40 + + + + + + @@ -825,10 +820,25 @@ Mask Leak Profiles &Events + + 4 + + + 0 + + + 4 + + + 0 + + + 0 + - This doesn't save yet... + Not entirely sure if this will get to live or not.. @@ -926,7 +936,7 @@ Mask Leak Profiles - Please Note: Graph visibility also depends on machine availability. + Double click on the (Y-axis) min/max values to edit them @@ -940,6 +950,9 @@ Mask Leak Profiles &Oximetry + + 4 + 2 @@ -1169,6 +1182,12 @@ p, li { white-space: pre-wrap; } &General + + 4 + + + 4 + @@ -1181,7 +1200,7 @@ p, li { white-space: pre-wrap; } 4 - + Bypass the login screen and load the most recent User Profile @@ -1191,115 +1210,20 @@ p, li { white-space: pre-wrap; } - - - - Allow use of multiple CPU cores where available to improve performance. May cause problems. - - - Enable Multithreading - - - - + Skip over Empty Days - - - - Keep session data in memory to speed up revisiting days. - - - Cache Session Data - - - - + - Allows graphs to be "screenshotted" for display purposes. -The Event Breakdown PIE chart uses this method, as does -the printing code. -Unfortunately some older computers/versions of Qt can cause -this application to be unstable with this feature enabled. + Allow use of multiple CPU cores where available to improve performance. May cause problems. - Enable event breakdown pie chart - - - - - - - Use RDI instead of AHI (PRS1 only) - - - - - - - - - - Show Compliance - - - true - - - - 4 - - - 4 - - - - - Regard days with under this usage as "incompliant". - - - hours - - - 1 - - - 8.000000000000000 - - - 4.000000000000000 - - - - - - - Qt::Horizontal - - - - 40 - 20 - - - - - - - - Define compliance as over - - - - - - - of usage per night + Enable Multithreading @@ -1326,6 +1250,70 @@ this application to be unstable with this feature enabled. + + + + Import Locations + + + + 0 + + + 0 + + + 4 + + + 0 + + + 0 + + + + + QAbstractItemView::NoEditTriggers + + + + + + + + + + Add + + + + + + + Remove + + + + + + + Qt::Vertical + + + + 20 + 40 + + + + + + + + + + @@ -1620,6 +1608,20 @@ this application to be unstable with this feature enabled. + + + + Allows graphs to be "screenshotted" for display purposes. +The Event Breakdown PIE chart uses this method, as does +the printing code. +Unfortunately some older computers/versions of Qt can cause +this application to be unstable with this feature enabled. + + + Show event breakdown pie chart + + +