diff --git a/sleepyhead/Graphs/gGraphView.h b/sleepyhead/Graphs/gGraphView.h index 545258c5..ab1a0074 100644 --- a/sleepyhead/Graphs/gGraphView.h +++ b/sleepyhead/Graphs/gGraphView.h @@ -382,6 +382,8 @@ class gGraphView //! \brief Returns true if all Graph objects contain NO day data. ie, graph area is completely empty. bool isEmpty(); + Day * day() { return m_day; } + //! \brief Tell all graphs to deslect any highlighted areas void deselect(); diff --git a/sleepyhead/Graphs/gLineChart.cpp b/sleepyhead/Graphs/gLineChart.cpp index 46b3d9ca..915cdb58 100644 --- a/sleepyhead/Graphs/gLineChart.cpp +++ b/sleepyhead/Graphs/gLineChart.cpp @@ -420,6 +420,7 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion) maxx = w.max_x, minx = w.min_x; } + // hmmm.. subtract_offset.. EventDataType miny = m_physminy; @@ -428,6 +429,7 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion) w.roundY(miny, maxy); + //#define DEBUG_AUTOSCALER #ifdef DEBUG_AUTOSCALER QString a = QString().sprintf("%.2f - %.2f",miny, maxy); diff --git a/sleepyhead/Graphs/gSummaryChart.cpp b/sleepyhead/Graphs/gSummaryChart.cpp index 1039903c..80f80edd 100644 --- a/sleepyhead/Graphs/gSummaryChart.cpp +++ b/sleepyhead/Graphs/gSummaryChart.cpp @@ -437,6 +437,8 @@ void SummaryChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion) int days = ceil(double(maxx-minx) / 86400000.0); + bool buttuglydaysteps = !p_profile->appearance->animations(); + double lcursor = w.graphView()->currentTime(); if (days >= 1) { @@ -444,10 +446,13 @@ void SummaryChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion) double a = lcursor - w.min_x; double c = a / b; - minx = floor(double(minx)/86400000.0); - minx *= 86400000L; + if (buttuglydaysteps) { + // this kills the beautiful smooth scrolling and makes days stop on day boundaries :( + minx = floor(double(minx)/86400000.0); + minx *= 86400000L; - maxx = minx + 86400000L * qint64(days)-1; + maxx = minx + 86400000L * qint64(days)-1; + } b = maxx - minx; double d = c * b; diff --git a/sleepyhead/Graphs/gXAxis.cpp b/sleepyhead/Graphs/gXAxis.cpp index b4ad8ebe..f1693871 100644 --- a/sleepyhead/Graphs/gXAxis.cpp +++ b/sleepyhead/Graphs/gXAxis.cpp @@ -6,13 +6,13 @@ * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ -#include "Graphs/gXAxis.h" - #include #include #include +#include "Graphs/gXAxis.h" +#include "SleepLib/profiles.h" #include "Graphs/glcommon.h" #include "Graphs/gGraph.h" #include "Graphs/gGraphView.h" @@ -121,11 +121,14 @@ void gXAxis::paint(QPainter &painter, gGraph &w, const QRegion ®ion) int days = ceil(double(maxx-minx) / 86400000.0); - if (m_roundDays && (days >= 1)) { - minx = floor(double(minx)/86400000.0); - minx *= 86400000L; + bool buttuglydaysteps = !p_profile->appearance->animations(); + if (buttuglydaysteps) { + if (m_roundDays && (days >= 1)) { + minx = floor(double(minx)/86400000.0); + minx *= 86400000L; - maxx = minx + 86400000L * qint64(days); + maxx = minx + 86400000L * qint64(days); + } } diff --git a/sleepyhead/SleepLib/day.cpp b/sleepyhead/SleepLib/day.cpp index b255a0b5..37b784fd 100644 --- a/sleepyhead/SleepLib/day.cpp +++ b/sleepyhead/SleepLib/day.cpp @@ -460,6 +460,157 @@ EventDataType Day::p90(ChannelID code) return percentile(code, 0.90F); } +EventDataType Day::rangeCount(ChannelID code, qint64 st, qint64 et) +{ + int cnt = 0; + + QList::iterator end = sessions.end(); + for (QList::iterator it = sessions.begin(); it != end; ++it) { + Session &sess = *(*it); + + if (sess.enabled()) { + cnt += sess.rangeCount(code, st, et); + } + } + + return cnt; +} +EventDataType Day::rangeSum(ChannelID code, qint64 st, qint64 et) +{ + double val = 0; + + QList::iterator end = sessions.end(); + for (QList::iterator it = sessions.begin(); it != end; ++it) { + Session &sess = *(*it); + + if (sess.enabled()) { + val += sess.rangeSum(code, st, et); + } + } + + return val; +} +EventDataType Day::rangeAvg(ChannelID code, qint64 st, qint64 et) +{ + double val = 0; + int cnt = 0; + + QList::iterator end = sessions.end(); + for (QList::iterator it = sessions.begin(); it != end; ++it) { + Session &sess = *(*it); + + if (sess.enabled()) { + val += sess.rangeSum(code, st, et); + cnt += sess.rangeCount(code, st,et); + } + } + + if (cnt == 0) { return 0; } + val /= double(cnt); + + return val; +} +EventDataType Day::rangeWavg(ChannelID code, qint64 st, qint64 et) +{ + double sum = 0; + double cnt = 0; + QList::iterator end = sessions.end(); + for (QList::iterator it = sessions.begin(); it != end; ++it) { + Session &sess = *(*it); + QHash >::iterator EVEC = sess.eventlist.find(code); + if (EVEC == sess.eventlist.end()) continue; + + QVector::iterator EL; + QVector::iterator EVEC_end = EVEC.value().end(); + for (EL = EVEC.value().begin(); EL != EVEC_end; ++EL) { + EventList * el = *EL; + if (el->count() < 1) continue; + EventDataType lastdata = el->data(0); + qint64 lasttime = el->time(0); + + if (lasttime < st) + lasttime = st; + + for (unsigned i=1; icount(); i++) { + double data = el->data(i); + qint64 time = el->time(i); + + if (time < st) { + lasttime = st; + lastdata = data; + continue; + } + + if (time > et) { + time = et; + } + + double duration = double(time - lasttime) / 1000.0; + sum += data * duration; + cnt += duration; + + if (time >= et) break; + + lasttime = time; + lastdata = data; + } + } + } + if (cnt < 0.000001) + return 0; + return sum / cnt; +} + + +// Boring non weighted percentile +EventDataType Day::rangePercentile(ChannelID code, float p, qint64 st, qint64 et) +{ + int count = rangeCount(code, st,et); + QVector list; + list.resize(count); + int idx = 0; + + QList::iterator end = sessions.end(); + for (QList::iterator it = sessions.begin(); it != end; ++it) { + Session &sess = *(*it); + QHash >::iterator EVEC = sess.eventlist.find(code); + if (EVEC == sess.eventlist.end()) continue; + + QVector::iterator EL; + QVector::iterator EVEC_end = EVEC.value().end(); + for (EL = EVEC.value().begin(); EL != EVEC_end; ++EL) { + EventList * el = *EL; + for (unsigned i=0; icount(); i++) { + qint64 time = el->time(i); + if ((time < st) || (time > et)) continue; + list[idx++] = el->data(i); + } + } + } + + // TODO: use nth_element instead.. + qSort(list); + + float b = float(idx) * p; + int a = floor(b); + int c = ceil(b); + + if ((a == c) || (c >= idx)) { + return list[a]; + } + + EventDataType v1 = list[a]; + EventDataType v2 = list[c]; + + EventDataType diff = v2 - v1; // the whole == C-A + + double ba = b - float(a); // A....B...........C == B-A + + double val = v1 + diff * ba; + + return val; +} + EventDataType Day::avg(ChannelID code) { double val = 0; @@ -470,15 +621,16 @@ EventDataType Day::avg(ChannelID code) for (QList::iterator it = sessions.begin(); it != end; ++it) { Session &sess = *(*it); - if (sess.enabled() && sess.m_avg.contains(code)) { - val += sess.avg(code); - cnt++; // hmm.. averaging averages doesn't feel right.. + if (sess.enabled()) { + val += sess.sum(code); + cnt += sess.count(code); } } if (cnt == 0) { return 0; } + val /= double(cnt); - return EventDataType(val / float(cnt)); + return val; } EventDataType Day::sum(ChannelID code) @@ -1234,25 +1386,36 @@ bool Day::removeSession(Session *sess) QString Day::getCPAPMode() { - Q_ASSERT(machine(MT_CPAP) != nullptr); + Machine * mach = machine(MT_CPAP); + if (!mach) return STR_MessageBox_Error; - CPAPMode mode = (CPAPMode)(int)qRound(settings_wavg(CPAP_Mode)); - if (mode == MODE_CPAP) { - return QObject::tr("Fixed"); - } else if (mode == MODE_APAP) { - return QObject::tr("Auto"); - } else if (mode == MODE_BILEVEL_FIXED ) { - return QObject::tr("Fixed Bi-Level"); - } else if (mode == MODE_BILEVEL_AUTO_FIXED_PS) { - return QObject::tr("Auto Bi-Level (Fixed PS)"); - } else if (mode == MODE_BILEVEL_AUTO_VARIABLE_PS) { - return QObject::tr("Auto Bi-Level (Variable PS)"); - } else if (mode == MODE_ASV) { - return QObject::tr("ASV Fixed EPAP"); - } else if (mode == MODE_ASV_VARIABLE_EPAP) { - return QObject::tr("ASV Variable EPAP"); - } - return STR_TR_Unknown; + CPAPLoader * loader = qobject_cast(mach->loader()); + + ChannelID modechan = loader->CPAPModeChannel(); + + schema::Channel & chan = schema::channel[modechan]; + + int mode = (CPAPMode)(int)qRound(settings_wavg(modechan)); + + return chan.option(mode); + + +// if (mode == MODE_CPAP) { +// return QObject::tr("Fixed"); +// } else if (mode == MODE_APAP) { +// return QObject::tr("Auto"); +// } else if (mode == MODE_BILEVEL_FIXED ) { +// return QObject::tr("Fixed Bi-Level"); +// } else if (mode == MODE_BILEVEL_AUTO_FIXED_PS) { +// return QObject::tr("Auto Bi-Level (Fixed PS)"); +// } else if (mode == MODE_BILEVEL_AUTO_VARIABLE_PS) { +// return QObject::tr("Auto Bi-Level (Variable PS)"); +// } else if (mode == MODE_ASV) { +// return QObject::tr("ASV Fixed EPAP"); +// } else if (mode == MODE_ASV_VARIABLE_EPAP) { +// return QObject::tr("ASV Variable EPAP"); +// } +// return STR_TR_Unknown; } QString Day::getPressureRelief() diff --git a/sleepyhead/SleepLib/day.h b/sleepyhead/SleepLib/day.h index d23c630d..cb663d63 100644 --- a/sleepyhead/SleepLib/day.h +++ b/sleepyhead/SleepLib/day.h @@ -46,6 +46,12 @@ class Day //! \brief Add Session to this Day object (called during Load) void addSession(Session *s); + EventDataType rangeCount(ChannelID code, qint64 st, qint64 et); + EventDataType rangeSum(ChannelID code, qint64 st, qint64 et); + EventDataType rangeAvg(ChannelID code, qint64 st, qint64 et); + EventDataType rangeWavg(ChannelID code, qint64 st, qint64 et); + EventDataType rangePercentile(ChannelID code, float p, qint64 st, qint64 et); + //! \brief Returns the count of all this days sessions' events for this day EventDataType count(ChannelID code); diff --git a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp index 9e7b4ba4..641933ec 100644 --- a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp @@ -31,7 +31,7 @@ QHash > Resmed_Model_Map; const QString STR_UnknownModel = "Resmed S9 ???"; -ChannelID RMS9_EPR, RMS9_EPRLevel; +ChannelID RMS9_EPR, RMS9_EPRLevel, RMS9_Mode; // Return the model name matching the supplied model number. @@ -226,8 +226,11 @@ void ResmedLoader::ParseSTR(Machine *mach, QStringList strfiles) if ((sig = str.lookupSignal(CPAP_Mode))) { int mod = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + R.rms9_mode = mod; - if (mod >= 8) { // mod 8 == vpap adapt variable epap + if (mod == 11) { + mode = MODE_APAP; + } else if (mod >= 8) { // mod 8 == vpap adapt variable epap mode = MODE_ASV_VARIABLE_EPAP; } else if (mod >= 7) { // mod 7 == vpap adapt mode = MODE_ASV; @@ -245,6 +248,12 @@ void ResmedLoader::ParseSTR(Machine *mach, QStringList strfiles) } R.mode = mode; + if ((mod == 0) && (sig = str.lookupLabel("S.C.StartPress"))) { + R.ramp_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if (((mod == 1) || (mod == 11)) && (sig = str.lookupLabel("S.AS.StartPress"))) { + R.ramp_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } } @@ -318,16 +327,15 @@ void ResmedLoader::ParseSTR(Machine *mach, QStringList strfiles) if (!haveipap) { - R.ipap = R.min_ipap = R.max_ipap = R.max_epap + R.max_ps; } -// if (mode == MODE_ASV_VARIABLE_EPAP) { -// // ResMed reuses this code on 36037.. the dummies :( -// } else if (mode == MODE_ASV) { -// if (!haveipap) { -// R.ipap = R.min_ipap = R.max_ipap = R.max_epap + R.max_ps; -// } -// } + if (mode == MODE_ASV_VARIABLE_EPAP) { + R.min_ipap = R.min_epap + R.min_ps; + R.max_ipap = R.max_epap + R.max_ps; + } else if (mode == MODE_ASV) { + R.min_ipap = R.epap + R.min_ps; + R.max_ipap = R.epap + R.max_ps; + } EventDataType epr = -1, epr_level = -1; if ((sig = str.lookupSignal(RMS9_EPR))) { @@ -356,6 +364,8 @@ void ResmedLoader::ParseSTR(Machine *mach, QStringList strfiles) } } + + if ((sig = str.lookupLabel("AHI"))) { R.ahi = EventDataType(sig->data[rec]) * sig->gain + sig->offset; } @@ -372,6 +382,53 @@ void ResmedLoader::ParseSTR(Machine *mach, QStringList strfiles) R.cai = EventDataType(sig->data[rec]) * sig->gain + sig->offset; } + + if ((sig = str.lookupLabel("S.RampTime"))) { + R.s_RampTime = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.RampEnable"))) { + R.s_RampEnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.EPR.ClinEnable"))) { + R.s_EPR_ClinEnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.EPR.EPREnable"))) { + R.s_EPREnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + + if ((sig = str.lookupLabel("S.ABFilter"))) { + R.s_ABFilter = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + + if ((sig = str.lookupLabel("S.ClimateControl"))) { + R.s_ClimateControl = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + + if ((sig = str.lookupLabel("S.Mask"))) { + R.s_Mask = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.PtAccess"))) { + R.s_PtAccess = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.SmartStart"))) { + R.s_SmartStart = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.HumEnable"))) { + R.s_HumEnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.HumLevel"))) { + R.s_HumLevel = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.TempEnable"))) { + R.s_TempEnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.Temp"))) { + R.s_Temp = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + if ((sig = str.lookupLabel("S.Tube"))) { + R.s_Tube = EventDataType(sig->data[rec]) * sig->gain + sig->offset; + } + laston = ontime; QDateTime dontime = QDateTime::fromTime_t(ontime); @@ -384,6 +441,8 @@ void ResmedLoader::ParseSTR(Machine *mach, QStringList strfiles) //QDateTime dofftime = QDateTime::fromTime_t(offtime); //qDebug() << "Mask on" << dontime << "Mask off" << dofftime; } + + // Wait... ResMed has a DST bug here...should I be replicating it by using multiples of 86400 seconds? dt = dt.addDays(1); } } @@ -675,18 +734,36 @@ void ResmedImport::run() } loader->saveMutex.unlock(); - if (!group.EVE.isEmpty()) { - loader->LoadEVE(sess, group.EVE); + Q_FOREACH(QString file, files[EDF_PLD]) { + loader->LoadPLD(sess, file); +#ifdef SESSION_DEBUG + sess->session_files.append(file); +#endif } - if (!group.BRP.isEmpty()) { - loader->LoadBRP(sess, group.BRP); + Q_FOREACH(QString file, files[EDF_BRP]) { + loader->LoadBRP(sess, file); +#ifdef SESSION_DEBUG + sess->session_files.append(file); +#endif } - if (!group.PLD.isEmpty()) { - loader->LoadPLD(sess, group.PLD); + Q_FOREACH(QString file, files[EDF_SAD]) { + loader->LoadSAD(sess, file); +#ifdef SESSION_DEBUG + sess->session_files.append(file); +#endif } - if (!group.SAD.isEmpty()) { - loader->LoadSAD(sess, group.SAD); + + // Load annotations afterwards so durations are set correctly + Q_FOREACH(QString file, files[EDF_CSL]) { +// loader->LoadCSL(sess, file); } + Q_FOREACH(QString file, files[EDF_EVE]) { + loader->LoadEVE(sess, file); +#ifdef SESSION_DEBUG + sess->session_files.append(file); +#endif + } + if (sess->first() == 0) { // loader->saveMutex.lock(); @@ -745,8 +822,13 @@ void ResmedImport::run() // Save maskon time in session setting so we can use it later to avoid doubleups. sess->settings[RMS9_MaskOnTime] = R.maskon; +#ifdef SESSION_DEBUG + sess->session_files.append("STR.edf"); +#endif + if (R.mode >= 0) { sess->settings[CPAP_Mode] = R.mode; + sess->settings[RMS9_Mode] = R.rms9_mode; if (R.mode == MODE_CPAP) { if (R.set_pressure >= 0) { sess->settings[CPAP_Pressure] = R.set_pressure; @@ -832,21 +914,27 @@ ResmedLoader::~ResmedLoader() void ResmedImportStage2::run() { + if (R.maskon == R.maskoff) return; Session * sess = new Session(mach, 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->setSummaryOnly(true); - +#ifdef SESSION_DEBUG + sess->session_files.append("STR.edf"); +#endif sess->SetChanged(true); // First take the settings if (R.mode >= 0) { sess->settings[CPAP_Mode] = R.mode; + sess->settings[RMS9_Mode] = R.rms9_mode; + if (R.mode == MODE_CPAP) { if (R.set_pressure >= 0) { sess->settings[CPAP_Pressure] = R.set_pressure; @@ -1079,9 +1167,143 @@ EDFType lookupEDFType(QString text) } else return EDF_UNKNOWN; } +// Pretend to parse the EVE file to get the duration out of it. +int PeekEVE(const QString & path, quint32 &start, quint32 &end) +{ + EDFParser edf(path); + if (!edf.Parse()) + return -1; + + QString t; + + double duration; + char *data; + char c; + long pos; + bool sign, ok; + double d; + double tt; + + int recs = 0; + int goodrecs = 0; + + // Notes: Event records have useless duration record. + + start = edf.startdate / 1000L; + // Process event annotation records + for (int s = 0; s < edf.GetNumSignals(); s++) { + recs = edf.edfsignals[s].nr * edf.GetNumDataRecords() * 2; + + data = (char *)edf.edfsignals[s].data; + pos = 0; + tt = edf.startdate; + duration = 0; + + while (pos < recs) { + c = data[pos]; + + if ((c != '+') && (c != '-')) { + break; + } + + if (data[pos++] == '+') { sign = true; } + else { sign = false; } + + t = ""; + c = data[pos]; + + do { + t += c; + pos++; + c = data[pos]; + } while ((c != 20) && (c != 21)); // start code + + d = t.toDouble(&ok); + + if (!ok) { + qDebug() << "Faulty EDF EVE file " << edf.filename; + break; + } + + if (!sign) { d = -d; } + + tt = edf.startdate + qint64(d * 1000.0); + + duration = 0; + // First entry + + if (data[pos] == 21) { + pos++; + // get duration. + t = ""; + + do { + t += data[pos]; + pos++; + } while ((data[pos] != 20) && (pos < recs)); // start code + + duration = t.toDouble(&ok); + + if (!ok) { + qDebug() << "Faulty EDF EVE file (at %" << pos << ") " << edf.filename; + break; + } + } + end = (tt / 1000.0); + + while ((data[pos] == 20) && (pos < recs)) { + t = ""; + pos++; + + if (data[pos] == 0) { + break; + } + + if (data[pos] == 20) { + pos++; + break; + } + + do { + t += tolower(data[pos++]); + } while ((data[pos] != 20) && (pos < recs)); // start code + + if (!t.isEmpty() && (t!="recording starts")) { + goodrecs++; +// if (matchSignal(CPAP_Obstructive, t)) { +// } else if (matchSignal(CPAP_Hypopnea, t)) { +// } else if (matchSignal(CPAP_Apnea, t)) { +// } else if (matchSignal(CPAP_ClearAirway, t)) { +// } else { +// if (t != "recording starts") { +// qDebug() << "Unobserved ResMed annotation field: " << t; +// } +// } + } + + if (pos >= recs) { + qDebug() << "Short EDF EVE file" << edf.filename; + break; + } + + // pos++; + } + + while ((data[pos] == 0) && (pos < recs)) { pos++; } + + if (pos >= recs) { break; } + } + + } + return goodrecs; +} + + // Looks inside an EDF or EDF.gz and grabs the start and duration EDFduration getEDFDuration(QString filename) { + QString ext = filename.section("_", -1).section(".",0,0).toUpper(); + bool ok1, ok2; int num_records; @@ -1169,31 +1391,34 @@ EDFduration getEDFDuration(QString filename) quint32 end = start + rec_duration * num_records; QString filedate = filename.section("/",-1).section("_",0,1); - QString ext = filename.section("_", -1).section(".",0,0).toUpper(); QDateTime dt2 = QDateTime::fromString(filedate, "yyyyMMdd_hhmmss"); quint32 st2 = dt2.toTime_t(); start = qMin(st2, start); - if (end < start) end = qMax(st2, start); // This alone should really cover the EVE.EDF condition + if (end < start) end = qMax(st2, start); -// if (ext == "EVE") { -// // This is an unavoidable kludge, because there genuinely is no duration given for EVE files. -// // It could partially be avoided by parsing the EDF annotations completely, but on days with no events, this would be pointless. + if (ext == "EVE") { + // S10 Forces us to parse EVE files to find their real durations + quint32 en2; -// // Add some seconds to make sure some overlap happens with related sessions. + // Have to get the actual duration of the EVE file by parsing the annotations. :( + int recs = PeekEVE(filename, st2, en2); + if (recs > 0) { + start = qMin(st2, start); + end = qMax(en2, end); + EDFduration dur(start, end, filename); -// // ************** Be cautious with this value ************** + dur.type = lookupEDFType(ext.toUpper()); -// // A Firmware bug causes (perhaps with failing SD card) sessions to sometimes take a long time to write, and it can screw this up -// // I've really got no way of detecting the other condition.. I can either have one or the other. - -// // Wait... EVE and BRP start at the same time.. that should be enough to counter overlaps! -// end += 1; -// } - - if ((end - start) < 10) end = start + 10; + return dur; + } else { + // empty EVE file, don't give a crap about it... + return EDFduration(0, 0, filename); + } + // A Firmware bug causes (perhaps with failing SD card) sessions to sometimes take a long time to write + } EDFduration dur(start, end, filename); @@ -1324,12 +1549,13 @@ int ResmedLoader::scanFiles(Machine * mach, QString datalog_path) QString fullname = fi.canonicalFilePath(); // Peek inside the EDF file and get the EDFDuration record for the session matching that follows + EDFduration dur = getEDFDuration(fullname); + dur.filename = filename; - QMap::iterator it = newfiles.insert(filename, getEDFDuration(fullname)); - EDFduration *dur = &it.value(); - dur->filename = filename; - - filesbytype[dur->type].append(dur); + if (dur.start != dur.end) { // make sure empty EVE's are skipped + QMap::iterator it = newfiles.insert(filename, getEDFDuration(fullname)); + filesbytype[dur.type].append(&it.value()); + } } } @@ -1337,30 +1563,28 @@ int ResmedLoader::scanFiles(Machine * mach, QString datalog_path) EDForder.push_back(EDF_PLD); EDForder.push_back(EDF_BRP); EDForder.push_back(EDF_SAD); - EDForder.push_back(EDF_EVE); EDForder.push_back(EDF_CSL); - for (int i=0; i<3; i++) { + for (int i=0; i<4; i++) { EDFType basetype = EDForder.takeFirst(); // Process PLD files QList & LIST = filesbytype[basetype]; - int pld_size = LIST.size(); - for (int f=0; f < pld_size; ++f) { + int base_size = LIST.size(); + for (int f=0; f < base_size; ++f) { const EDFduration * dur = LIST.at(f); quint32 start = dur->start; if (start == 0) continue; quint32 end = dur->end; - QHash grp; - grp[EDF_PLD] = create_backups ? backup(dur->path, backup_path) : dur->path;; + QHash grp; + grp[basetype].append(create_backups ? backup(dur->path, backup_path) : dur->path); QStringList files; files.append(dur->filename); - for (int o=0; o::iterator list_end = EDF_list.end(); for (item = EDF_list.begin(); item != list_end; ++item) { const EDFduration * dur2 = *item; + if (dur2->start == 0) continue; // Do the sessions Overlap? if ((start < dur2->end) && ( dur2->start < end)) { @@ -1377,18 +1602,41 @@ int ResmedLoader::scanFiles(Machine * mach, QString datalog_path) files.append(dur2->filename); - grp[type] = create_backups ? backup(dur2->path, backup_path) : dur2->path; + grp[type].append(create_backups ? backup(dur2->path, backup_path) : dur2->path); filesbytype[type].erase(item); - break; } } } + + // EVE annotation files can cover multiple sessions + QList & EDF_list = filesbytype[EDF_EVE]; + QList::iterator item; + QList::iterator list_end = EDF_list.end(); + for (item = EDF_list.begin(); item != list_end; ++item) { + const EDFduration * dur2 = *item; + if (dur2->start == 0) continue; + + // Do the sessions Overlap? + if ((start < dur2->end) && ( dur2->start < end)) { +// start = qMin(start, dur2->start); +// end = qMax(end, dur2->end); + + files.append(dur2->filename); + + grp[EDF_EVE].append(create_backups ? backup(dur2->path, backup_path) : dur2->path); + } + } + + + if (mach->SessionExists(start) == nullptr) { - EDFGroup group(grp[EDF_BRP], grp[EDF_EVE], grp[EDF_PLD], grp[EDF_SAD], grp[EDF_CSL]); - queTask(new ResmedImport(this, start, group, mach)); - for (int i=0; i 0) { + queTask(new ResmedImport(this, start, grp, mach)); + for (int i=0; i::iterator it; - QMap::iterator end = strsess.end(); - QHash::iterator sessit; - QHash::iterator sessend = m->sessionlist.end();; int size = m->sessionlist.size(); int cnt=0; Session * sess; - // 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(); - QMap::iterator e = strsess.find(key); - if (e != end) { - strsess.erase(e); + // 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(); + +// // Ugly.. need to check sessions overlaps.. + +// QMap::iterator e = strsess.find(key); +// if (e != end) { +// strsess.erase(e); +// } +// } +/// + + QHash::iterator sessit; + QHash::iterator sessend = m->sessionlist.end();; + + QMap::iterator sit; + QMap::iterator ns_end = new_sessions.end(); + + + QMap::iterator it; + QMap::iterator end = strsess.end(); + + QList strlist; + for (it = strsess.begin(); it != end; ++it) { + STRRecord & R = it.value(); + quint32 s1 = R.maskon; + quint32 e1 = R.maskoff; + bool fnd = false; + for (sessit = m->sessionlist.begin(); sessit != sessend; ++sessit) { + sess = sessit.value(); + quint32 s2 = sess->session(); + quint32 e2 = s2 + (sess->length() / 1000L); + + if ((s1 < e2) && (s2 < e1)) { + strlist.push_back(it.key()); + fnd = true; + break; + } + } + if (!fnd) for (sit = new_sessions.begin(); sit != ns_end; ++sit) { + sess = sit.value(); + quint32 s2 = sess->session(); + quint32 e2 = s2 + (sess->length() / 1000L); + + if ((s1 < e2) && (s2 < e1)) { + strlist.push_back(it.key()); + fnd = true; + break; + } + } } + for (int i=0; iupdateFirst(tt); + // sess->updateFirst(tt); duration = 0; while (pos < recs) { @@ -2040,18 +2336,19 @@ bool ResmedLoader::LoadEVE(Session *sess, const QString & path) if (!t.isEmpty()) { if (matchSignal(CPAP_Obstructive, t)) { - OA->AddEvent(tt, duration); + + if (sess->checkInside(tt)) OA->AddEvent(tt, duration); } else if (matchSignal(CPAP_Hypopnea, t)) { - HY->AddEvent(tt, duration + 10); // Only Hyponea's Need the extra duration??? + if (sess->checkInside(tt)) HY->AddEvent(tt, duration + 10); // Only Hyponea's Need the extra duration??? } else if (matchSignal(CPAP_Apnea, t)) { - UA->AddEvent(tt, duration); + if (sess->checkInside(tt)) UA->AddEvent(tt, duration); } else if (matchSignal(CPAP_ClearAirway, t)) { // Not all machines have it, so only create it when necessary.. if (!CA) { if (!(CA = sess->AddEventList(CPAP_ClearAirway, EVL_Event))) { return false; } } - CA->AddEvent(tt, duration); + if (sess->checkInside(tt)) CA->AddEvent(tt, duration); } else { if (t != "recording starts") { qDebug() << "Unobserved ResMed annotation field: " << t; @@ -2072,7 +2369,7 @@ bool ResmedLoader::LoadEVE(Session *sess, const QString & path) if (pos >= recs) { break; } } - sess->updateLast(tt); + // sess->updateLast(tt); } return true; @@ -2366,7 +2663,6 @@ bool ResmedLoader::LoadPLD(Session *sess, const QString & path) ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); } else if (matchSignal(CPAP_IPAP, es.label)) { code = CPAP_IPAP; - sess->settings[CPAP_Mode] = MODE_BILEVEL_FIXED; es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); @@ -2564,6 +2860,8 @@ void ResInitModelMap() resmed_codes[RMS9_SetPressure].push_back("Inställt tryck"); resmed_codes[RMS9_SetPressure].push_back("Inställt tryck"); resmed_codes[RMS9_EPR].push_back("EPR"); + resmed_codes[RMS9_EPR].push_back("S.EPR.EPRType"); + resmed_codes[RMS9_EPR].push_back("\xE5\x91\xBC\xE6\xB0\x94\xE9\x87\x8A\xE5\x8E\x8B\x28\x45\x50"); // Chinese resmed_codes[RMS9_EPRLevel].push_back("EPR Level"); resmed_codes[RMS9_EPRLevel].push_back("EPR-Stufe"); @@ -2621,6 +2919,7 @@ void ResInitModelMap() } +ChannelID ResmedLoader::CPAPModeChannel() { return RMS9_Mode; } ChannelID ResmedLoader::PresReliefMode() { return RMS9_EPR; } ChannelID ResmedLoader::PresReliefLevel() { return RMS9_EPRLevel; } @@ -2628,6 +2927,28 @@ void ResmedLoader::initChannels() { using namespace schema; Channel * chan = nullptr; + channel.add(GRP_CPAP, chan = new Channel(RMS9_Mode = 0xe203, SETTING, MT_CPAP, SESSION, + "RMS9_Mode", + QObject::tr("Mode"), + QObject::tr("CPAP Mode"), + QObject::tr("Mode"), + "", LOOKUP, Qt::green)); + + chan->addOption(0, QObject::tr("CPAP")); + chan->addOption(1, QObject::tr("APAP")); + chan->addOption(2, QObject::tr("VPAP-T")); + chan->addOption(3, QObject::tr("VPAP-S")); + chan->addOption(4, QObject::tr("VPAP-S/T")); + chan->addOption(5, QObject::tr("??")); + chan->addOption(6, QObject::tr("VPAPauto")); + chan->addOption(7, QObject::tr("ASV")); + chan->addOption(8, QObject::tr("ASVAuto")); + chan->addOption(9, QObject::tr("???")); + chan->addOption(10, QObject::tr("???")); + chan->addOption(11, QObject::tr("Auto for Her")); + + + channel.add(GRP_CPAP, chan = new Channel(RMS9_EPR = 0xe201, SETTING, MT_CPAP, SESSION, "EPR", QObject::tr("EPR"), QObject::tr("ResMed Exhale Pressure Relief"), diff --git a/sleepyhead/SleepLib/loader_plugins/resmed_loader.h b/sleepyhead/SleepLib/loader_plugins/resmed_loader.h index 1ad13f94..49e9fdf5 100644 --- a/sleepyhead/SleepLib/loader_plugins/resmed_loader.h +++ b/sleepyhead/SleepLib/loader_plugins/resmed_loader.h @@ -109,6 +109,7 @@ struct STRRecord maskdur = 0; maskevents = -1; mode = -1; + rms9_mode = -1; set_pressure = -1; epap = -1; max_pressure = -1; @@ -136,6 +137,26 @@ struct STRRecord leakmax = -1; leakgain = 0; + s_RampTime = -1; + s_RampEnable = -1; + s_EPR_ClinEnable = -1; + s_EPREnable = -1; + + s_PtAccess = -1; + s_ABFilter = -1; + s_Mask = -1; + s_Tube = -1; + s_ClimateControl = -1; + s_HumEnable = -1; + s_HumLevel = -1; + s_TempEnable = -1; + s_Temp = -1; + s_SmartStart = -1; + + ramp_pressure = -1; + + + date=QDate(); } STRRecord(const STRRecord & copy) { @@ -144,6 +165,7 @@ struct STRRecord maskdur = copy.maskdur; maskevents = copy.maskevents; mode = copy.mode; + rms9_mode = copy.rms9_mode; set_pressure = copy.set_pressure; epap = copy.epap; max_pressure = copy.max_pressure; @@ -169,12 +191,26 @@ struct STRRecord leak95 = copy.leak95; leakmax = copy.leakmax; leakgain = copy.leakgain; + + s_SmartStart = copy.s_SmartStart; + s_PtAccess = copy.s_PtAccess; + s_ABFilter = copy.s_ABFilter; + s_Mask = copy.s_Mask; + + s_Tube = copy.s_Tube; + s_ClimateControl = copy.s_ClimateControl; + s_HumEnable = copy.s_HumEnable; + s_HumLevel = copy.s_HumLevel; + s_TempEnable = copy.s_TempEnable; + s_Temp = copy.s_Temp; + ramp_pressure = copy.ramp_pressure; } quint32 maskon; quint32 maskoff; EventDataType maskdur; EventDataType maskevents; EventDataType mode; + EventDataType rms9_mode; EventDataType set_pressure; EventDataType max_pressure; EventDataType min_pressure; @@ -199,7 +235,25 @@ struct STRRecord EventDataType leak95; EventDataType leakmax; EventDataType leakgain; + EventDataType ramp_pressure; QDate date; + + EventDataType s_RampTime; + int s_RampEnable; + int s_EPR_ClinEnable; + int s_EPREnable; + + int s_PtAccess; + int s_ABFilter; + int s_Mask; + int s_Tube; + int s_ClimateControl; + int s_HumEnable; + EventDataType s_HumLevel; + int s_TempEnable; + EventDataType s_Temp; + int s_SmartStart; + }; @@ -306,14 +360,14 @@ struct EDFGroup { class ResmedImport:public ImportTask { public: - ResmedImport(ResmedLoader * l, SessionID s, EDFGroup g, Machine * m): loader(l), sessionid(s), group(g), mach(m) {} + ResmedImport(ResmedLoader * l, SessionID s, QHash f, Machine * m): loader(l), sessionid(s), files(f), mach(m) {} virtual ~ResmedImport() {} virtual void run(); protected: ResmedLoader * loader; SessionID sessionid; - EDFGroup group; + QHash files; Machine * mach; }; @@ -396,6 +450,8 @@ class ResmedLoader : public CPAPLoader virtual ChannelID PresReliefMode(); virtual ChannelID PresReliefLevel(); + virtual ChannelID CPAPModeChannel(); + //////////////////////////////////////////////////////////////////////////////////////////////////////////// diff --git a/sleepyhead/SleepLib/machine_loader.h b/sleepyhead/SleepLib/machine_loader.h index 7b2ebfae..4d28f0d5 100644 --- a/sleepyhead/SleepLib/machine_loader.h +++ b/sleepyhead/SleepLib/machine_loader.h @@ -125,6 +125,7 @@ public: virtual ChannelID PresReliefLevel() { return NoChannel; } virtual ChannelID HumidifierConnected() { return NoChannel; } virtual ChannelID HumidifierLevel() { return CPAP_HumidSetting; } + virtual ChannelID CPAPModeChannel() { return CPAP_Mode; } virtual void initChannels() {} }; diff --git a/sleepyhead/SleepLib/session.h b/sleepyhead/SleepLib/session.h index 69350ea2..9cbe1813 100644 --- a/sleepyhead/SleepLib/session.h +++ b/sleepyhead/SleepLib/session.h @@ -11,6 +11,8 @@ #ifndef SESSION_H #define SESSION_H +#define SESSION_DEBUG + #include #include #include @@ -36,6 +38,10 @@ class Session Session(Machine *, SessionID); virtual ~Session(); + inline bool checkInside(qint64 time) { + return ((time >= s_first) && (time <= s_last)); + } + //! \brief Stores the session in the directory supplied by path bool Store(QString path); @@ -338,6 +344,10 @@ class Session const QString & eventFile() { return s_eventfile; } +#if defined(SESSION_DEBUG) + QStringList session_files; +#endif + protected: SessionID s_session; @@ -356,8 +366,6 @@ protected: // for debugging bool destroyed; - - }; diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index bc061923..2f5801fe 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -59,7 +59,7 @@ void Daily::setSidebarVisible(bool visible) { QList a; - int panel_width = visible ? 350 : 0; + int panel_width = visible ? 370 : 0; a.push_back(panel_width); a.push_back(this->width() - panel_width); ui->splitter_2->setStretchFactor(1,1); @@ -160,7 +160,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared) // gGraph * SG; // graphlist[STR_GRAPH_DailySummary] = SG = new gGraph(STR_GRAPH_DailySummary, GraphView, QObject::tr("Summary"), QObject::tr("Summary of this daily information"), default_height); // SG->AddLayer(new gLabelArea(nullptr),LayerLeft,gYAxis::Margin); -// SG->AddLayer(AddCPAP(new gDailySummary())); +// SG->AddLayer(new gDailySummary()); graphlist[STR_GRAPH_SleepFlags] = SF = new gGraph(STR_GRAPH_SleepFlags, GraphView, STR_TR_EventFlags, STR_TR_EventFlags, default_height); SF->setPinned(true); @@ -911,6 +911,7 @@ QString Daily::getSessionInformation(Day * day) sess->settings[SESSION_ENABLED]=true; } bool b=sess->settings[SESSION_ENABLED].toBool(); + html+=QString("%2" "" "" @@ -927,6 +928,11 @@ QString Daily::getSessionInformation(Day * day) .arg(fd.date().toString(Qt::SystemLocaleShortDate)) .arg(fd.toString("HH:mm:ss")) .arg(ld.toString("HH:mm:ss")); +#ifdef SESSION_DEBUG + for (int i=0; i< sess->session_files.size(); ++i) { + html+=QString("%1").arg(sess->session_files[i].section("/",-1)); + } +#endif } } @@ -962,10 +968,20 @@ QString Daily::getMachineSettings(Day * day) { } QMap first; + CPAPLoader * loader = qobject_cast(cpap->loader()); + + ChannelID cpapmode = loader->CPAPModeChannel(); + schema::Channel & chan = schema::channel[cpapmode]; + first[cpapmode] = QString("%1%2%3") + .arg(schema::channel[cpapmode].label()) + .arg(schema::channel[cpapmode].description()) + .arg(day->getCPAPMode()); + + if (sess) for (; it != it_end; ++it) { ChannelID code = it.key(); - if ((code <= 1) || (code == RMS9_MaskOnTime)) continue; + if ((code <= 1) || (code == RMS9_MaskOnTime) || (code == CPAP_Mode) || (code == cpapmode)) continue; schema::Channel & chan = schema::channel[code]; @@ -986,8 +1002,7 @@ QString Daily::getMachineSettings(Day * day) { .arg(data); - if ((code == CPAP_Mode) - || (code == CPAP_IPAP) + if ((code == CPAP_IPAP) || (code == CPAP_EPAP) || (code == CPAP_IPAPHi) || (code == CPAP_EPAPHi) @@ -1005,7 +1020,7 @@ QString Daily::getMachineSettings(Day * day) { } } - ChannelID order[] = { CPAP_Mode, CPAP_Pressure, CPAP_PressureMin, CPAP_PressureMax, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_PS, CPAP_PSMin, CPAP_PSMax }; + ChannelID order[] = { cpapmode, CPAP_Pressure, CPAP_PressureMin, CPAP_PressureMax, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_PS, CPAP_PSMin, CPAP_PSMax }; int os = sizeof(order) / sizeof(ChannelID); for (int i=0 ;i < os; ++i) { if (first.contains(order[i])) html += first[order[i]]; @@ -1955,18 +1970,35 @@ void Daily::on_LineCursorUpdate(double time) void Daily::on_RangeUpdate(double minx, double maxx) { -// static qint64 last_minx = 0; -// static qint64 last_maxx = 0; - - //if ((last_minx != minx) || (last_maxx != maxx)) { if (minx > 1) { dateDisplay->setText(GraphView->getRangeString()); } else { dateDisplay->setText(QString(GraphView->emptyText())); } - //} -// last_minx=minx; -// last_maxx=maxx; + +/* // Delay render some stats... + Day * day = GraphView->day(); + if (day) { + QTime time; + time.start(); + QList list = day->getSortedMachineChannels(schema::WAVEFORM); + for (int i=0; i< list.size();i++) { + schema::Channel & chan = schema::channel[list.at(i)]; + ChannelID code = chan.id(); + if (!day->channelExists(code)) continue; + float avg = day->rangeAvg(code, minx, maxx); + float wavg = day->rangeWavg(code, minx, maxx); + float median = day->rangePercentile(code, 0.5, minx, maxx); + float p90 = day->rangePercentile(code, 0.9, minx, maxx); +// qDebug() << chan.label() +// << "AVG=" << avg +// << "WAVG=" << wavg; + // << "MED" << median + // << "90%" << p90; + } + + qDebug() << time.elapsed() << "ms"; + }*/ } diff --git a/sleepyhead/overview.cpp b/sleepyhead/overview.cpp index e9919482..57dfe388 100644 --- a/sleepyhead/overview.cpp +++ b/sleepyhead/overview.cpp @@ -381,8 +381,8 @@ gGraph *Overview::createGraph(QString code, QString name, QString units, YTicker void Overview::on_LineCursorUpdate(double time) { if (time > 1) { - QDateTime dt = QDateTime::fromMSecsSinceEpoch(time); - QString txt = dt.toString("dd MMM yyyy"); + QDateTime dt = QDateTime::fromMSecsSinceEpoch(time,Qt::UTC); + QString txt = dt.toString("dd MMM yyyy (dddd)"); dateLabel->setText(txt); } else dateLabel->setText(QString(GraphView->emptyText())); }