From b530f27ca0ca3d1729dd1ddc603b7c9162b8ecf5 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Mon, 5 May 2014 04:02:41 +1000 Subject: [PATCH] ResMed loader cleanup, fix missing snore graph, versioning change. Versioning system just got changed to standard 3 segments Major, Minor and Patch level. Bumped to version 0.9.5 --- .../SleepLib/loader_plugins/resmed_loader.cpp | 672 +++++++----------- sleepyhead/docs/release_notes.html | 15 +- sleepyhead/mainwindow.cpp | 20 +- sleepyhead/sleepyhead.pro | 4 +- sleepyhead/version.h | 14 +- 5 files changed, 289 insertions(+), 436 deletions(-) diff --git a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp index 35071625..55ea1b57 100644 --- a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp @@ -28,8 +28,28 @@ #endif extern QProgressBar *qprogress; -QHash RMS9ModelMap; -QHash > resmed_codes; + +QHash > Resmed_Model_Map; + +const QString STR_UnknownModel = "Resmed S9 ???"; + +// Return the model name matching the supplied model number. +const QString & lookupModel(quint16 model) +{ + + auto end = Resmed_Model_Map.end(); + for (auto it = Resmed_Model_Map.begin(); it != end; ++it) { + QList & list = it.value(); + for (int i=0; i < list.size(); ++i) { + if (list.at(i) == model) { + return it.key(); + } + } + } + return STR_UnknownModel; +} + +QHash resmed_codes; const QString STR_ext_TGT = "tgt"; const QString STR_ext_CRC = "crc"; @@ -40,27 +60,51 @@ const QString STR_ext_gz = ".gz"; // Looks up foreign language Signal names that match this channelID EDFSignal *EDFParser::lookupSignal(ChannelID ch) { - QHash >::iterator ci; - QHash::iterator jj; - ci = resmed_codes.find(ch); + // Get list of all known foreign language names for this channel + auto channames = resmed_codes.find(ch); - if (ci == resmed_codes.end()) { + if (channames == resmed_codes.end()) { + // no alternatives strings found for this channel return nullptr; } - for (int i = 0; i < ci.value().size(); i++) { - jj = lookup.find(ci.value()[i]); + // Scan through EDF's list of signals to see if any match + for (int i = 0; i < channames.value().size(); i++) { + auto jj = lookup.find(channames.value()[i]); if (jj == lookup.end()) { continue; } + // return the EDFSignal record return jj.value(); } + // Failed return nullptr; } +// Check if given string matches any alternative signal names for this channel +bool matchSignal(ChannelID ch, const QString & name) +{ + auto channames = resmed_codes.find(ch); + + if (channames == resmed_codes.end()) { + return false; + } + + QStringList & strings = channames.value(); + int size = strings.size(); + + for (int i=0; i < size; ++i) { + // Using starts with, because ResMed is very lazy about consistency + if (name.startsWith(strings.at(i), Qt::CaseInsensitive)) { + return true; + } + } + return false; +} + EDFSignal *EDFParser::lookupName(QString name) { QHash::iterator i = lookup.find(name); @@ -439,13 +483,19 @@ int ResmedLoader::Open(QString &path, Profile *profile) line = f.readLine().trimmed(); if (!line.isEmpty()) { - key = line.section(" ", 0, 0); + key = line.section(" ", 0, 0).section("#", 1); value = line.section(" ", 1); - key = key.section("#", 1); - if (key == "SRN") { + if (key == "SRN") { // Serial Number key = STR_PROP_Serial; serial = value; + + } else if (key == "PNA") { // Product Name + key = STR_PROP_Model; + value.replace("_"," "); + + } else if (key == "PCD") { // Product Code + key = STR_PROP_ModelNumber; } idmap[key] = value; @@ -489,25 +539,15 @@ int ResmedLoader::Open(QString &path, Profile *profile) } if (path == backup_path) { + // Don't create backups if importing from backup folder create_backups = false; } /////////////////////////////////////////////////////////////////////////////////// // Parse the idmap into machine objects properties, (overwriting any old values) /////////////////////////////////////////////////////////////////////////////////// - for (QHash::iterator i = idmap.begin(); i != idmap.end(); i++) { + for (auto i = idmap.begin(); i != idmap.end(); i++) { m->properties[i.key()] = i.value(); - - if (i.key() == "PCD") { // Lookup Product Code for real model string - bool ok; - int j = i.value().toInt(&ok); - - if (ok) { - m->properties[STR_PROP_Model] = RMS9ModelMap[j]; - } - - m->properties[STR_PROP_ModelNumber] = i.value(); - } } /////////////////////////////////////////////////////////////////////////////////// @@ -1360,9 +1400,6 @@ int ResmedLoader::Open(QString &path, Profile *profile) mode = sig->data[dn]; } else { mode = 0; } - - - // Ramp, Fulltime // AutoSV machines don't have both fields sig = stredf.lookupSignal(RMS9_EPR); @@ -1414,85 +1451,39 @@ int ResmedLoader::Open(QString &path, Profile *profile) sess->settings[CPAP_Mode] = MODE_BIPAP; } - EventDataType tmp, epap = 0, ipap = 0; - - // 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 = { + { CPAP_EPAP, "EPAP" }, + { CPAP_IPAP, "IPAP" }, + { CPAP_EPAPLo, "Min EPAP" }, + { CPAP_EPAPHi, "Max EPAP" }, + { CPAP_IPAPLo, "Min IPAP" }, + { CPAP_IPAPHi, "Max IPAP" }, + { CPAP_PS, "PS" }, + { CPAP_PSMin, "Min PS" }, + { CPAP_PSMax, "Max PS" }, + { CPAP_RespRate, "RR" }, // Is this a setting to force respiratory rate on S/T machines? or an average + { CPAP_PresReliefSet, "Easy-Breathe" }, + }; + for (auto it = hash.begin(); it != hash.end(); ++it) { + auto a = stredf.lookup.find(it.value()); + if (a != stredf.lookup.end()) { + sig = a.value(); + sess->settings[it.key()] = EventDataType(sig->data[dn] * sig->gain); - if (stredf.lookup.contains("EPAP")) { - sig = stredf.lookup["EPAP"]; - epap = sig->data[dn] * sig->gain; - sess->settings[CPAP_EPAP] = epap; + 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 (stredf.lookup.contains("IPAP")) { - sig = stredf.lookup["IPAP"]; - ipap = sig->data[dn] * sig->gain; - sess->settings[CPAP_IPAP] = ipap; - } - - if (stredf.lookup.contains("Min EPAP")) { - sig = stredf.lookup["Min EPAP"]; - epap = sig->data[dn] * sig->gain; - sess->settings[CPAP_EPAPLo] = epap; - } - - if (stredf.lookup.contains("Max EPAP")) { - sig = stredf.lookup["Max EPAP"]; - epap = sig->data[dn] * sig->gain; - sess->settings[CPAP_EPAPHi] = epap; - } - - if (stredf.lookup.contains("Min IPAP")) { - sig = stredf.lookup["Min IPAP"]; - ipap = sig->data[dn] * sig->gain; - sess->settings[CPAP_IPAPLo] = ipap; - } - - if (stredf.lookup.contains("Max IPAP")) { - sig = stredf.lookup["Max IPAP"]; - ipap = sig->data[dn] * sig->gain; - sess->settings[CPAP_IPAPHi] = ipap; - } - - if (stredf.lookup.contains("PS")) { - sig = stredf.lookup["PS"]; - tmp = sig->data[dn] * sig->gain; - sess->settings[CPAP_PS] = tmp; // plain VPAP Pressure support - } - - if (stredf.lookup.contains("Min PS")) { - sig = stredf.lookup["Min PS"]; - tmp = sig->data[dn] * sig->gain; - sess->settings[CPAP_PSMin] = tmp; - } - - if (stredf.lookup.contains("Max PS")) { - sig = stredf.lookup["Max PS"]; - tmp = sig->data[dn] * sig->gain; - sess->settings[CPAP_PSMax] = 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; - } - - // this is not a setting on any machine I've played with, I think it's just an indication of the type of motor - 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; @@ -1876,7 +1867,6 @@ QString ResmedLoader::backup(QString fullname, QString backup_path, bool compres bool ResmedLoader::LoadEVE(Session *sess, EDFParser &edf) { - // EVEnt records have useless duration record. QString t; long recs; @@ -1887,21 +1877,20 @@ bool ResmedLoader::LoadEVE(Session *sess, EDFParser &edf) bool sign, ok; double d; double tt; - //ChannelID code; - //Event *e; - //totaldur=edf.GetNumDataRecords()*edf.GetDuration(); - // EventList *EL[4]={nullptr}; + // Notes: Event records have useless duration record. sess->updateFirst(edf.startdate); - //if (edf.enddate>edf.startdate) sess->set_last(edf.enddate); EventList *OA = nullptr, *HY = nullptr, *CA = nullptr, *UA = nullptr; // Allow for empty sessions.. + + // Create EventLists OA = sess->AddEventList(CPAP_Obstructive, EVL_Event); HY = sess->AddEventList(CPAP_Hypopnea, EVL_Event); UA = sess->AddEventList(CPAP_Apnea, EVL_Event); + // Process event annotation records for (int s = 0; s < edf.GetNumSignals(); s++) { recs = edf.edfsignals[s].nr * edf.GetNumDataRecords() * 2; @@ -1979,13 +1968,13 @@ bool ResmedLoader::LoadEVE(Session *sess, EDFParser &edf) } while ((data[pos] != 20) && (pos < recs)); // start code if (!t.isEmpty()) { - if (t == "obstructive apnea") { + if (matchSignal(CPAP_Obstructive, t)) { OA->AddEvent(tt, duration); - } else if (t == "hypopnea") { + } else if (matchSignal(CPAP_Hypopnea, t)) { HY->AddEvent(tt, duration + 10); // Only Hyponea's Need the extra duration??? - } else if (t == "apnea") { + } else if (matchSignal(CPAP_Apnea, t)) { UA->AddEvent(tt, duration); - } else if (t == "central apnea") { + } 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; } @@ -2007,57 +1996,62 @@ bool ResmedLoader::LoadEVE(Session *sess, EDFParser &edf) // pos++; } - while ((data[pos] == 0) && pos < recs) { pos++; } + while ((data[pos] == 0) && (pos < recs)) { pos++; } if (pos >= recs) { break; } } sess->updateLast(tt); - // qDebug(data); } return true; } bool ResmedLoader::LoadBRP(Session *sess, EDFParser &edf) { - QString t; sess->updateFirst(edf.startdate); + qint64 duration = edf.GetNumDataRecords() * edf.GetDuration(); sess->updateLast(edf.startdate + duration); for (int s = 0; s < edf.GetNumSignals(); s++) { EDFSignal &es = edf.edfsignals[s]; - //qDebug() << "BRP:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum; + long recs = es.nr * edf.GetNumDataRecords(); ChannelID code; - if (es.label == "Flow") { + if (matchSignal(CPAP_FlowRate, es.label)) { + code = CPAP_FlowRate; es.gain *= 60.0; es.physical_minimum *= 60.0; es.physical_maximum *= 60.0; es.physical_dimension = "L/M"; - code = CPAP_FlowRate; - } else if (es.label.startsWith("Mask Pres")) { + + } else if (matchSignal(CPAP_MaskPressureHi, es.label)) { code = CPAP_MaskPressureHi; - } else if (es.label.startsWith("Resp Event")) { + + } else if (matchSignal(CPAP_RespEvent, es.label)) { code = CPAP_RespEvent; + } else { qDebug() << "Unobserved ResMed BRP Signal " << es.label; continue; } - double rate = double(duration) / double(recs); - EventList *a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); - a->setDimension(es.physical_dimension); - a->AddWaveform(edf.startdate, es.data, recs, duration); - sess->setMin(code, a->Min()); - sess->setMax(code, a->Max()); - sess->setPhysMin(code, es.physical_minimum); - sess->setPhysMax(code, es.physical_maximum); + if (code) { + double rate = double(duration) / double(recs); + EventList *a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); + a->setDimension(es.physical_dimension); + a->AddWaveform(edf.startdate, es.data, recs, duration); + sess->setMin(code, a->Min()); + sess->setMax(code, a->Max()); + sess->setPhysMin(code, es.physical_minimum); + sess->setPhysMax(code, es.physical_maximum); + } } return true; } + void ResmedLoader::ToTimeDelta(Session *sess, EDFParser &edf, EDFSignal &es, ChannelID code, long recs, qint64 duration, EventDataType t_min, EventDataType t_max, bool square) { @@ -2071,10 +2065,9 @@ void ResmedLoader::ToTimeDelta(Session *sess, EDFParser &edf, EDFSignal &es, Cha time.start(); #endif - double rate = (duration / recs); // milliseconds per record double tt = edf.startdate; - //sess->UpdateFirst(tt); + EventStoreType c, last; int startpos = 0; @@ -2118,8 +2111,8 @@ void ResmedLoader::ToTimeDelta(Session *sess, EDFParser &edf, EDFSignal &es, Cha return; } - for (; sptr < eptr; sptr++) { //int i=startpos;iupdateFirst(edf.startdate); qint64 duration = edf.GetNumDataRecords() * edf.GetDuration(); sess->updateLast(edf.startdate + duration); @@ -2220,37 +2212,30 @@ bool ResmedLoader::LoadSAD(Session *sess, EDFParser &edf) long recs = es.nr * edf.GetNumDataRecords(); ChannelID code; - if (es.label.startsWith("Puls")) { - code = OXI_Pulse; - } else if (es.label == "SpO2") { - code = OXI_SPO2; - } else { - qDebug() << "Unobserved ResMed SAD Signal " << es.label; - continue; - } - bool hasdata = false; - for (int i = 0; i < recs; i++) { + for (int i = 0; i < recs; ++i) { if (es.data[i] != -1) { hasdata = true; break; } } + if (!hasdata) continue; - if (hasdata) { - if (code == OXI_Pulse) { - ToTimeDelta(sess, edf, es, code, recs, duration); - sess->setPhysMax(code, 180); - sess->setPhysMin(code, 18); - } else if (code == OXI_SPO2) { - es.physical_minimum = 60; - ToTimeDelta(sess, edf, es, code, recs, duration); - sess->setPhysMax(code, 100); - sess->setPhysMin(code, 60); - } + if (matchSignal(code = OXI_Pulse, es.label)) { + ToTimeDelta(sess, edf, es, code, recs, duration); + sess->setPhysMax(code, 180); + sess->setPhysMin(code, 18); + + } else if (matchSignal(code = OXI_SPO2, es.label)) { + es.physical_minimum = 60; + ToTimeDelta(sess, edf, es, code, recs, duration); + sess->setPhysMax(code, 100); + sess->setPhysMin(code, 60); + + } else { + qDebug() << "Unobserved ResMed SAD Signal " << es.label; } - } return true; @@ -2282,38 +2267,30 @@ bool ResmedLoader::LoadPLD(Session *sess, EDFParser &edf) rate = double(duration) / double(recs); //qDebug() << "EVE:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum << es.gain; - if (es.label == "Snore Index") { - code = CPAP_Snore; - ToTimeDelta(sess, edf, es, code, recs, duration, es.digital_maximum); - } else if (es.label.startsWith("Therapy Pres")) { - code = CPAP_Pressure; //TherapyPressure; + if (matchSignal(code = CPAP_Snore, es.label)) { + ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); + } else if (matchSignal(code = CPAP_Pressure, es.label)) { es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (es.label == "Insp Pressure") { - code = CPAP_IPAP; + } else if (matchSignal(code = CPAP_IPAP, es.label)) { sess->settings[CPAP_Mode] = MODE_BIPAP; es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if ((es.label == "MV") || (es.label == "VM")) { - code = CPAP_MinuteVent; + } else if (matchSignal(code = CPAP_MinuteVent,es.label)) { ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if ((es.label == "RR") || (es.label == "AF") || (es.label == "FR")) { - code = CPAP_RespRate; + } else if (matchSignal(code = CPAP_RespRate, es.label)) { a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); - } else if ((es.label == "Vt") || (es.label == "VC")) { - code = CPAP_TidalVolume; + } else if (matchSignal(code = CPAP_TidalVolume, es.label)) { es.gain *= 1000.0; es.physical_maximum *= 1000.0; es.physical_minimum *= 1000.0; // es.digital_maximum*=1000.0; // es.digital_minimum*=1000.0; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if ((es.label == "Leak") || (es.label.startsWith("Leck")) || (es.label.startsWith("Lekk")) - || (es.label.startsWith("Läck")) || es.label.startsWith("Läck")) { - code = CPAP_Leak; + } else if (matchSignal(code = CPAP_Leak, es.label)) { es.gain *= 60; es.physical_maximum *= 60; es.physical_minimum *= 60; @@ -2323,38 +2300,31 @@ bool ResmedLoader::LoadPLD(Session *sess, EDFParser &edf) ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0, true); sess->setPhysMax(code, 120.0); sess->setPhysMin(code, 0); - } else if (es.label == "FFL Index") { - code = CPAP_FLG; + } else if (matchSignal(code = CPAP_FLG, es.label)) { ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (es.label.startsWith("Mask Pres")) { - code = CPAP_MaskPressure; + } else if (matchSignal(code = CPAP_MaskPressure, es.label)) { es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (es.label.startsWith("Exp Press")) { - code = CPAP_EPAP; //ExpiratoryPressure + } else if (matchSignal(code = CPAP_EPAP, es.label)) { // Expiratory Pressure es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (es.label.startsWith("I:E")) { - code = CPAP_IE; //I:E ratio? + } else if (matchSignal(code = CPAP_IE, es.label)) { //I:E ratio a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); //a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0); - } else if (es.label.startsWith("Ti")) { - code = CPAP_Ti; + } else if (matchSignal(code = CPAP_Ti, es.label)) { a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); //a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0); - } else if (es.label.startsWith("Te")) { - code = CPAP_Te; + } else if (matchSignal(code = CPAP_Te, es.label)) { a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); //a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0); - } else if (es.label.startsWith("TgMV")) { - code = CPAP_TgMV; + } else if (matchSignal(code = CPAP_TgMV, es.label)) { a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); //a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0); @@ -2387,241 +2357,85 @@ bool ResmedLoader::LoadPLD(Session *sess, EDFParser &edf) return true; } -const QString RMS9_STR_Escape = "S9 Escape"; -const QString RMS9_STR_EscapeAuto = "S9 Escape Auto"; -const QString RMS9_STR_Elite = "S9 Elite"; -const QString RMS9_STR_AutoSet = "S9 AutoSet"; -const QString RMS9_STR_AutoSetForHer = "S9 AutoSet for Her"; -const QString RMS9_STR_AutoSetCS = "S9 AutoSet CS"; -const QString RMS9_STR_AutoSet25 = "S9 AutoSet 25"; -const QString RMS9_STR_VPAP_S = "S9 VPAP S"; -const QString RMS9_STR_VPAP_Auto = "S9 VPAP Auto"; -const QString RMS9_STR_VPAP_Adapt = "S9 VPAP Adapt"; -const QString RMS9_STR_VPAP_ST = "S9 VPAP ST"; -const QString RMS9_STR_VPAP_STA = "S9 VPAP ST-A"; -const QString RMS9_STR_VPAP_ST22 = "S9 VPAP ST 22"; - void ResInitModelMap() { - // Escape Series - RMS9ModelMap[36001] = RMS9ModelMap[36011] = RMS9ModelMap[36021] = RMS9ModelMap[36141] = - RMS9ModelMap[36201] = RMS9ModelMap[36221] = RMS9ModelMap[36261] = RMS9ModelMap[36301] = - RMS9ModelMap[36361] = RMS9_STR_Escape; - - // Escape Auto Series - RMS9ModelMap[36002] = RMS9ModelMap[36012] = RMS9ModelMap[36022] = RMS9ModelMap[36302] = - RMS9ModelMap[36362] = RMS9_STR_EscapeAuto; - - // Elite Series - RMS9ModelMap[36003] = RMS9ModelMap[36013] = RMS9ModelMap[36023] = RMS9ModelMap[36103] = - RMS9ModelMap[36113] = RMS9ModelMap[36123] = RMS9ModelMap[36143] = RMS9ModelMap[36203] = - RMS9ModelMap[36223] = RMS9ModelMap[36243] = RMS9ModelMap[36263] = RMS9ModelMap[36303] = - RMS9ModelMap[36343] = RMS9ModelMap[36363] = RMS9_STR_Elite; - - // AutoSet Series - RMS9ModelMap[36005] = RMS9ModelMap[36015] = RMS9ModelMap[36025] = RMS9ModelMap[36105] = - RMS9ModelMap[36115] = RMS9ModelMap[36125] = RMS9ModelMap[36145] = RMS9ModelMap[36205] = - RMS9ModelMap[36225] = RMS9ModelMap[36245] = RMS9ModelMap[36265] = RMS9ModelMap[36305] = - RMS9ModelMap[36325] = RMS9ModelMap[36345] = RMS9ModelMap[36365] = RMS9_STR_AutoSet; - - // AutoSet CS Series - RMS9ModelMap[36100] = RMS9ModelMap[36110] = RMS9ModelMap[36120] = RMS9ModelMap[36140] = - RMS9ModelMap[36200] = RMS9ModelMap[36220] = RMS9ModelMap[36360] = RMS9_STR_AutoSetCS; - - // AutoSet 25 Series - RMS9ModelMap[36106] = RMS9ModelMap[36116] = RMS9ModelMap[36126] = RMS9ModelMap[36146] = - RMS9ModelMap[36206] = RMS9ModelMap[36226] = RMS9ModelMap[36366] = RMS9_STR_AutoSet25; - - // Girly "For Her" AutoSet Series - RMS9ModelMap[36065] = RMS9_STR_AutoSetForHer; - - // VPAP S Series (+H5i +Climate Control) - RMS9ModelMap[36004] = RMS9ModelMap[36014] = RMS9ModelMap[36024] = RMS9ModelMap[36114] = - RMS9ModelMap[36124] = RMS9ModelMap[36144] = RMS9ModelMap[36204] = RMS9ModelMap[36224] = - RMS9ModelMap[36284] = RMS9ModelMap[36304] = RMS9_STR_VPAP_S; - - // VPAP Auto Series (+H5i +Climate Control) - RMS9ModelMap[36006] = RMS9ModelMap[36016] = RMS9ModelMap[36026] = RMS9_STR_VPAP_Auto; - - - // VPAP Adapt Series (+H5i +Climate Control) - // Trev's 36037 supports variable EPAP... - RMS9ModelMap[36037] = RMS9ModelMap[36007] = RMS9ModelMap[36017] = RMS9ModelMap[36027] = - RMS9ModelMap[36367] = RMS9_STR_VPAP_Adapt; - - // VPAP ST Series (+H5i +Climate Control) - RMS9ModelMap[36008] = RMS9ModelMap[36018] = RMS9ModelMap[36028] = RMS9ModelMap[36108] = - RMS9ModelMap[36148] = RMS9ModelMap[36208] = RMS9ModelMap[36228] = RMS9ModelMap[36368] = - RMS9_STR_VPAP_ST; - - // VPAP ST 22 Series - RMS9ModelMap[36118] = RMS9ModelMap[36128] = RMS9_STR_VPAP_ST22; - - // VPAP ST-A Series - RMS9ModelMap[36039] = RMS9ModelMap[36159] = RMS9ModelMap[36169] = RMS9ModelMap[36379] = - RMS9_STR_VPAP_STA; - - - // 36003, 36013, 36023, 36103, 36113, 36123, 36143, 36203, - // 36223, 36243, 36263, 36303, 36343, 36363 S9 Elite Series - // 36005, 36015, 36025, 36105, 36115, 36125, 36145, 36205, - // 36225, 36245, 36265, 36305, 36325, 36345, 36365 S9 AutoSet Series - // 36065 S9 AutoSet for Her - // 36001, 36011, 36021, 36141, 36201, 36221, 36261, 36301, - // 36361 S9 Escape - // 36002, 36012, 36022, 36302, 36362 S9 Escape Auto - // 36004, 36014, 36024, 36114, 36124, 36144, 36204, 36224, - // 36284, 36304 S9 VPAP S (+ H5i, + Climate Control) - // 36006, 36016, 36026 S9 VPAP AUTO (+ H5i, + Climate Control) - - // 36007, 36017, 36027, 36367 - // S9 VPAP ADAPT (+ H5i, + Climate - // Control) - // 36008, 36018, 36028, 36108, 36148, 36208, 36228, 36368 S9 VPAP ST (+ H5i, + Climate Control) - // 36100, 36110, 36120, 36140, 36200, 36220, 36360 S9 AUTOSET CS - // 36106, 36116, 36126, 36146, 36206, 36226, 36366 S9 AUTOSET 25 - // 36118, 36128 S9 VPAP ST 22 - // 36039, 36159, 36169, 36379 S9 VPAP ST-A - // 24921, 24923, 24925, 24926, 24927 ResMed Power Station II (RPSII) - // 33030 S8 Compact - // 33001, 33007, 33013, 33036, 33060 S8 Escape - // 33032 S8 Lightweight - // 33033 S8 AutoScore - // 33048, 33051, 33052, 33053, 33054, 33061 S8 Escape II - // 33055 S8 Lightweight II - // 33021 S8 Elite - // 33039, 33045, 33062, 33072, 33073, 33074, 33075 S8 Elite II - // 33044 S8 AutoScore II - // 33105, 33112, 33126 S8 AutoSet (including Spirit & Vantage) - // 33128, 33137 S8 Respond - // 33129, 33141, 33150 S8 AutoSet II - // 33136, 33143, 33144, 33145, 33146, 33147, 33148 S8 AutoSet Spirit II - // 33138 S8 AutoSet C - // 26101, 26121 VPAP Auto 25 - // 26119, 26120 VPAP S - // 26110, 26122 VPAP ST - // 26104, 26105, 26125, 26126 S8 Auto 25 - // 26102, 26103, 26106, 26107, 26108, 26109, 26123, 26127 VPAP IV - // 26112, 26113, 26114, 26115, 26116, 26117, 26118, 26124 VPAP IV ST - - - /* S8 Series - RMS9ModelMap[33007]="S8 Escape"; - RMS9ModelMap[33039]="S8 Elite II"; - RMS9ModelMap[33051]="S8 Escape II"; - RMS9ModelMap[33064]="S8 Escape II AutoSet"; - RMS9ModelMap[33064]="S8 Escape II AutoSet"; - RMS9ModelMap[33129]="S8 AutoSet II"; - */ - + // don't really need this anymore + Resmed_Model_Map = { + { "S9 Escape", { 36001, 36011, 36021, 36141, 36201, 36221, 36261, 36301, 36361 } }, + { "S9 Escape Auto", { 36002, 36012, 36022, 36302, 36362 } }, + { "S9 Elite", { 36003, 36013, 36023, 36103, 36113, 36123, 36143, 36203, 36223, 36243, 36263, 36303, 36343, 36363 } }, + { "S9 Autoset", { 36005, 36015, 36025, 36105, 36115, 36125, 36145, 36205, 36225, 36245, 36265, 36305, 36325, 36345, 36365 } }, + { "S9 AutoSet CS", { 36100, 36110, 36120, 36140, 36200, 36220, 36360 } }, + { "S9 AutoSet 25", { 36106, 36116, 36126, 36146, 36206, 36226, 36366 } }, + { "S9 AutoSet for Her", { 36065 } }, + { "S9 VPAP S", { 36004, 36014, 36024, 36114, 36124, 36144, 36204, 36224, 36284, 36304 } }, + { "S9 VPAP Auto", { 36006, 36016, 36026 } }, + { "S9 VPAP Adapt", { 36037, 36007, 36017, 36027, 36367 } }, + { "S9 VPAP ST", { 36008, 36018, 36028, 36108, 36148, 36208, 36228, 36368 } }, + { "S9 VPAP ST 22", { 36118, 36128 } }, + { "S9 VPAP ST-A", { 36039, 36159, 36169, 36379 } }, +/* S8 Series + { "S8 Escape", { 33007 } }, + { "S8 Elite II", { 33039 } }, + { "S8 Escape II", { 33051 } }, + { "S8 Escape II AutoSet", { 33064 } }, + { "S8 AutoSet II", { 33129 } }, + */ + }; //////////////////////////////////////////////////////////////////////////// // Translation lookup table for non-english machines //////////////////////////////////////////////////////////////////////////// - resmed_codes[CPAP_FlowRate] = {"Flow"}; + // Only put the first part, enough to be identifiable, because ResMed likes + // to signal names crop short + resmed_codes = { + { CPAP_FlowRate, { "Flow" } }, + { CPAP_MaskPressureHi, { "Mask Pres", } }, + { CPAP_MaskPressure, { "Mask Pres", } }, + { CPAP_RespEvent, { "Resp Event" } }, + { CPAP_Pressure, { "Therapy Pres", } }, + { CPAP_IPAP, { "Insp Pres" } }, + { CPAP_EPAP, { "Exp Pres", } }, + { CPAP_Leak, { "Leak", "Leck", "Lekk", "Läck", "Läck", } }, + { CPAP_RespRate, { "RR", "AF", "FR" } }, + { CPAP_MinuteVent, { "MV", "VM" } }, + { CPAP_TidalVolume, { "Vt", "VC" } }, + { CPAP_IE, { "I:E" } }, + { CPAP_Snore, { "Snore" } }, + { CPAP_FLG, { "FFL Index" } }, + { CPAP_Ti, { "Ti" } }, + { CPAP_Te, { "Te" } }, + { CPAP_TgMV, { "TgMV" } }, + { OXI_Pulse, { "Pulse", "Puls", "Pols" } }, + { OXI_SPO2, { "SpO2" } }, + + { CPAP_Obstructive, { "Obstructive apnea" } }, + { CPAP_Hypopnea, { "Hypopnea" } }, + { CPAP_Apnea, { "Apnea" } }, + { CPAP_ClearAirway, { "Central apnea" } }, + { CPAP_Mode, { "Mode", "Modus", "Funktion" } }, + + { RMS9_SetPressure, { "SetPressure", "Eingest. Druck", "Ingestelde druk", "Pres. prescrite", "Inställt tryck", "Inställt tryck" } }, + { RMS9_EPR, { "EPR" } }, + { RMS9_EPRSet, { "EPR Level", "EPR-Stufe", "EPR-niveau", "Niveau EPR", "EPR-nivå", "EPR-nivÃ¥" } }, + + { CPAP_PressureMax, { + "Max Pressure", + "Max. Druck", // German + "Max druk", // Dutch + "Pression max.", // French + "Max tryck", // Swedish + } }, + { CPAP_PressureMin, { + "Min Pressure", + "Min. Druck", // German + "Min druk", // Dutch + "Pression min.", // French + "Min tryck", // Swedish + } }, - resmed_codes[CPAP_MaskPressureHi] = { - "Mask Pres", - "Mask Pressure" // VPAP }; - resmed_codes[CPAP_MaskPressure] = { - "Mask Pres", - "Mask Pressure" // VPAP - }; - - resmed_codes[CPAP_RespEvent] = {"Resp Event"}; - resmed_codes[CPAP_Pressure] = {"Therapy Pres"}; - resmed_codes[CPAP_IPAP] = {"Insp Pressure"}; - resmed_codes[CPAP_EPAP] = {"Exp Press", "Exp Pressure"}; - - resmed_codes[CPAP_Leak] = { - "Leak", - "Leck.", - "Läcka" - }; - - resmed_codes[CPAP_RespRate] = { - "RR", - "AF", - "FR" - }; - resmed_codes[CPAP_TidalVolume] = { - "Vt", - "VC", - }; - resmed_codes[CPAP_MinuteVent] = { - "MV", - "VM" - }; - - resmed_codes[CPAP_IE] = {"I:E"}; - resmed_codes[CPAP_Snore] = {"Snore Index"}; - resmed_codes[CPAP_FLG] = {"FFL Index"}; - resmed_codes[CPAP_RespEvent] = {"RE"}; - resmed_codes[CPAP_Ti] = {"Ti"}; - resmed_codes[CPAP_Te] = {"Te"}; - - // SAD (oximetry) - - resmed_codes[OXI_Pulse] = { - "Pulse", - "Puls", // German & Swedish - "Pols", // Dutch - }; - - resmed_codes[OXI_SPO2] = {"SpO2"}; - - // Event annotations - resmed_codes[CPAP_Obstructive] = {"Obstructive apnea"}; - resmed_codes[CPAP_Hypopnea] = {"Hypopnea"}; - resmed_codes[CPAP_Apnea] = {"Apnea"}; - resmed_codes[CPAP_ClearAirway] = {"Central apnea"}; - - resmed_codes[CPAP_Mode] = { - "Mode", - "Modus", // Dutch & German - "Funktion", // Swedish - }; - - resmed_codes[RMS9_SetPressure] = { - "Set Pressure", // English - Prescription - "Eingest. Druck", // German - "Ingestelde druk", // Dutch - "Pres. prescrite", // French - "Inställt tryck", // Swedish - "Inställt tryck", // Swedish, QT5.2 - }; - - resmed_codes[RMS9_EPR] = {"EPR"}; - - resmed_codes[RMS9_EPRSet] = { - "EPR Level", - "EPR-Stufe", // French - "EPR-niveau", // Dutch - "Niveau EPR", // German - "EPR-nivå", // Swedish - "EPR-nivÃ¥", // Swedish, QT5.2 - }; - - - resmed_codes[CPAP_PressureMax] = { - "Max Pressure", - "Max. Druck", // German - "Max druk", // Dutch - "Pression max.", // French - "Max tryck", // Swedish - }; - - resmed_codes[CPAP_PressureMin] = { - "Min Pressure", - "Min. Druck", // German - "Min druk", // Dutch - "Pression min.", // French - "Min tryck", // Swedish - }; - - // STR.edf } @@ -2636,3 +2450,47 @@ void ResmedLoader::Register() resmed_initialized = true; } +//////////////////////////////////////////////////////////////////////////////////////////////// +// Model number information +// 36003, 36013, 36023, 36103, 36113, 36123, 36143, 36203, +// 36223, 36243, 36263, 36303, 36343, 36363 S9 Elite Series +// 36005, 36015, 36025, 36105, 36115, 36125, 36145, 36205, +// 36225, 36245, 36265, 36305, 36325, 36345, 36365 S9 AutoSet Series +// 36065 S9 AutoSet for Her +// 36001, 36011, 36021, 36141, 36201, 36221, 36261, 36301, +// 36361 S9 Escape +// 36002, 36012, 36022, 36302, 36362 S9 Escape Auto +// 36004, 36014, 36024, 36114, 36124, 36144, 36204, 36224, +// 36284, 36304 S9 VPAP S (+ H5i, + Climate Control) +// 36006, 36016, 36026 S9 VPAP AUTO (+ H5i, + Climate Control) + +// 36007, 36017, 36027, 36367 +// S9 VPAP ADAPT (+ H5i, + Climate +// Control) +// 36008, 36018, 36028, 36108, 36148, 36208, 36228, 36368 S9 VPAP ST (+ H5i, + Climate Control) +// 36100, 36110, 36120, 36140, 36200, 36220, 36360 S9 AUTOSET CS +// 36106, 36116, 36126, 36146, 36206, 36226, 36366 S9 AUTOSET 25 +// 36118, 36128 S9 VPAP ST 22 +// 36039, 36159, 36169, 36379 S9 VPAP ST-A +// 24921, 24923, 24925, 24926, 24927 ResMed Power Station II (RPSII) +// 33030 S8 Compact +// 33001, 33007, 33013, 33036, 33060 S8 Escape +// 33032 S8 Lightweight +// 33033 S8 AutoScore +// 33048, 33051, 33052, 33053, 33054, 33061 S8 Escape II +// 33055 S8 Lightweight II +// 33021 S8 Elite +// 33039, 33045, 33062, 33072, 33073, 33074, 33075 S8 Elite II +// 33044 S8 AutoScore II +// 33105, 33112, 33126 S8 AutoSet (including Spirit & Vantage) +// 33128, 33137 S8 Respond +// 33129, 33141, 33150 S8 AutoSet II +// 33136, 33143, 33144, 33145, 33146, 33147, 33148 S8 AutoSet Spirit II +// 33138 S8 AutoSet C +// 26101, 26121 VPAP Auto 25 +// 26119, 26120 VPAP S +// 26110, 26122 VPAP ST +// 26104, 26105, 26125, 26126 S8 Auto 25 +// 26102, 26103, 26106, 26107, 26108, 26109, 26123, 26127 VPAP IV +// 26112, 26113, 26114, 26115, 26116, 26117, 26118, 26124 VPAP IV ST + diff --git a/sleepyhead/docs/release_notes.html b/sleepyhead/docs/release_notes.html index e2a9898b..55bb169b 100644 --- a/sleepyhead/docs/release_notes.html +++ b/sleepyhead/docs/release_notes.html @@ -1,17 +1,10 @@ -

