diff --git a/sleepyhead/Graphs/gSummaryChart.cpp b/sleepyhead/Graphs/gSummaryChart.cpp index c3eeeb73..4afcbac2 100644 --- a/sleepyhead/Graphs/gSummaryChart.cpp +++ b/sleepyhead/Graphs/gSummaryChart.cpp @@ -256,7 +256,7 @@ void SummaryChart::SetDay(Day *nullday) break; case ST_CPH: - tmp = day->cph(code); + tmp = day->count(code) / day->hours(); break; case ST_SPH: diff --git a/sleepyhead/SleepLib/day.cpp b/sleepyhead/SleepLib/day.cpp index b9cd1523..7cd26737 100644 --- a/sleepyhead/SleepLib/day.cpp +++ b/sleepyhead/SleepLib/day.cpp @@ -767,7 +767,7 @@ int Day::count(ChannelID code) for (QList::iterator it = sessions.begin(); it != end; ++it) { Session & sess = *(*it); - if (sess.enabled()) { + if (sess.enabled() && sess.channelDataExists(code)) { sum += sess.count(code); } } diff --git a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp index 945da0e3..b8c23241 100644 --- a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp @@ -68,16 +68,12 @@ EDFSignal *EDFParser::lookupSignal(ChannelID ch) return nullptr; } + // This is bad, because ResMed thinks it was a cool idea to use two channels with the same name. + // Scan through EDF's list of signals to see if any match for (int i = 0; i < channames.value().size(); i++) { - QHash::iterator jj = lookup.find(channames.value()[i]); - - if (jj == lookup.end()) { - continue; - } - - // return the EDFSignal record - return jj.value(); + EDFSignal *sig = lookupLabel(channames.value()[i]); + if (sig) return sig; } // Failed @@ -105,13 +101,12 @@ bool matchSignal(ChannelID ch, const QString & name) return false; } -EDFSignal *EDFParser::lookupName(QString name) +EDFSignal *EDFParser::lookupLabel(QString name) { - QHash::iterator i = lookup.find(name); + int idx = signal_labels.indexOf(name); + if (idx < 0) return nullptr; - if (i != lookup.end()) { return i.value(); } - - return nullptr; + return signal[idx]; } EDFParser::EDFParser(QString name) @@ -128,6 +123,182 @@ EDFParser::~EDFParser() if (buffer) { delete [] buffer; } } +void ResmedLoader::ParseSTR(Machine *mach, QStringList strfiles) +{ + QStringList::iterator strend = strfiles.end(); + for (QStringList::iterator it = strfiles.begin(); it != strend; ++it) { + EDFParser str(*it); + if (!str.Parse()) continue; + if (mach->properties[STR_PROP_Serial] != str.serialnumber) { + qDebug() << "Trying to import a STR.edf from another machine, skipping" << mach->properties[STR_PROP_Serial] << str.serialnumber; + qDebug() << (*it); + continue; + } + + QDateTime start = QDateTime::fromMSecsSinceEpoch(str.startdate); + QDate date = start.date(); + quint32 timestamp = start.toTime_t(); + + qDebug() << "Parsing" << *it << date << str.GetNumDataRecords() << str.GetNumSignals(); + + EDFSignal *maskon = str.lookupLabel("Mask On"); + EDFSignal *maskoff = str.lookupLabel("Mask Off"); + EDFSignal *sig = nullptr; + quint32 laston = 0; + + bool skipday; + + int size = str.GetNumDataRecords(); + int cnt=0; + + // For each data record, representing 1 day each + for (int rec = 0; rec < str.GetNumDataRecords(); ++rec) { + int recstart = rec * maskon->nr; + + skipday = false; + if ((++cnt % 10) == 0) { + // TODO: Change me to emit once MachineLoader is QObjectified... + if (qprogress) { qprogress->setValue(10.0 + (float(cnt) / float(size) * 90.0)); } + + QApplication::processEvents(); + } + + // Scan the mask on/off events + for (int s = 0; s < maskon->nr; ++s) { + qint32 on = maskon->data[recstart+s]; + qint32 off = maskoff->data[recstart+s]; + quint32 ontime = timestamp + on * 60; + quint32 offtime = timestamp + off * 60; + + // -1 marks empty record, but can start with mask off, if sleep crosses noon + if (on < 0) { + if (off < 0) continue; // Both are -1, skip the rest of this day + // laston stops on this record + + QMap::iterator si = strsess.find(laston); + if (si != strsess.end()) { + if (si.value().maskoff == 0) { + si.value().maskoff = offtime; + } else { + if (si.value().maskoff != offtime) { + // not sure why this happens. + qDebug() << "WTF??" << si.value().maskoff << "!=" << offtime; + } + //Q_ASSERT(si.value().maskoff == offtime); + } + } + continue; + } + bool skip = false; + QMap::iterator sid = strsess.find(ontime); + if (sid != strsess.end()) { + skip=true; + } + + // For every mask on, there will be a session within 1 minute either way + // We can use that for data matching + STRRecord R; + + R.maskon = ontime; + if (offtime > 0) { + R.maskoff = offtime; + } + + if (sig = str.lookupLabel("Mask Dur")) { + R.maskdur = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if (sig == str.lookupLabel("Leak Med")) { + R.leakmed = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if (sig == str.lookupLabel("Leak Max")) { + R.leakmax = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if (sig == str.lookupLabel("Leak 95")) { + R.leak95 = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + + + if ((sig = str.lookupSignal(CPAP_PressureMax))) { + R.max_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_PressureMin))) { + R.max_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(RMS9_SetPressure))) { + R.set_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_EPAP))) { + R.epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_EPAPHi))) { + R.max_epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_EPAPLo))) { + R.min_epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_IPAP))) { + R.ipap = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_IPAPHi))) { + R.max_ipap = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_IPAPLo))) { + R.min_ipap = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_PS))) { + R.ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_PSMax))) { + R.max_ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_PSMin))) { + R.min_ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(RMS9_EPR))) { + R.epr = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(RMS9_EPRSet))) { + R.epr_set = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupSignal(CPAP_Mode))) { + int mode = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + // convert this + R.mode = mode; + } + if ((sig = str.lookupLabel("AHI"))) { + R.ahi = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("AI"))) { + R.ai = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("HI"))) { + R.hi = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("UAI"))) { + R.uai = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("CAI"))) { + R.cai = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + + laston = ontime; + if (skip) continue; + + + QDateTime dontime = QDateTime::fromTime_t(ontime); + date = dontime.date(); + R.date = date; + strsess[ontime] = R; + strdate[date].push_back(&strsess[ontime]); + QDateTime dofftime = QDateTime::fromTime_t(offtime); + qDebug() << "Mask on" << dontime << "Mask off" << dofftime; + + } + timestamp += 86400; + } + } +} + // Read a 16 bits integer qint16 EDFParser::Read16() { @@ -248,10 +419,11 @@ bool EDFParser::Parse() edfsignals.resize(num_signals); for (int i = 0; i < num_signals; i++) { - EDFSignal &signal = edfsignals[i]; - signal.data = nullptr; - signal.label = Read(16); - lookup[signal.label] = &signal; // Safe: edfsignals won't move. + EDFSignal &sig = edfsignals[i]; + sig.data = nullptr; + sig.label = Read(16); + signal_labels.push_back(sig.label); + signal.push_back(&sig); } for (int i = 0; i < num_signals; i++) { edfsignals[i].transducer_type = Read(80); } @@ -599,6 +771,23 @@ int ResmedLoader::Open(QString &path, Profile *profile) /////////////////////////////////////////////////////////////////////////////////// // Open and Parse STR.edf file /////////////////////////////////////////////////////////////////////////////////// + QStringList strfiles; + strfiles.push_back(strpath); + QDir dir(path + "STR_Backup"); + dir.setFilter(QDir::Files | QDir::Hidden | QDir::Readable); + QFileInfoList flist = dir.entryInfoList(); + + for (int i = 0; i < flist.size(); i++) { + QFileInfo fi = flist.at(i); + filename = fi.fileName(); + if (filename.startsWith("STR", Qt::CaseInsensitive)) { + strfiles.push_back(fi.filePath()); + } + } + + strsess.clear(); + ParseSTR(m, strfiles); + EDFParser stredf(strpath); if (!stredf.Parse()) { @@ -612,7 +801,7 @@ int ResmedLoader::Open(QString &path, Profile *profile) // Creating early as we need the object - QDir dir(newpath); + dir.setPath(newpath); /////////////////////////////////////////////////////////////////////////////////// @@ -691,60 +880,8 @@ int ResmedLoader::Open(QString &path, Profile *profile) QVector dayused; dayused.resize(days); - QList strfirst; - QList strlast; - QList strday; - QList dayfoo; - QHash > daystarttimes; - QHash > dayendtimes; - //qint16 on,off; - //qint16 o1[10],o2[10]; - //time_t st,et; time_t time = stredf.startdate / 1000L; // == 12pm on first day - // for (int i=0;idata[j+k]; - // off=maskoff->data[j+k]; - // o1[k]=on; - // o2[k]=off; - // if (on >= 0) ckon++; - // if (off >= 0) ckoff++; - // } - - // // set to true if day starts with machine running - // int offset=ckoff-ckon; - // dayfoo.push_back(offset>0); - - // st=0,et=0; - // time_t l,f; - - // // Find the Min & Max times for this day - // for (int k=0;k f)) st=f; - // if (!et || (et < l)) et=l; - // } - // strfirst.push_back(st); - // strlast.push_back(et); - // strday.push_back(i); - // dayused[i]=ckon; - // time+=86400; - // } // reset time to first day time = stredf.startdate / 1000; @@ -756,7 +893,7 @@ int ResmedLoader::Open(QString &path, Profile *profile) QStringList dirs; dirs.push_back(newpath); dir.setFilter(QDir::Dirs | QDir::Hidden | QDir::NoDotAndDotDot); - QFileInfoList flist = dir.entryInfoList(); + flist = dir.entryInfoList(); bool ok; for (int i = 0; i < flist.size(); i++) { @@ -906,433 +1043,21 @@ int ResmedLoader::Open(QString &path, Profile *profile) QString fn; Session *sess; int cnt = 0; - size = sessfiles.size(); + size = filegroups.size(); QHash sessday; - - ///////////////////////////////////////////////////////////////////////////// - // 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(); - - // // Earliest possible day number - // int edn=((sessionid-time)/86400)-1; - // if (edn<0) edn=0; - - // // Find real day number from str.edf mask on/off data. - // 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; - // } - // } - EDFSignal *sig; - ///////////////////////////////////////////////////////////////////////////// - // For all days not in session lists, (to get at days without data records) - ///////////////////////////////////////////////////////////////////////////// - // for (dn=0;dnSessionExists(st)) - // continue; - - // et=dayendtimes[dn].at(j); - - // // Create the session - // sess=new Session(m,st); - // sess->really_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; - // if (sess) { - // ///////////////////////////////////////////////////////////////////// - // // CPAP Mode - // ///////////////////////////////////////////////////////////////////// - // int mode; - // sig=stredf.lookupSignal(CPAP_Mode); - // if (sig) { - // mode=sig->data[dn]; - // } else mode=0; - - - // ///////////////////////////////////////////////////////////////////// - // // EPR Settings - // ///////////////////////////////////////////////////////////////////// - // sess->settings[CPAP_PresReliefType]=PR_EPR; - - // // Note: AutoSV machines don't have both fields - // sig=stredf.lookupSignal(RMS9_EPR); - // if (sig) { - // sess->settings[CPAP_PresReliefMode]=EventDataType(sig->data[dn])*sig->gain; - // } - // sig=stredf.lookupSignal(RMS9_EPRSet); - // if (sig) { - // sess->settings[CPAP_PresReliefSet]=EventDataType(sig->data[dn])*sig->gain; - // } - - - // ///////////////////////////////////////////////////////////////////// - // // Set Min & Max pressures depending on CPAP mode - // ///////////////////////////////////////////////////////////////////// - // if (mode==0) { - // sess->settings[CPAP_Mode]=MODE_CPAP; - // sig=stredf.lookupSignal(RMS9_SetPressure); // ?? What's meant by Set Pressure? - // if (sig) { - // EventDataType pressure=sig->data[dn]*sig->gain; - // sess->settings[CPAP_Pressure]=pressure; - // } - // } else { // VPAP or Auto - // if (mode>5) { - // if (mode>=7) - // sess->settings[CPAP_Mode]=MODE_ASV; - // else - // sess->settings[CPAP_Mode]=MODE_BIPAP; - - // EventDataType tmp,epap=0,ipap=0; - // if ((sig=stredf.lookupName("EPAP"))) { - // epap=sig->data[dn]*sig->gain; - // sess->settings[CPAP_EPAP]=epap; - // sess->setMin(CPAP_EPAP,epap); - // } - // if ((sig=stredf.lookupName("IPAP"))) { - // ipap=sig->data[dn]*sig->gain; - // sess->settings[CPAP_IPAP]=ipap; - // } - // if ((sig=stredf.lookupName("PS"))) { - // tmp=sig->data[dn]*sig->gain; - // sess->settings[CPAP_PS]=tmp; // technically this is IPAP-EPAP - // if (!ipap) { - // // not really possible. but anyway, just in case.. - // sess->settings[CPAP_IPAP]=epap+tmp; - // } - // } - // if ((sig=stredf.lookupName("Min PS"))) { - // tmp=sig->data[dn]*sig->gain; - // sess->settings[CPAP_PSMin]=tmp; - // sess->settings[CPAP_IPAPLo]=epap+tmp; - // sess->setMin(CPAP_IPAP,epap+tmp); - // } - // if ((sig=stredf.lookupName("Max PS"))) { - // tmp=sig->data[dn]*sig->gain; - // sess->settings[CPAP_PSMax]=tmp; - // sess->settings[CPAP_IPAPHi]=epap+tmp; - // } - // if ((sig=stredf.lookupName("RR"))) { // Is this a setting to force respiratory rate on S/T machines? - // tmp=sig->data[dn]; - // sess->settings[CPAP_RespRate]=tmp*sig->gain; - // } - - // if ((sig=stredf.lookupName("Easy-Breathe"))) { - // tmp=sig->data[dn]*sig->gain; - - // sess->settings[CPAP_PresReliefSet]=tmp; - // sess->settings[CPAP_PresReliefType]=(int)PR_EASYBREATHE; - // sess->settings[CPAP_PresReliefMode]=(int)PM_FullTime; - // } - - // } else { - // sess->settings[CPAP_Mode]=MODE_APAP; - // sig=stredf.lookupSignal(CPAP_PressureMin); - // if (sig) { - // EventDataType pressure=sig->data[dn]*sig->gain; - // sess->settings[CPAP_PressureMin]=pressure; - // //sess->setMin(CPAP_Pressure,pressure); - // } - // sig=stredf.lookupSignal(CPAP_PressureMax); - // if (sig) { - // EventDataType pressure=sig->data[dn]*sig->gain; - // 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]; - // if (valmed>=0) { - // sess->m_gain[CPAP_Leak]=sig->gain*60.0; - - // sess->m_valuesummary[CPAP_Leak][valmed]=51; - // } - // } - // if ((sig=stredf.lookupName("Leak 95"))) { - // val95=sig->data[dn]; - // if (val95>=0) - // sess->m_valuesummary[CPAP_Leak][val95]=45; - // } - // if ((sig=stredf.lookupName("Leak Max"))) { - // valmax=sig->data[dn]; - // if (valmax>=0) { - // sess->setMax(CPAP_Leak,valmax*sig->gain*60.0); - // 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; - // sess->m_valuesummary[CPAP_MinuteVent][valmed]=51; - // } - // if ((sig=stredf.lookupName("Min Vent 95"))) { - // val95=sig->data[dn]; - // sess->m_valuesummary[CPAP_MinuteVent][val95]=45; - // } - // if ((sig=stredf.lookupName("Min Vent Max"))) { - // valmax=sig->data[dn]; - // 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; - // sess->m_valuesummary[CPAP_RespRate][valmed]=51; - // } - // if ((sig=stredf.lookupName("RR 95"))) { - // val95=sig->data[dn]; - // sess->m_valuesummary[CPAP_RespRate][val95]=45; - // } - // if ((sig=stredf.lookupName("RR Max"))) { - // valmax=sig->data[dn]; - // sess->setMax(CPAP_RespRate,valmax*sig->gain); - // 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; - // sess->m_valuesummary[CPAP_TidalVolume][valmed]=51; - // } - // if ((sig=stredf.lookupName("Tid Vol 95"))) { - // val95=sig->data[dn]; - // sess->m_valuesummary[CPAP_TidalVolume][val95]=45; - // } - // if ((sig=stredf.lookupName("Tid Vol Max"))) { - // valmax=sig->data[dn]; - // sess->setMax(CPAP_TidalVolume,valmax*sig->gain*1000.0); - // 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; - // sess->m_valuesummary[CPAP_TgMV][valmed]=51; - // } - // if ((sig=stredf.lookupName("Targ Vent 95"))) { - // val95=sig->data[dn]; - // sess->m_valuesummary[CPAP_TgMV][val95]=45; - // } - // if ((sig=stredf.lookupName("Targ Vent Max"))) { - // valmax=sig->data[dn]; - // sess->setMax(CPAP_TgMV,valmax*sig->gain); - // sess->m_valuesummary[CPAP_TgMV][valmax]=4; - // } - - - // ///////////////////////////////////////////////////////////////////// - // // I:E Summary - // ///////////////////////////////////////////////////////////////////// - // if ((sig=stredf.lookupName("I:E Med"))) { - // valmed=sig->data[dn]; - // sess->m_gain[CPAP_IE]=sig->gain; - // sess->m_valuesummary[CPAP_IE][valmed]=51; - // } - // if ((sig=stredf.lookupName("I:E 95"))) { - // val95=sig->data[dn]; - // sess->m_valuesummary[CPAP_IE][val95]=45; - // } - // if ((sig=stredf.lookupName("I:E Max"))) { - // valmax=sig->data[dn]; - // sess->setMax(CPAP_IE,valmax*sig->gain); - // sess->m_valuesummary[CPAP_IE][valmax]=4; - // } - - // ///////////////////////////////////////////////////////////////////// - // // Mask Pressure Summary - // ///////////////////////////////////////////////////////////////////// - // if ((sig=stredf.lookupName("Mask Pres Med"))) { - // valmed=sig->data[dn]; - // if (valmed >= 0) { - // sess->m_gain[CPAP_Pressure]=sig->gain; - // sess->m_valuesummary[CPAP_Pressure][valmed]=51; - // } - // } - // if ((sig=stredf.lookupName("Mask Pres 95"))) { - // val95=sig->data[dn]; - // if (val95 >= 0) { - // sess->m_valuesummary[CPAP_Pressure][val95]=45; - // } - // } - // if ((sig=stredf.lookupName("Mask Pres Max"))) { - // valmax=sig->data[dn]; - // if (valmax >= 0) { - // sess->setMax(CPAP_Pressure,valmax*sig->gain); - // sess->m_valuesummary[CPAP_Pressure][valmax]=4; - // } - // } - // ///////////////////////////////////////////////////////////////////// - // // Therapy Pressure Summary - // ///////////////////////////////////////////////////////////////////// - // if ((sig=stredf.lookupName("Therapy Pres Me"))) { - // valmed=sig->data[dn]; - // if (valmed >= 0) { - // //sess->m_gain[CPAP_Pressure]=sig->gain; - // //sess->m_valuesummary[CPAP_Pressure][valmed]=51; - // } - // } - // if ((sig=stredf.lookupName("Therapy Pres 95"))) { - // val95=sig->data[dn]; - // if (val95 >= 0) { - //// sess->m_valuesummary[CPAP_Pressure][val95]=45; - // } - // } - // if ((sig=stredf.lookupName("Therapy Pres Ma"))) { - // valmax=sig->data[dn]; - // if (valmax >= 0) { - //// sess->setMax(CPAP_Pressure,valmax*sig->gain); - //// 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; - // sess->m_valuesummary[CPAP_IPAP][valmed]=51; - // } - // if ((sig=stredf.lookupName("Insp Pres 95"))) { - // val95=sig->data[dn]; - // sess->m_valuesummary[CPAP_IPAP][val95]=45; - // } - // if ((sig=stredf.lookupName("Insp Pres Max"))) { - // valmax=sig->data[dn]; - // 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]; - // if (valmed>=0) { - // sess->m_gain[CPAP_EPAP]=sig->gain; - // sess->m_valuesummary[CPAP_EPAP][valmed]=51; - // } - // } - // if ((sig=stredf.lookupName("Exp Pres 95"))) { - // if (val95>=0) { - // val95=sig->data[dn]; - // sess->m_valuesummary[CPAP_EPAP][val95]=45; - // } - // } - // if ((sig=stredf.lookupName("Exp Pres Max"))) { - // valmax=sig->data[dn]; - // if (valmax>=0) { - // sess->setMax(CPAP_EPAP,valmax*sig->gain); - // sess->m_valuesummary[CPAP_EPAP][valmax]=4; - // } - // } - - // ///////////////////////////////////////////////////////////////////// - // // Duration and Event Indices - // ///////////////////////////////////////////////////////////////////// - // dur=0; - // if ((sig=stredf.lookupName("Mask Dur"))) { - // dur=sig->data[dn]*sig->gain; - // dur/=60.0f; // convert to hours. - // } - // if ((sig=stredf.lookupName("OAI"))) { // Obstructive Apnea Index - // tmp=sig->data[dn]*sig->gain; - // if (tmp>=0) { - // sess->setCph(CPAP_Obstructive,tmp); - // sess->setCount(CPAP_Obstructive,tmp*dur); // Converting from indice to counts.. - // } - // } - // if ((sig=stredf.lookupName("HI"))) { // Hypopnea Index - // tmp=sig->data[dn]*sig->gain; - // if (tmp>=0) { - // sess->setCph(CPAP_Hypopnea,tmp); - // sess->setCount(CPAP_Hypopnea,tmp*dur); - // } - // } - // if ((sig=stredf.lookupName("UAI"))) { // Unspecified Apnea Index - // tmp=sig->data[dn]*sig->gain; - // if (tmp>=0) { - // sess->setCph(CPAP_Apnea,tmp); - // sess->setCount(CPAP_Apnea,tmp*dur); - // } - // } - // if ((sig=stredf.lookupName("CAI"))) { // "Central" Apnea Index - // tmp=sig->data[dn]*sig->gain; - // if (tmp>=0) { - // sess->setCph(CPAP_ClearAirway,tmp); - // sess->setCount(CPAP_ClearAirway,tmp*dur); - // } - // } - - // } - - // } backup_path += RMS9_STR_datalog + "/"; - QString backupfile, backfile, crcfile, yearstr, bkuppath; - // Have to sacrifice these features to get access to summary data. p_profile->session->setCombineCloseSessions(0); p_profile->session->setDaySplitTime(QTime(12,0,0)); p_profile->session->setIgnoreShortSessions(false); + QList trashstr; // list of strsess records to destroy afterwards + ///////////////////////////////////////////////////////////////////////////// // Scan through new file list and import sessions @@ -1340,13 +1065,29 @@ int ResmedLoader::Open(QString &path, Profile *profile) for (fgit = filegroups.begin(); fgit != filegroups.end(); ++fgit) { sessionid = fgit.key(); - // Skip file if already imported - if (m->SessionExists(sessionid)) { - continue; + sess = m->SessionExists(sessionid); + if (sess) { + if (sess->setting(CPAP_SummaryOnly).toBool()) { + // Reuse this session + sess->wipeSummary(); + } else { + continue; + } + } else { + // Could be importing from an older backup.. if so, destroy the summary only records + quint32 key = int(sessionid / 60) * 60; + sess = m->SessionExists(key); + if (sess) { + if (sess->setting(CPAP_SummaryOnly).toBool()) { + sess->Destroy(); + delete sess; + } + } + + // Create the session + sess = new Session(m, sessionid); } - // Create the session - sess = new Session(m, sessionid); if (!fgit.value().EVE.isEmpty()) { EDFParser edf(fgit.value().EVE); @@ -1382,169 +1123,73 @@ int ResmedLoader::Open(QString &path, Profile *profile) if (!sess->first()) { delete sess; continue; - } else { + } + sess->settings[CPAP_SummaryOnly] = false; + sess->SetChanged(true); - sess->SetChanged(true); - - dif = sess->first() - stredf.startdate; - - dn = dif / 86400000L; - - if ((dn >= 0) && (dn < days)) { - sig = stredf.lookupSignal(CPAP_Mode); - - if (sig) { - mode = sig->data[dn]; - } else { mode = 0; } - - // Ramp, Fulltime - // AutoSV machines don't have both fields - sig = stredf.lookupSignal(RMS9_EPR); - - if (sig) { - sess->settings[CPAP_PresReliefType] = PR_EPR; - prmode = EventDataType(sig->data[dn]) * sig->gain; - - // Off, - if (prmode < 0) { - // Kaart's data has -1 here.. Not sure what it means. - prmode = 0; - } else if (prmode > sig->physical_maximum) { - prmode = sig->physical_maximum; - } - - // My VPAP (using EasyBreathe) and JM's Elite (using none) have 0 - sess->settings[CPAP_PresReliefMode] = prmode; - } - - sig = stredf.lookupSignal(RMS9_EPRSet); - - if (sig) { - prset = EventDataType(sig->data[dn]) * sig->gain; - - if (prset > sig->physical_maximum) { - prset = sig->physical_maximum; - } else if (prset < sig->physical_minimum) { - prset = sig->physical_minimum; - } - - sess->settings[CPAP_PresReliefSet] = prset; - } + ///////////////////////////////////////////////////////////////////////////////// + // Process STR.edf now all valid Session data is imported + ///////////////////////////////////////////////////////////////////////////////// - if (mode == 0) { - sess->settings[CPAP_Mode] = MODE_CPAP; - sig = stredf.lookupSignal(RMS9_SetPressure); // ?? What's meant by Set Pressure? + QMap::iterator strsess_end = strsess.end(); + quint32 key = int(sessionid / 60) * 60; // round to 1 minute + QMap::iterator it = strsess.find(key); - if (sig) { - EventDataType pressure = sig->data[dn] * sig->gain; - sess->settings[CPAP_Pressure] = pressure; - } - } else if (mode > 5) { - if (mode >= 7) { - sess->settings[CPAP_Mode] = - MODE_ASV; // interestingly, last digit of model number matches these when in full mode. - } else { - sess->settings[CPAP_Mode] = MODE_BIPAP; - } + if (it == strsess_end) { + // ResMed merges mask on/off groups that are less than a minute apart + // this means have to jump back to the last session closest. - // All S9 machines have Set Pressure - // Elite has Min Pressure and Max Pressure - // VPAP Auto has EPAP, Min EPAP, IPAP and Max IPAP, and PS - // VPAP Adapt 36007 has just EPAP and PSLo/Hi, - // VPAP Adapt 36037 has EPAPLo, EPAPHi and PSLo/Hi - QHash hash; - hash[CPAP_EPAP] = "EPAP"; - hash[CPAP_IPAP] = "IPAP"; - hash[CPAP_EPAPLo] = "Min EPAP"; - hash[CPAP_EPAPHi] = "Max EPAP"; - hash[CPAP_IPAPLo] = "Min IPAP"; - hash[CPAP_IPAPHi] = "Max IPAP"; - hash[CPAP_PS] = "PS"; - hash[CPAP_PSMin] = "Min PS"; - hash[CPAP_PSMax] = "Max PS"; - hash[CPAP_RespRate] = "RR"; // Is this a setting to force respiratory rate on S/T machines? or an average - hash[CPAP_PresReliefSet] = "Easy-Breathe"; + QMap::iterator sit; + int min = 86400; - for (QHash::iterator it = hash.begin(); it != hash.end(); ++it) { - QHash::iterator a = stredf.lookup.find(it.value()); - if (a != stredf.lookup.end()) { - sig = a.value(); - sess->settings[it.key()] = EventDataType(sig->data[dn] * sig->gain); + // Look for the closest matching str record that starts before sessionid. + for (sit = strsess.begin(); sit != strsess_end; ++sit) { + STRRecord & R = *sit; - if (it.key() == CPAP_PresReliefSet) { - // this is not a setting on any machine I've played with, I think it's just an indication of the type of motor - sess->settings[CPAP_PresReliefType] = (int)PR_EASYBREATHE; - sess->settings[CPAP_PresReliefMode] = (int)PM_FullTime; - } - } - } + if (R.maskon > sessionid) + break; + int t = sessionid - R.maskon; - } else { - sess->settings[CPAP_Mode] = MODE_APAP; - sig = stredf.lookupSignal(CPAP_PressureMin); - - if (sig) { - EventDataType pressure = sig->data[dn] * sig->gain; - sess->settings[CPAP_PressureMin] = pressure; - //sess->setMin(CPAP_Pressure,pressure); - } - - sig = stredf.lookupSignal(CPAP_PressureMax); - - if (sig) { - EventDataType pressure = sig->data[dn] * sig->gain; - sess->settings[CPAP_PressureMax] = pressure; - //sess->setMax(CPAP_Pressure,pressure); - } - + if (qAbs(t) < min) { + it = sit; + min = t; } } + } + if (it != strsess_end) { + // This is the right session ID + STRRecord & R = it.value(); + QDateTime dt = QDateTime::fromTime_t(sessionid); + QDateTime rt = QDateTime::fromTime_t(R.maskon); + qDebug() << "Closest matching STR record for" << dt << (sess->length() / 1000L) << "is" << rt; - // The following only happens when the STR.edf file is not up to date.. - // This will only happen when the user fails to back up their SDcard properly. - // Basically takes a guess. - // bool dodgy=false; - // if (!sess->settings.contains(CPAP_Mode)) { - // //The following is a lame assumption if 50th percentile == max, then it's CPAP - // EventDataType max=sess->Max(CPAP_Pressure); - // EventDataType p50=sess->percentile(CPAP_Pressure,0.60); - // EventDataType p502=sess->percentile(CPAP_MaskPressure,0.60); - // p50=qMax(p50, p502); - // if (max==0) { - // dodgy=true; - // } else if (qAbs(max-p50)<1.8) { - // max=round(max*10.0)/10.0; - // sess->settings[CPAP_Mode]=MODE_CPAP; - // if (max<1) { - // int i=5; - // } - // sess->settings[CPAP_PressureMin]=max; - // EventDataType epr=round(sess->Max(CPAP_EPAP)*10.0)/10.0; - // int i=max-epr; - // sess->settings[CPAP_PresReliefType]=PR_EPR; - // prmode=(i>0) ? 0 : 1; - // sess->settings[CPAP_PresReliefMode]=prmode; - // sess->settings[CPAP_PresReliefSet]=i; + // Claim this session + R.sessionid = sessionid; + // Save maskon time in session setting so we can use it later to avoid doubleups. + sess->settings[RMS9_MaskOnTime]=R.maskon; - // } else { - // // It's not cpap, so just take the highest setting for this machines history. - // // This may fail if the missing str data is at the beginning of a fresh import. - // CPAPMode mode=(CPAPMode)(int)PROFILE.calcSettingsMax(CPAP_Mode,MT_CPAP,sess->machine()->FirstDay(),sess->machine()->LastDay()); - // if (modesettings[CPAP_Mode]=mode; - // // Assuming 10th percentile should cover for ramp/warmup - // sess->settings[CPAP_PressureMin]=sess->percentile(CPAP_Pressure,0.10); - // sess->settings[CPAP_PressureMax]=sess->Max(CPAP_Pressure); - // } - // } - //Rather than take a dodgy guess, EPR settings can take a hit, and this data can simply be missed.. + // Grab all the system settings + if (R.set_pressure >= 0) sess->settings[RMS9_SetPressure] = R.set_pressure; + if (R.min_pressure >= 0) sess->settings[CPAP_PressureMin] = R.min_pressure; + if (R.max_pressure >= 0) sess->settings[CPAP_PressureMax] = R.max_pressure; + if (R.ps >= 0) sess->settings[CPAP_PS] = R.ps; + if (R.min_ps >= 0) sess->settings[CPAP_PSMin] = R.min_ps; + if (R.max_ps >= 0) sess->settings[CPAP_PSMax] = R.max_ps; + if (R.epap >= 0) sess->settings[CPAP_EPAP] = R.epap; + if (R.max_epap >= 0) sess->settings[CPAP_EPAPHi] = R.max_epap; + if (R.min_epap >= 0) sess->settings[CPAP_EPAPLo] = R.min_epap; + if (R.ipap >= 0) sess->settings[CPAP_IPAP] = R.ipap; + if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap; + if (R.min_ipap >= 0) sess->settings[CPAP_IPAPLo] = R.min_ipap; + if (R.mode >= 0) sess->settings[CPAP_Mode] = R.mode; + if (R.epr >= 0) sess->settings[RMS9_EPR] = R.epr; + if (R.epr_set >= 0) sess->settings[RMS9_EPRSet] = R.epr_set; - // Add the session to the machine & profile objects - //if (!dodgy) + // Ignore all the rest of the sumary data, because there is enough available to calculate it with higher accuracy. if (sess->length() > 0) { m->AddSession(sess, profile); @@ -1556,209 +1201,141 @@ int ResmedLoader::Open(QString &path, Profile *profile) } } - ///////////////////////////////////////////////////////////////////////////////// - // Process STR.edf now all valid Session data is imported - ///////////////////////////////////////////////////////////////////////////////// - /* - qint64 tt=stredf.startdate; - QDateTime dt=QDateTime::fromMSecsSinceEpoch(tt); - QDateTime mt; - QDate d; + // Now look for any new summary data that can be extracted from STR.edf records + QMap::iterator it; + QMap::iterator end = strsess.end(); - EDFSignal *maskon=stredf.lookup["Mask On"]; - EDFSignal *maskoff=stredf.lookup["Mask Off"]; - int nr1=maskon->nr; - int nr2=maskoff->nr; - - qint64 mon, moff; - - int mode; - EventDataType prset, prmode; - - SessionID sid; - for (int dn=0; dn < days; dn++, tt+=86400000L) { - dt=QDateTime::fromMSecsSinceEpoch(tt); - d=dt.date(); - Day * day=PROFILE.GetDay(d, MT_CPAP); - if (day) { - continue; - } - QString a; - // Todo: check session start times. - // mask time is in minutes per day, assuming starting from 12 noon - // Also to think about: users who are too lazy to set their clocks, or who have flat clock batteries. - - int nr=maskon->nr; - int j=dn * nr; - qint16 m_on=-1, m_off=-1, m_off2=0; - for (int i=0;i<10;i++) { - m_on=maskon->data[j+i]; - if ((i>0) && (m_on >=0) && (m_on < m_off)) { - qDebug() << "Mask on before previous off"; - } - m_off=maskoff->data[j+i]; - m_off2=m_off; - - - if ((m_on >= 0) && (m_off < 0)) { - // valid session.. but machine turned off the next day - // look ahead and pinch the end time from tomorrows record - if ((dn+1) > days) { - qDebug() << "Last record should have contained a mask off event :("; - continue; - } - m_off=maskoff->data[j + nr]; - - if (maskon->data[j + nr] < 0) { - qDebug() << dn << "Looking ahead maskon should be < 0"; - continue; - } - if (m_off < 0) { - qDebug() << dn << "Looking ahead maskoff should be > 0"; - continue; - } - - // It's in the next day, so add one day in minutes.. - m_off+=1440; - - // Valid - - } else if ((m_off >= 0) && (m_on < 0)) { - - if (i>0) { - qDebug() << "WTH!??? Mask off but no on"; - } - // first record of day.. might already be on (crossing noon) - // Safely ignore because they are picked up on the other day. - continue; - } else if ((m_off < 0) && (m_on < 0)) - continue; - - mon=tt + m_on * 60000L; - moff=tt + m_off * 60000L; - - sid=mon/1000L; - QDateTime on=QDateTime::fromMSecsSinceEpoch(mon); - QDateTime off=QDateTime::fromMSecsSinceEpoch(moff); - - sess=new Session(m,sid); - sess->set_first(mon); - sess->set_last(moff); - - sig=stredf.lookupSignal(CPAP_Mode); - if (sig) { - mode=sig->data[dn]; - } else mode=0; - - sess->settings[CPAP_PresReliefType]=PR_EPR; - - // AutoSV machines don't have both fields - sig=stredf.lookupSignal(RMS9_EPR); - if (sig) { - prmode=EventDataType(sig->data[dn])*sig->gain; - if (prmode>sig->physical_maximum) { - int i=5; - } - sess->settings[CPAP_PresReliefMode]=prmode; - } - - sig=stredf.lookupSignal(RMS9_EPRSet); - if (sig) { - prset=EventDataType(sig->data[dn])*sig->gain; - if (prset>sig->physical_maximum) { - int i=5; - } - sess->settings[CPAP_PresReliefSet]=prset; - } - - - if (mode==0) { - sess->settings[CPAP_Mode]=MODE_CPAP; - sig=stredf.lookupSignal(RMS9_SetPressure); // ?? What's meant by Set Pressure? - if (sig) { - EventDataType pressure=sig->data[dn]*sig->gain; - sess->settings[CPAP_Pressure]=pressure; - } - } else if (mode>5) { - if (mode>=7) - sess->settings[CPAP_Mode]=MODE_ASV; - else - sess->settings[CPAP_Mode]=MODE_BIPAP; - - EventDataType tmp,epap=0,ipap=0; - if (stredf.lookup.contains("EPAP")) { - sig=stredf.lookup["EPAP"]; - epap=sig->data[dn]*sig->gain; - sess->settings[CPAP_EPAP]=epap; - } - if (stredf.lookup.contains("IPAP")) { - sig=stredf.lookup["IPAP"]; - ipap=sig->data[dn]*sig->gain; - sess->settings[CPAP_IPAP]=ipap; - } - if (stredf.lookup.contains("PS")) { - sig=stredf.lookup["PS"]; - tmp=sig->data[dn]*sig->gain; - sess->settings[CPAP_PS]=tmp; // technically this is IPAP-EPAP - if (!ipap) { - // not really possible. but anyway, just in case.. - sess->settings[CPAP_IPAP]=epap+tmp; - } - } - if (stredf.lookup.contains("Min PS")) { - sig=stredf.lookup["Min PS"]; - tmp=sig->data[dn]*sig->gain; - sess->settings[CPAP_PSMin]=tmp; - sess->settings[CPAP_IPAPLo]=epap+tmp; - sess->setMin(CPAP_IPAP,epap+tmp); - } - if (stredf.lookup.contains("Max PS")) { - sig=stredf.lookup["Max PS"]; - tmp=sig->data[dn]*sig->gain; - sess->settings[CPAP_PSMax]=tmp; - sess->settings[CPAP_IPAPHi]=epap+tmp; - } - if (stredf.lookup.contains("RR")) { // Is this a setting to force respiratory rate on S/T machines? - sig=stredf.lookup["RR"]; - tmp=sig->data[dn]; - sess->settings[CPAP_RespRate]=tmp*sig->gain; - } - - if (stredf.lookup.contains("Easy-Breathe")) { - sig=stredf.lookup["Easy-Breathe"]; - tmp=sig->data[dn]*sig->gain; - - sess->settings[CPAP_PresReliefSet]=tmp; - sess->settings[CPAP_PresReliefType]=(int)PR_EASYBREATHE; - sess->settings[CPAP_PresReliefMode]=(int)PM_FullTime; - } - - } else { - sess->settings[CPAP_Mode]=MODE_APAP; - sig=stredf.lookupSignal(CPAP_PressureMin); - if (sig) { - EventDataType pressure=sig->data[dn]*sig->gain; - sess->settings[CPAP_PressureMin]=pressure; - //sess->setMin(CPAP_Pressure,pressure); - } - sig=stredf.lookupSignal(CPAP_PressureMax); - if (sig) { - EventDataType pressure=sig->data[dn]*sig->gain; - sess->settings[CPAP_PressureMax]=pressure; - //sess->setMax(CPAP_Pressure,pressure); - } - - } - - m->AddSession(sess,profile); - - a=QString("[%3] %1:%2, ").arg(on.toString()).arg(off.toString()).arg(sid); - qDebug() << a.toStdString().data(); - } + QHash::iterator sessit; + QHash::iterator sessend = m->sessionlist.end();; + size = m->sessionlist.size(); + cnt=0; + // Scan through all sessions, and remove any strsess records that have a matching session already + for (sessit = m->sessionlist.begin(); sessit != sessend; ++sessit) { + sess = *sessit; + quint32 key = sess->settings[RMS9_MaskOnTime].toUInt(); + if ((++cnt % 10) == 0) { + // TODO: Change me to emit once MachineLoader is QObjectified... + if (qprogress) { qprogress->setValue(10.0 + (float(cnt) / float(size) * 90.0)); } + QApplication::processEvents(); } - */ + + QMap::iterator e = strsess.find(key); + if (e != end) { + strsess.erase(e); + } + } + + + size = strsess.size(); + cnt=0; + // Look for the nearest matching str record + for (it = strsess.begin(); it != end; ++it) { + STRRecord &R = *it; + Q_ASSERT(R.sessionid == 0); + //if (R.sessionid > 0) continue; + + + if ((++cnt % 10) == 0) { + // TODO: Change me to emit once MachineLoader is QObjectified... + if (qprogress) { qprogress->setValue(10.0 + (float(cnt) / float(size) * 90.0)); } + + QApplication::processEvents(); + } + + sess = new Session(m, R.maskon); + + sess->really_set_first(qint64(R.maskon) * 1000L); + sess->really_set_last(qint64(R.maskoff) * 1000L); + + // Claim this record for future imports + sess->settings[RMS9_MaskOnTime] = R.maskon; + sess->settings[CPAP_SummaryOnly] = true; + + sess->SetChanged(true); + + // First take the settings + if (R.set_pressure >= 0) sess->settings[RMS9_SetPressure] = R.set_pressure; + if (R.min_pressure >= 0) sess->settings[CPAP_PressureMin] = R.min_pressure; + if (R.max_pressure >= 0) sess->settings[CPAP_PressureMax] = R.max_pressure; + if (R.ps >= 0) sess->settings[CPAP_PS] = R.ps; + if (R.min_ps >= 0) sess->settings[CPAP_PSMin] = R.min_ps; + if (R.max_ps >= 0) sess->settings[CPAP_PSMax] = R.max_ps; + if (R.epap >= 0) sess->settings[CPAP_EPAP] = R.epap; + if (R.max_epap >= 0) sess->settings[CPAP_EPAPHi] = R.max_epap; + if (R.min_epap >= 0) sess->settings[CPAP_EPAPLo] = R.min_epap; + if (R.ipap >= 0) sess->settings[CPAP_IPAP] = R.ipap; + if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap; + if (R.min_ipap >= 0) sess->settings[CPAP_IPAPLo] = R.min_ipap; + if (R.mode >= 0) sess->settings[CPAP_Mode] = R.mode; + if (R.epr >= 0) sess->settings[RMS9_EPR] = R.epr; + if (R.epr_set >= 0) sess->settings[RMS9_EPRSet] = R.epr_set; + if (R.leakmax >= 0) sess->setMax(CPAP_Leak, R.leakmax); + if (R.leakmax >= 0) sess->setMin(CPAP_Leak, 0); + + + // Find the matching date group for this record + QMap >::iterator dtit = strdate.find(R.date); + + // should not be possible, but my brain hurts... + Q_ASSERT(dtit != strdate.end()); + + + if (dtit != strdate.end()) { + QList & dayrecs = dtit.value(); + int entries = dayrecs.count(); + bool hasdatasess=false; + EventDataType ai=0, hi=0, uai=0, time=0, totaltime=0; + + for (int c=0; c < dayrecs.size(); ++c) { + STRRecord *r = dayrecs[c]; + if (r->sessionid > 0) { + // get complicated.. calculate all the counts for valid sessions, and use the summary to make up the rest + hasdatasess = true; + } + totaltime += r->maskoff - r->maskon; + } + + + if (!hasdatasess) { + for (int c=0; c < dayrecs.size(); ++c) { + STRRecord *r = dayrecs[c]; + time = r->maskoff - r->maskon; + float ratio = time / totaltime; + + // Add the time weighted proportion of the events counts + if (r->ai >= 0) { + sess->setCount(CPAP_Obstructive, r->ai / ratio); + sess->setCph(CPAP_Obstructive, (r->ai / ratio) / (time / 3600.0)); + } + if (r->uai >= 0) { + sess->setCount(CPAP_Apnea, r->uai / ratio); + sess->setCph(CPAP_Apnea, (r->uai / ratio) / (time / 3600.0)); + } + if (r->hi >= 0) { + sess->setCount(CPAP_Hypopnea, r->hi / ratio); + sess->setCph(CPAP_Hypopnea, (r->hi / ratio) / (time / 3600.0)); + } + if (r->cai >= 0) { + sess->setCount(CPAP_ClearAirway, r->cai / ratio); + sess->setCph(CPAP_ClearAirway, (r->ai / ratio) / (time / 3600.0)); + } + if ((r->leakmed >= 0) && (r->leak95 >= 0) && (r->leakmax >= 0)) { + sess->m_valuesummary[CPAP_Leak][(short)r->leakmax]=100; + sess->m_valuesummary[CPAP_Leak][(short)r->leak95]=90; + sess->m_valuesummary[CPAP_Leak][(short)r->leakmed]=50; + } + + } + + } + } + + + + m->AddSession(sess, profile); + } #ifdef DEBUG_EFFICIENCY { @@ -2417,7 +1994,18 @@ void ResInitModelMap() resmed_codes[CPAP_RespEvent].push_back("Resp Event"); resmed_codes[CPAP_Pressure].push_back("Therapy Pres"); resmed_codes[CPAP_IPAP].push_back("Insp Pres"); + resmed_codes[CPAP_IPAP].push_back("IPAP"); resmed_codes[CPAP_EPAP].push_back("Exp Pres"); + resmed_codes[CPAP_EPAP].push_back("EPAP"); + resmed_codes[CPAP_EPAPHi].push_back("Max EPAP"); + resmed_codes[CPAP_EPAPLo].push_back("Min EPAP"); + resmed_codes[CPAP_IPAPHi].push_back("Max IPAP"); + resmed_codes[CPAP_IPAPLo].push_back("Min IPAP"); + + resmed_codes[CPAP_PS].push_back("PS"); + resmed_codes[CPAP_PSMin].push_back("Min PS"); + resmed_codes[CPAP_PSMax].push_back("Max PS"); + resmed_codes[CPAP_Leak].push_back("Leak"); resmed_codes[CPAP_Leak].push_back("Leck"); resmed_codes[CPAP_Leak].push_back("Lekk"); diff --git a/sleepyhead/SleepLib/loader_plugins/resmed_loader.h b/sleepyhead/SleepLib/loader_plugins/resmed_loader.h index 5d495a6c..a33e8adf 100644 --- a/sleepyhead/SleepLib/loader_plugins/resmed_loader.h +++ b/sleepyhead/SleepLib/loader_plugins/resmed_loader.h @@ -101,6 +101,103 @@ struct EDFSignal { int pos; }; +struct STRRecord +{ + STRRecord() { + maskon = 0; + maskoff = 0; + maskdur = 0; + maskevents = -1; + mode = -1; + set_pressure = -1; + epap = -1; + max_pressure = -1; + min_pressure = -1; + max_epap = -1; + min_epap = -1; + max_ps = -1; + min_ps = -1; + ps = -1; + ipap = -1; + max_ipap = -1; + min_ipap = -1; + epr = -1; + epr_set = -1; + sessionid = 0; + + ahi = -1; + ai = -1; + hi = -1; + uai = -1; + cai = -1; + + leakmed = -1; + leak95 = -1; + leakmax = -1; + date=QDate(); + } + STRRecord(const STRRecord & copy) { + maskon = copy.maskon; + maskoff = copy.maskoff; + maskdur = copy.maskdur; + maskevents = copy.maskevents; + mode = copy.mode; + set_pressure = copy.set_pressure; + epap = copy.epap; + max_pressure = copy.max_pressure; + min_pressure = copy.min_pressure; + max_ps = copy.max_ps; + min_ps = copy.min_ps; + ps = copy.ps; + max_epap = copy.max_epap; + min_epap = copy.min_epap; + ipap = copy.ipap; + max_ipap = copy.max_ipap; + min_ipap = copy.min_ipap; + epr = copy.epr; + epr_set = copy.epr_set; + sessionid = copy.sessionid; + ahi = copy.ahi; + ai = copy.ai; + hi = copy.hi; + uai = copy.uai; + cai = copy.cai; + date = copy.date; + leakmed = copy.leakmed; + leak95 = copy.leak95; + leakmax = copy.leakmax; + } + quint32 maskon; + quint32 maskoff; + EventDataType maskdur; + EventDataType maskevents; + EventDataType mode; + EventDataType set_pressure; + EventDataType max_pressure; + EventDataType min_pressure; + EventDataType epap; + EventDataType max_ps; + EventDataType min_ps; + EventDataType ps; + EventDataType max_epap; + EventDataType min_epap; + EventDataType ipap; + EventDataType max_ipap; + EventDataType min_ipap; + EventDataType epr; + EventDataType epr_set; + quint32 sessionid; + EventDataType ahi; + EventDataType ai; + EventDataType hi; + EventDataType uai; + EventDataType cai; + EventDataType leakmed; + EventDataType leak95; + EventDataType leakmax; + QDate date; +}; + /*! \class EDFParser \author Mark Watkins \brief Parse an EDF+ data file into a list of EDFSignal's @@ -127,11 +224,13 @@ class EDFParser QVector edfsignals; //! \brief An by-name indexed into the EDFSignal data - QHash lookup; + QStringList signal_labels; + + QList signal; //! \brief Look up signal names by SleepLib ChannelID.. A little "ResMed"ified.. :/ EDFSignal *lookupSignal(ChannelID); - EDFSignal *lookupName(QString name); + EDFSignal *lookupLabel(QString name); //! \brief Returns the number of signals contained in this EDF file long GetNumSignals() { return num_signals; } @@ -220,9 +319,15 @@ class ResmedLoader : public MachineLoader //! This contains the Pressure, Leak, Respiratory Rate, Minute Ventilation, Tidal Volume, etc.. bool LoadPLD(Session *sess, EDFParser &edf); + void ParseSTR(Machine *mach, QStringList strfiles); + + QString backup(QString file, QString backup_path, bool compress = false); QMap sessfiles; + QMap strsess; + QMap > strdate; + #ifdef DEBUG_EFFICIENCY QHash channel_efficiency; QHash channel_time; diff --git a/sleepyhead/SleepLib/machine.cpp b/sleepyhead/SleepLib/machine.cpp index 18917805..0993f3aa 100644 --- a/sleepyhead/SleepLib/machine.cpp +++ b/sleepyhead/SleepLib/machine.cpp @@ -528,7 +528,7 @@ PositionSensor::~PositionSensor() } -ChannelID NoChannel, SESSION_ENABLED; +ChannelID NoChannel, SESSION_ENABLED, CPAP_SummaryOnly; ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, CPAP_Pressure, CPAP_PS, CPAP_Mode, CPAP_AHI, CPAP_PressureMin, CPAP_PressureMax, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive, @@ -544,7 +544,7 @@ ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAP CPAP_Test2; -ChannelID RMS9_E01, RMS9_E02, RMS9_EPR, RMS9_EPRSet, RMS9_SetPressure; +ChannelID RMS9_E01, RMS9_E02, RMS9_EPR, RMS9_EPRSet, RMS9_SetPressure, RMS9_MaskOnTime; ChannelID INTP_SmartFlex; ChannelID INTELLIPAP_Unknown1, INTELLIPAP_Unknown2; diff --git a/sleepyhead/SleepLib/machine_common.h b/sleepyhead/SleepLib/machine_common.h index b979a323..a01b6ae0 100644 --- a/sleepyhead/SleepLib/machine_common.h +++ b/sleepyhead/SleepLib/machine_common.h @@ -85,7 +85,7 @@ enum MCDataType { MC_bool = 0, MC_int, MC_long, MC_float, MC_double, MC_string, MC_datetime }; -extern ChannelID NoChannel, SESSION_ENABLED; +extern ChannelID NoChannel, SESSION_ENABLED, CPAP_SummaryOnly; extern ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, CPAP_Pressure, CPAP_PS, CPAP_PSMin, CPAP_PSMax, CPAP_Mode, CPAP_AHI, @@ -100,7 +100,7 @@ extern ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CP CPAP_UserFlag1, CPAP_UserFlag2, CPAP_UserFlag3, CPAP_BrokenSummary, CPAP_BrokenWaveform, CPAP_RDI, CPAP_PresReliefSet, CPAP_PresReliefMode, CPAP_PresReliefType, CPAP_Test1, CPAP_Test2; -extern ChannelID RMS9_E01, RMS9_E02, RMS9_EPR, RMS9_EPRSet, RMS9_SetPressure; +extern ChannelID RMS9_E01, RMS9_E02, RMS9_EPR, RMS9_EPRSet, RMS9_SetPressure, RMS9_MaskOnTime; extern ChannelID INTP_SmartFlex; extern ChannelID PRS1_00, PRS1_01, PRS1_08, PRS1_0A, PRS1_0B, PRS1_0C, PRS1_0E, PRS1_0F, CPAP_LargeLeak, PRS1_12, diff --git a/sleepyhead/SleepLib/profiles.cpp b/sleepyhead/SleepLib/profiles.cpp index f1b89dfa..334ed4f9 100644 --- a/sleepyhead/SleepLib/profiles.cpp +++ b/sleepyhead/SleepLib/profiles.cpp @@ -482,10 +482,16 @@ void Profile::RemoveSession(Session *sess) } } + if (day->size() == 0) { + di.value().removeAll(day); + delete day; + } + // day->setFirst(first); // day->setLast(last); return; } + } } } diff --git a/sleepyhead/SleepLib/schema.cpp b/sleepyhead/SleepLib/schema.cpp index 13123b77..db6645ed 100644 --- a/sleepyhead/SleepLib/schema.cpp +++ b/sleepyhead/SleepLib/schema.cpp @@ -380,6 +380,15 @@ void init() QObject::tr("Upright angle in degrees"), QObject::tr("Inclination"), STR_UNIT_Degrees, DEFAULT, QColor("dark magenta"))); + schema::channel.add(GRP_CPAP, new Channel(RMS9_MaskOnTime = 0x1025, SETTING, SESSION, + "MaskOnTime", QObject::tr("Mask On Time"), + QObject::tr("Time started according to str.edf"), QObject::tr("Mask On Time"), STR_UNIT_Unknown, + DEFAULT, Qt::black)); + + schema::channel.add(GRP_CPAP, new Channel(CPAP_SummaryOnly = 0x1026, SETTING, SESSION, + "SummaryOnly", QObject::tr("Summary Only"), + QObject::tr("CPAP Session contains summary data onlyf"), QObject::tr("Summary Only"), STR_UNIT_Unknown, + DEFAULT, Qt::black)); NoChannel = 0; diff --git a/sleepyhead/SleepLib/session.cpp b/sleepyhead/SleepLib/session.cpp index 836d3605..485853ac 100644 --- a/sleepyhead/SleepLib/session.cpp +++ b/sleepyhead/SleepLib/session.cpp @@ -103,6 +103,24 @@ bool Session::OpenEvents() return s_events_loaded = true; } +bool Session::Destroy() +{ + QString path = PROFILE.Get(s_machine->properties[STR_PROP_Path]); + + PROFILE.RemoveSession(this); + s_machine->sessionlist.erase(s_machine->sessionlist.find(s_session)); + + QDir dir(path); + QString base; + base.sprintf("%08lx", s_session); + base = path + "/" + base; + + dir.remove(base + ".000"); + dir.remove(base + ".001"); + + return !dir.exists(base + ".000") && !dir.exists(base + ".001"); +} + bool Session::Store(QString path) // Storing Session Data in our format // {DataDir}/{MachineID}/{SessionID}.{ext} @@ -866,7 +884,9 @@ void Session::UpdateSummaries() for (; c != ev_end; c++) { id = c.key(); - if (schema::channel[id].type() == schema::DATA) { + + schema::ChanType ctype = schema::channel[id].type(); + if (ctype != schema::SETTING) { //sum(id); // avg calculates this and cnt. if (c.value().size() > 0) { EventList *el = c.value()[0]; @@ -1510,9 +1530,11 @@ int Session::count(ChannelID id) int sum = 0; int evec_size=evec.size(); + if (evec_size == 0) + return 0; - for (int i = 0; i < evec_size; i++) { - sum += evec[i]->count(); + for (int i = 0; i < evec_size; ++i) { + sum += evec.at(i)->count(); } m_cnt[id] = sum; diff --git a/sleepyhead/SleepLib/session.h b/sleepyhead/SleepLib/session.h index 72057073..a48b2c8e 100644 --- a/sleepyhead/SleepLib/session.h +++ b/sleepyhead/SleepLib/session.h @@ -72,6 +72,7 @@ class Session return s_session; } + //! \brief Returns whether or not session is being used. bool enabled(); @@ -134,12 +135,21 @@ class Session return s_changed; } + //! \brief Contains all the EventLists, indexed by ChannelID QHash > eventlist; //! \brief Sessions Settings List, contianing single settings for this session. QHash settings; + QVariant setting(ChannelID) { + QHash::iterator it = settings.find(CPAP_SummaryOnly); + if (it != settings.end()) { + return (*it); + } + return QVariant(); + } + // Session caches QHash m_cnt; QHash m_sum; @@ -300,7 +310,19 @@ class Session //! \brief Returns this sessions MachineID Machine *machine() { return s_machine; } - protected: + + bool Destroy(); + + void wipeSummary() { + s_first = s_last = 0; + s_enabled = true; + m_cph.clear(); + m_sum.clear(); + m_cnt.clear(); + } + + const QString & eventFile() { return s_eventfile; } +protected: SessionID s_session; Machine *s_machine; @@ -314,6 +336,8 @@ class Session bool s_events_loaded; char s_enabled; QString s_eventfile; + + }; diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index 3214c677..b0443160 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -1219,6 +1219,10 @@ void Daily::Load(QDate date) if (!PROFILE.session->cacheSessions()) { // Getting trashed on purge last day... + + // lastcpapday can get purged and be invalid + + if (lastcpapday && (lastcpapday!=cpap)) { for (QList::iterator s=lastcpapday->begin();s!=lastcpapday->end();++s) { (*s)->TrashEvents(); diff --git a/sleepyhead/mainwindow.cpp b/sleepyhead/mainwindow.cpp index 06616e90..6f338d79 100644 --- a/sleepyhead/mainwindow.cpp +++ b/sleepyhead/mainwindow.cpp @@ -304,6 +304,26 @@ void MainWindow::Notify(QString s, QString title, int ms) } } +void MainWindow::PopulatePurgeMenu() +{ + QList actions = ui->menu_Purge_CPAP_Data->actions(); + for (int i=0; i < actions.size(); i++) { + ui->menu_Purge_CPAP_Data->disconnect(ui->menu_Purge_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionPurgeMachine(QAction *))); + } + ui->menu_Purge_CPAP_Data->clear(); + QList machines = PROFILE.GetMachines(MT_CPAP); + for (int i=0; i < machines.size(); ++i) { + Machine *mach = machines.at(i); + QString name = mach->properties[STR_PROP_Brand]+" "+ + mach->properties[STR_PROP_Model]+" "+ + mach->properties[STR_PROP_Serial]; + QAction * action = new QAction(name, ui->menu_Purge_CPAP_Data); + action->setData(mach->GetClass()+":"+mach->properties[STR_PROP_Serial]); + ui->menu_Purge_CPAP_Data->addAction(action); + ui->menu_Purge_CPAP_Data->connect(ui->menu_Purge_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionPurgeMachine(QAction*))); + } +} + void MainWindow::Startup() { qDebug() << STR_TR_SleepyHeadVersion.toLocal8Bit().data() << "built with Qt" << QT_VERSION_STR << @@ -320,6 +340,8 @@ void MainWindow::Startup() // profile is a global variable set in main after login PROFILE.LoadMachineData(); + PopulatePurgeMenu(); + SnapshotGraph = new gGraphView(this, daily->graphView()); // the following are platform overides for the UsePixmapCache preference settings @@ -614,13 +636,28 @@ void MainWindow::on_action_Import_Data_triggered() #else const QString documentsFolder = QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation); #endif + + w.setDirectory(documentsFolder); w.setFileMode(QFileDialog::Directory); w.setOption(QFileDialog::ShowDirsOnly, true); - w.setOption(QFileDialog::DontUseNativeDialog,true); -//#if defined(Q_OS_MAC) && (QT_VERSION < QT_VERSION_CHECK(4,8,0)) -// // Fix for tetragon, 10.6 barfs up Qt's custom dialog -// w.setOption(QFileDialog::DontUseNativeDialog, true); + + // This doesn't work on WinXP + +#ifdef Q_OS_MAC +#if (QT_VERSION < QT_VERSION_CHECK(4,8,0)) + // Fix for tetragon, 10.6 barfs up Qt's custom dialog + w.setOption(QFileDialog::DontUseNativeDialog, true); +#else + w.setOption(QFileDialog::DontUseNativeDialog,false); +#endif // version check + +#elif Q_OS_UNIX + w.setOption(QFileDialog::DontUseNativeDialog,false); +#elif Q_OS_WIN + // check the Os version.. winxp chokes + w.setOption(QFileDialog::DontUseNativeDialog, true); +#endif //#else // w.setOption(QFileDialog::DontUseNativeDialog, false); @@ -692,6 +729,7 @@ void MainWindow::on_action_Import_Data_triggered() } else { mainwin->Notify(tr("Import Problem\n\nCouldn't find any new Machine Data at the locations given")); } + PopulatePurgeMenu(); } QMenu *MainWindow::CreateMenu(QString title) { @@ -1659,8 +1697,6 @@ void MainWindow::on_actionPurge_Current_Day_triggered() delete sess; } - - QList &dl = PROFILE.daylist[date]; QList::iterator it;//=dl.begin(); @@ -1679,39 +1715,99 @@ void MainWindow::on_actionPurge_Current_Day_triggered() getDaily()->LoadDate(date); } -void MainWindow::on_actionAll_Data_for_current_CPAP_machine_triggered() +void MainWindow::on_actionPurgeMachine(QAction *action) { - QDate date = getDaily()->getDate(); - Day *day = PROFILE.GetDay(date, MT_CPAP); - Machine *m; - - if (day) { - m = day->machine; - - if (!m) { - qDebug() << "Gah!! no machine to purge"; - return; - } - - if (QMessageBox::question(this, - tr("Are you sure?"), - tr("Are you sure you want to purge all CPAP data for the following machine:\n\n") + - m->properties[STR_PROP_Brand] + " " + m->properties[STR_PROP_Model] + " " + - m->properties[STR_PROP_ModelNumber] + " (" + m->properties[STR_PROP_Serial] + ")", - QMessageBox::Yes | QMessageBox::No, - QMessageBox::No) == QMessageBox::Yes) { - - - if (m->Purge(3478216)) { - // Turn on automatic re-import - // Note: I deliberately haven't added a Profile help for this - PROFILE.Save(); - } - // delete or not to delete.. this needs to delete later.. :/ - //delete m; - RestartApplication(); + QString data = action->data().toString(); + QString cls = data.section(":",0,0); + QString serial = data.section(":", 1); + QList machines = PROFILE.GetMachines(MT_CPAP); + Machine * mach = nullptr; + for (int i=0; i < machines.size(); ++i) { + Machine * m = machines.at(i); + if ((m->GetClass() == cls) && (m->properties[STR_PROP_Serial] == serial)) { + mach = m; + break; } } + if (!mach) return; + purgeMachine(mach); +} + +void MainWindow::purgeMachine(Machine * mach) +{ + + if (QMessageBox::question(this, + STR_MessageBox_Question, + tr("Are you sure you want to purge all CPAP data for the following machine:\n\n") + + mach->properties[STR_PROP_Brand] + " " + mach->properties[STR_PROP_Model] + " " + + mach->properties[STR_PROP_ModelNumber] + " (" + mach->properties[STR_PROP_Serial] + ")", + QMessageBox::Yes | QMessageBox::No, + QMessageBox::No) == QMessageBox::No) { + return; + } + + QHash::iterator it; + QHash::iterator s_end = mach->sessionlist.end(); + QList sessions; + for (it = mach->sessionlist.begin(); it != s_end; ++it) { + Session * sess = *it; + sessions.push_back(sess); + } + bool success = true; + for (int i=0; i < sessions.size(); ++i) { + Session * sess = sessions[i]; + if (!sess->Destroy()) { + qDebug() << "Could not destroy "+mach->GetClass()+" ("+mach->properties[STR_PROP_Serial]+") session" << sess->session(); + success = false; + } else { + mach->sessionlist.erase(mach->sessionlist.find(sess->session())); + } + delete sess; + } + if (success) { + mach->sessionlist.clear(); + mach->day.clear(); + + } else { + QMessageBox::warning(this, STR_MessageBox_Error, + tr("Not all session data could be removed, you have to delete the following folder manually.") + +"\n\n"+ + QDir::toNativeSeparators(PROFILE.Get(mach->properties[STR_PROP_Path])), QMessageBox::Ok, QMessageBox::Ok); + if (overview) { overview->ReloadGraphs(); } + if (daily) { + daily->clearLastDay(); // otherwise Daily will crash + daily->LoadDate(daily->getDate()); + } + GenerateStatistics(); + return; + } + + if (overview) { overview->ReloadGraphs(); } + if (daily) { + daily->clearLastDay(); // otherwise Daily will crash + daily->LoadDate(daily->getDate()); + } + GenerateStatistics(); + QApplication::processEvents(); + + if (QMessageBox::question(this, + STR_MessageBox_Question, + tr("Machine data has been successfully purged.") + "\n\n" + + tr("Would you like to reimport from the backup folder?") + "\n\n" + + QDir::toNativeSeparators(PROFILE.Get(mach->properties[STR_PROP_BackupPath])), + QMessageBox::Yes | QMessageBox::No, + QMessageBox::Yes) == QMessageBox::No) { + PROFILE.machlist.erase(PROFILE.machlist.find(mach->id())); + delete mach; + } else { + importCPAP(PROFILE.Get(mach->properties[STR_PROP_BackupPath]),tr("Please wait, importing...")); + if (overview) { overview->ReloadGraphs(); } + if (daily) { + daily->LoadDate(daily->getDate()); + } + } + GenerateStatistics(); + PROFILE.Save(); } void MainWindow::keyPressEvent(QKeyEvent *event) diff --git a/sleepyhead/mainwindow.h b/sleepyhead/mainwindow.h index 3cb66e53..3b877144 100644 --- a/sleepyhead/mainwindow.h +++ b/sleepyhead/mainwindow.h @@ -257,9 +257,6 @@ class MainWindow : public QMainWindow //! \brief Destroy the CPAP data for the currently selected day, so it can be freshly imported again void on_actionPurge_Current_Day_triggered(); - //! \brief Destroy ALL the CPAP data for the currently selected machine, so it can be freshly imported again - void on_actionAll_Data_for_current_CPAP_machine_triggered(); - void on_action_Sidebar_Toggle_toggled(bool arg1); void on_recordsBox_linkClicked(const QUrl &arg1); @@ -311,6 +308,9 @@ class MainWindow : public QMainWindow void on_reportModeStandard_clicked(); + void on_actionPurgeMachine(QAction *action); + + private: int importCPAP(const QString &path, const QString &message); void importCPAPBackups(); @@ -335,6 +335,13 @@ private: QString bookmarkFilter; bool m_restartRequired; volatile bool m_inRecalculation; + + void PopulatePurgeMenu(); + + //! \brief Destroy ALL the CPAP data for the selected machine + void purgeMachine(Machine *); + + }; #endif // MAINWINDOW_H diff --git a/sleepyhead/mainwindow.ui b/sleepyhead/mainwindow.ui index 6489949f..23607ab9 100644 --- a/sleepyhead/mainwindow.ui +++ b/sleepyhead/mainwindow.ui @@ -3125,18 +3125,16 @@ border-radius: 10px; &Purge CPAP Data - - + - @@ -3304,7 +3302,7 @@ border-radius: 10px; - &Current Selected Day + Purge &Current Selected Day