From c9b774a47acfe2e1da14e68a4cf206f060eba553 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Sat, 27 Feb 2016 13:37:56 +1000 Subject: [PATCH] Dreamstation 0x14 flag is missing hypopnea, change CSR to PB on P.R. machines --- .../SleepLib/loader_plugins/prs1_loader.cpp | 28 ++++++++++------- sleepyhead/SleepLib/machine_common.cpp | 2 +- sleepyhead/SleepLib/machine_common.h | 2 +- sleepyhead/SleepLib/schema.cpp | 8 ++++- sleepyhead/daily.cpp | 4 +-- sleepyhead/docs/release_notes.html | 10 ++++--- sleepyhead/reports.cpp | 3 ++ sleepyhead/statistics.cpp | 30 +++++++++++++++++++ 8 files changed, 67 insertions(+), 20 deletions(-) diff --git a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp index 0215c742..2dc78baf 100644 --- a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp @@ -730,7 +730,7 @@ bool PRS1Import::ParseF5Events() { ChannelID Codes[] = { PRS1_00, PRS1_01, CPAP_Pressure, CPAP_EPAP, CPAP_PressurePulse, CPAP_Obstructive, - CPAP_ClearAirway, CPAP_Hypopnea, PRS1_08, CPAP_FlowLimit, PRS1_0A, CPAP_CSR, + CPAP_ClearAirway, CPAP_Hypopnea, PRS1_08, CPAP_FlowLimit, PRS1_0A, CPAP_PB, PRS1_0C, CPAP_VSnore, PRS1_0E, PRS1_0F, CPAP_LargeLeak, // Large leak apparently CPAP_LeakTotal, PRS1_12 @@ -741,7 +741,7 @@ bool PRS1Import::ParseF5Events() EventList *OA = session->AddEventList(CPAP_Obstructive, EVL_Event); EventList *HY = session->AddEventList(CPAP_Hypopnea, EVL_Event); - EventList *CSR = session->AddEventList(CPAP_CSR, EVL_Event); + EventList *PB = session->AddEventList(CPAP_PB, EVL_Event); EventList *LEAK = session->AddEventList(CPAP_LeakTotal, EVL_Event); EventList *LL = session->AddEventList(CPAP_LargeLeak, EVL_Event); EventList *SNORE = session->AddEventList(CPAP_Snore, EVL_Event); @@ -928,14 +928,14 @@ bool PRS1Import::ParseF5Events() //tt-=delta; tt -= qint64(data[1]) * 1000L; - if (!CSR) { - if (!(CSR = session->AddEventList(cpapcode, EVL_Event))) { - qDebug() << "!CSR addeventlist exit"; + if (!PB) { + if (!(PB = session->AddEventList(cpapcode, EVL_Event))) { + qDebug() << "!PB addeventlist exit"; return false; } } - CSR->AddEvent(tt, data[0]); + PB->AddEvent(tt, data[0]); break; case 0x0c: @@ -1086,7 +1086,7 @@ bool PRS1Import::ParseF0Events() EventList *OA = session->AddEventList(CPAP_Obstructive, EVL_Event); EventList *HY = session->AddEventList(CPAP_Hypopnea, EVL_Event); - EventList *CSR = session->AddEventList(CPAP_CSR, EVL_Event); + EventList *PB = session->AddEventList(CPAP_PB, EVL_Event); EventList *LEAK = session->AddEventList(CPAP_LeakTotal, EVL_Event); EventList *SNORE = session->AddEventList(CPAP_Snore, EVL_Event); @@ -1141,6 +1141,7 @@ bool PRS1Import::ParseF0Events() if (code != 0x12) { delta = buffer[pos + 1] << 8 | buffer[pos]; pos += 2; + t += qint64(delta) * 1000L; tt = t; } @@ -1232,8 +1233,10 @@ bool PRS1Import::ParseF0Events() break; case 0x04: // Pressure Pulse + data[0] = buffer[pos++]; + tt = t - (qint64(data[0]) * 1000L); - PP->AddEvent(t, buffer[pos++]); + PP->AddEvent(tt, data[0]); break; case 0x05: // RERA @@ -1297,7 +1300,7 @@ bool PRS1Import::ParseF0Events() //qDebug() << hex << data[0] << data[1] << data[2]; //session->AddEvent(new Event(t,cpapcode, 0, data, 3)); //tt-=data[1]*1000; - //session->AddEvent(new Event(t,CPAP_CSR, data, 2)); + //session->AddEvent(new Event(t,CPAP_PB, data, 2)); break; case 0x0f: // Cheyne Stokes Respiration @@ -1305,7 +1308,7 @@ bool PRS1Import::ParseF0Events() pos += 2; data[1] = buffer[pos++]; tt = t - qint64(data[1]) * 1000L; - CSR->AddEvent(tt, data[0]); + PB->AddEvent(tt, data[0]); break; case 0x10: // Large Leak data[0] = buffer[pos + 1] << 8 | buffer[pos]; @@ -1349,9 +1352,12 @@ bool PRS1Import::ParseF0Events() break; case 0x14: // DreamStation unknown code + data[0] = buffer[pos++]; tt = t - (qint64(data[0]) * 1000L); + HY->AddEvent(tt, data[0]); +// tt = t - (qint64(data[0]) * 1000L); - pos++; +// pos++; break; default: diff --git a/sleepyhead/SleepLib/machine_common.cpp b/sleepyhead/SleepLib/machine_common.cpp index c4eaecf6..e1310a33 100644 --- a/sleepyhead/SleepLib/machine_common.cpp +++ b/sleepyhead/SleepLib/machine_common.cpp @@ -15,7 +15,7 @@ ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAP CPAP_PS, CPAP_Mode, CPAP_AHI, CPAP_PressureMin, CPAP_PressureMax, CPAP_Ramp, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive, CPAP_Hypopnea, - CPAP_ClearAirway, CPAP_Apnea, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore, + CPAP_ClearAirway, CPAP_Apnea, CPAP_PB, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore, CPAP_VSnore2, CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_SensAwake, CPAP_FlowRate, CPAP_MaskPressure, CPAP_MaskPressureHi, diff --git a/sleepyhead/SleepLib/machine_common.h b/sleepyhead/SleepLib/machine_common.h index 77234aad..434f5049 100644 --- a/sleepyhead/SleepLib/machine_common.h +++ b/sleepyhead/SleepLib/machine_common.h @@ -139,7 +139,7 @@ extern ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CP CPAP_Mode, CPAP_AHI, CPAP_PressureMin, CPAP_PressureMax, CPAP_Ramp, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive, CPAP_Hypopnea, - CPAP_ClearAirway, CPAP_Apnea, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore, + CPAP_ClearAirway, CPAP_Apnea, CPAP_PB, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore, CPAP_VSnore2, CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_SensAwake, CPAP_FlowRate, CPAP_MaskPressure, CPAP_MaskPressureHi, diff --git a/sleepyhead/SleepLib/schema.cpp b/sleepyhead/SleepLib/schema.cpp index 12fe35b6..57482dbd 100644 --- a/sleepyhead/SleepLib/schema.cpp +++ b/sleepyhead/SleepLib/schema.cpp @@ -57,6 +57,7 @@ QHash Scopes; bool schema_initialized = false; void setOrders() { + schema::channel[CPAP_PB].setOrder(1); schema::channel[CPAP_CSR].setOrder(1); schema::channel[CPAP_Ramp].setOrder(2); schema::channel[CPAP_LargeLeak].setOrder(2); @@ -183,9 +184,14 @@ void init() // Flags schema::channel.add(GRP_CPAP, new Channel(CPAP_CSR = 0x1000, SPAN, MT_CPAP, SESSION, "CSR", QObject::tr("Cheyne Stokes Respiration"), - QObject::tr("An abnormal period of Periodic Breathing/Cheyne Stokes Respiration"), + QObject::tr("An abnormal period of Cheyne Stokes Respiration"), QObject::tr("CSR"), STR_UNIT_Percentage, DEFAULT, COLOR_CSR)); + schema::channel.add(GRP_CPAP, new Channel(CPAP_PB = 0x1028, SPAN, MT_CPAP, SESSION, "PB", + QObject::tr("Periodic Breathing"), + QObject::tr("An abnormal period of Periodic Breathing"), + QObject::tr("PB"), STR_UNIT_Percentage, DEFAULT, COLOR_CSR)); + schema::channel.add(GRP_CPAP, new Channel(CPAP_ClearAirway = 0x1001, FLAG, MT_CPAP, SESSION, "ClearAirway", QObject::tr("Clear Airway"), diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index 55be7593..a25db8f8 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -664,8 +664,8 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day) for (quint32 o=0;o

Release Notes

Greetings!

-

I know it's been a while, but here is new and improved SleepyHead build :)

- -

Warning: This is a test build, not all features are complete or will work 100% as intended yet!

+

Warning: This is a pre-release test build, not all features are complete or will work 100% as intended yet!

+

This one is specifically to test CMS50i on Mac, and nothing else.

+

Important: Back up your data folder manually, by making a copy the SleepyHeadData folder, as attempting to roll back later may break things.

+

DreamStation support is not ready yet, so please, I *don't* want feedback from DreamStation users testing this particular build.(You're of course welcome to try, but you are likely to be a little disappointed.)

+

Please don't share this with anyone else or reupload anywhere, it's not a public release.

Sleep Well, and good luck!

JediMark


Changes and fixes in v0.9.8-2 -
  • DreamStation support (Thanks for your assistance JediBob!) +
  • Initial DreamStation support (Thanks for your assistance JediBob!)
  • AirCurve pressure reporting glitches
  • Fix F&P Import hang
  • Pinch to zoom support
  • diff --git a/sleepyhead/reports.cpp b/sleepyhead/reports.cpp index 564ae4a8..ee82f041 100644 --- a/sleepyhead/reports.cpp +++ b/sleepyhead/reports.cpp @@ -210,6 +210,7 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date) float hours = day->hours(MT_CPAP); ahi /= hours; float csr = (100.0 / hours) * (day->sum(CPAP_CSR) / 3600.0); + float pb = (100.0 / hours) * (day->sum(CPAP_PB) / 3600.0); float uai = day->count(CPAP_Apnea) / hours; float oai = day->count(CPAP_Obstructive) / hours; float hi = (day->count(CPAP_ExP) + day->count(CPAP_Hypopnea)) / hours; @@ -270,6 +271,8 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date) 2); if (cpap->loaderName() == STR_MACH_PRS1) { + + // FIXME: PB / CSR are now separate flags stats += QObject::tr("REI=%1 VSI=%2 FLI=%3 PB/CSR=%4%%") .arg(rei, 0, 'f', 2).arg(vsi, 0, 'f', 2) .arg(fli, 0, 'f', 2).arg(csr, 0, 'f', 2); diff --git a/sleepyhead/statistics.cpp b/sleepyhead/statistics.cpp index f053a48c..71c0a276 100644 --- a/sleepyhead/statistics.cpp +++ b/sleepyhead/statistics.cpp @@ -1959,6 +1959,36 @@ void Statistics::UpdateRecordsBox() html += "
    "; } } + if (p_profile->hasChannel(CPAP_PB)) { + ahilist.clear(); + for (QDate date = first; date <= last; date = date.addDays(1)) { + Day * day = p_profile->GetGoodDay(date, MT_CPAP); + if (!day) continue; + + float leak = day->calcPON(CPAP_PB); + ahilist.insert(leak, date); + } + + if (ahilist.size() > (show_records * 2)) { + html += ""+QObject::tr("Worst PB")+"
    "; + + it = ahilist.end() - 1; + it_end = ahilist.begin(); + for (int i=0; (i < show_records) && (it != it_end); ++i, --it) { + + if (it.key() > 0) { + html += QString("").arg(it.value().toString(Qt::ISODate)) + +QObject::tr("Date: %1 PB: %2%").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
    "; + cnt++; + } + } + if (cnt == 0) { + html+= ""+QObject::tr("No PB on record")+"
    "; + } + + html += "
    "; + } + } } else { html += "
    "+QObject::tr("Want more information?")+"
    ";