SleepyHead v0.9.4 BETA

+

SleepyHead v0.9.5 BETA

Release Notes

Greetings!

A long time in the making, but better late than never, here is a new and hopefully improved SleepyHead build.

-

The primary purpose of this build is to bring all SleepyHead's platform releases back into sync.

-

To native Mac users this is a fairly large update because you missed out on the last one, if you weren't using the windows one in Parallels/etc.... (Thanks for being so patient!)

-

Oximetry importing support has been improved, so now you should be able to import directly from your CMS50X oximeters without messing around with the windows software.
-Please remember, for achieving the best sync, always start your oximeter at the same time as your CPAP machine!
-

Fisher & Paykel Icon users should have a better experience, as SleepyHead now imports flow waveform (and I'm not even sure the last build worked properly!)

- -

I am extremly grateful for those who have stepped up to help out while I've been out of action. I'm glad to report I'm now fully back into the action.

New features & bugs fixes in this Update:
@@ -34,10 +27,6 @@ Please remember, for achieving the best sync, always start your oximeter at the

Sleep Well, and have fun!

-

The SleepyHead Team
-Mark Watkins (JediMark) [ Programming, and unintentional bug maker :-} ]
-James Marshall (Breathe Jimbo) [ Mac build magician & lead tester ]
-Richard Freeman (rich0) [ Linux support, bugfixes & packaging ]
-A big thanks to Pugsy over at CPAPTalk for all her hard work on SleepyHead documentation

+

JediMark
diff --git a/sleepyhead/mainwindow.cpp b/sleepyhead/mainwindow.cpp index e0dc192b..2edb1ac5 100644 --- a/sleepyhead/mainwindow.cpp +++ b/sleepyhead/mainwindow.cpp @@ -99,7 +99,11 @@ MainWindow::MainWindow(QWidget *parent) : QString version = FullVersionString; -// if (QString(GIT_BRANCH) != "master") { version += QString(" ") + QString(GIT_BRANCH); } + if (QString(GIT_BRANCH) != "master") { version += QString(" ") + QString(GIT_BRANCH); } + +#ifdef TEST_BUILD + version += QString(STR_TestBuild); +#endif this->setWindowTitle(STR_TR_SleepyHead + QString(" v%1 (" + tr("Profile") + ": %2)").arg(version).arg(PREF[STR_GEN_Profile].toString())); //ui->tabWidget->setCurrentIndex(1); @@ -849,8 +853,8 @@ void MainWindow::on_action_About_triggered() "" "" "
" - "

" + STR_TR_SleepyHead + " v%1.%2.%3-%4 (%8)

" + - tr("Build Date") + ": %5 %6
%7
" + tr("Data Folder Location") + ": %9


" + + "

" + STR_TR_SleepyHead + " v%1 (%2)

" + + tr("Build Date") + ": %3 %4
%5
" + tr("Data Folder Location") + ": %6


" + tr("Copyright") + " ©2011-2014 Mark Watkins (jedimark)
\n" + tr("This software is released under the GNU Public License v3.0
") + "
" @@ -910,8 +914,12 @@ void MainWindow::on_action_About_triggered() + "

" + tr("Use this software entirely at your own risk.") + "

" "
" - ).arg(major_version).arg(minor_version).arg(revision_number).arg(release_number).arg(__DATE__).arg( - __TIME__).arg(gitrev).arg(ReleaseStatus).arg(QDir::toNativeSeparators(GetAppRoot())); + ). arg(VersionString). + arg(ReleaseStatus). + arg(__DATE__). + arg(__TIME__). + arg(gitrev). + arg(QDir::toNativeSeparators(GetAppRoot())); //"" QDialog aboutbox; @@ -928,7 +936,7 @@ void MainWindow::on_action_About_triggered() layout.insertWidget(0, &webview, 1); - QHBoxLayout layout2(&aboutbox); + QHBoxLayout layout2; layout.insertLayout(1, &layout2, 1); QPushButton okbtn(QObject::tr("&Close"), &aboutbox); aboutbox.connect(&okbtn, SIGNAL(clicked()), SLOT(reject())); diff --git a/sleepyhead/sleepyhead.pro b/sleepyhead/sleepyhead.pro index c8ac05ad..7117a62f 100644 --- a/sleepyhead/sleepyhead.pro +++ b/sleepyhead/sleepyhead.pro @@ -39,13 +39,15 @@ exists(../.git):{ DEFINES += GIT_BRANCH=\\\"$$GIT_BRANCH\\\" DEFINES += GIT_REVISION=\\\"$$system(git rev-parse --short HEAD)\\\" - contains(GIT_BRANCH,"testing"):DEFINES += TEST_BUILD +# contains(GIT_BRANCH,"testing"): } else { DEFINES += GIT_BRANCH=\\\"UNKNOWN\\\" DEFINES += GIT_REVISION=\\\"UNKNOWN\\\" } +#Comment out for official builds +DEFINES += TEST_BUILD unix:!macx { diff --git a/sleepyhead/version.h b/sleepyhead/version.h index 521ba103..b575b50c 100644 --- a/sleepyhead/version.h +++ b/sleepyhead/version.h @@ -14,11 +14,9 @@ #include -const int major_version = 0; -const int minor_version = 9; -const int revision_number = 4; -const int release_number = 5; - +const int major_version = 0; // incompatible API changes +const int minor_version = 9; // new features that don't break things +const int patch_number = 5; // bugfixes, revisions #ifdef TEST_BUILD const QString ReleaseStatus = "testing"; @@ -26,10 +24,8 @@ const QString ReleaseStatus = "testing"; const QString ReleaseStatus = "beta"; #endif -const QString VersionString = QString().sprintf("%i.%i.%i", major_version, minor_version, - revision_number); -const QString FullVersionString = QString().sprintf("%i.%i.%i-%i", major_version, minor_version, - revision_number, release_number) + " " + ReleaseStatus; +const QString VersionString = QString().sprintf("%i.%i.%i", major_version, minor_version, patch_number); +const QString FullVersionString = QString().sprintf("%i.%i.%i", major_version, minor_version, patch_number)+"-"+ReleaseStatus; #ifdef Q_OS_MAC const QString PlatformString = "MacOSX";