diff --git a/Htmldocs/release_notes.html b/Htmldocs/release_notes.html
index faf057be..e2dd1c9b 100644
--- a/Htmldocs/release_notes.html
+++ b/Htmldocs/release_notes.html
@@ -32,6 +32,7 @@
[fix] Newly entered notes no longer lost when importing new day or purging oximetry data.
[fix] Purge currently selected day no longer deletes bookmarks for that day.
[fix] Remove warning from Chromebook when importing from previously used local folder.
+ [fix] Update link to Contec drivers.
Changes and fixes in OSCAR v1.2.0
diff --git a/oscar/Graphs/gFlagsLine.cpp b/oscar/Graphs/gFlagsLine.cpp
index 50f39e37..f4738c1f 100644
--- a/oscar/Graphs/gFlagsLine.cpp
+++ b/oscar/Graphs/gFlagsLine.cpp
@@ -366,23 +366,25 @@ void gFlagsLine::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
x1 = double(X - minx) * xmult + left;
x2 = double(X2 - minx) * xmult + left;
+ int width = x1-x2;
+ width = qMax(2,width); // Insure Rectangle will be visable. Flag events are 2 pixels wide.
brush = QBrush(color);
- painter.fillRect(x2, bartop, x1-x2, bottom-bartop, brush);
- if (!w.selectingArea() && !hover && QRect(x2, bartop, x1-x2, bottom-bartop).contains(w.graphView()->currentMousePos())) {
+ painter.fillRect(x2, bartop, width, bottom-bartop, brush);
+ if (!w.selectingArea() && !hover && QRect(x2, bartop, width , bottom-bartop).contains(w.graphView()->currentMousePos())) {
hover = true;
painter.setPen(QPen(Qt::red,1));
- painter.drawRect(x2, bartop, x1-x2, bottom-bartop);
+ painter.drawRect(x2, bartop, width, bottom-bartop);
int x,y;
- int s = *dptr;
- int m = s / 60;
- s %= 60;
+ double s = *dptr;
+ double m;
+ s=60*modf(s/60,&m);
QString lab = QString("%1").arg(schema::channel[m_code].fullname());
if (m>0) {
lab += QObject::tr(" (%2 min, %3 sec)").arg(m).arg(s);
} else {
- lab += QObject::tr(" (%3 sec)").arg(m).arg(s);
+ lab += QObject::tr(" (%3 sec)").arg(s);
}
GetTextExtent(lab, x, y);
w.ToolTip(lab, x2 - 10, bartop + (3 * w.printScaleY()), TT_AlignRight, tooltipTimeout);
diff --git a/oscar/Graphs/gGraph.cpp b/oscar/Graphs/gGraph.cpp
index 5244d6be..9c82f9b8 100644
--- a/oscar/Graphs/gGraph.cpp
+++ b/oscar/Graphs/gGraph.cpp
@@ -275,7 +275,9 @@ void gGraph::setDay(Day *day)
}
rmin_y = rmax_y = 0;
- ResetBounds();
+ // This resets weight and bmi overview graphs to full date range when they are changed.
+ // is it required ever?
+ // ResetBounds();
}
void gGraph::setZoomY(short zoom)
diff --git a/oscar/Graphs/gLineOverlay.cpp b/oscar/Graphs/gLineOverlay.cpp
index b6e1484f..ce42c37c 100644
--- a/oscar/Graphs/gLineOverlay.cpp
+++ b/oscar/Graphs/gLineOverlay.cpp
@@ -28,7 +28,6 @@ void gLineOverlayBar::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
if (!schema::channel[m_code].enabled())
return;
-
int left = region.boundingRect().left();
int topp = region.boundingRect().top(); // FIXME: Misspelling intentional.
double width = region.boundingRect().width();
@@ -42,10 +41,12 @@ void gLineOverlayBar::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
double xx = w.max_x - w.min_x;
//double yy = w.max_y - w.min_y;
+
+ if (xx <= 0) { return; }
+
double jj = width / xx;
- if (xx <= 0) { return; }
double x1, x2;
@@ -138,12 +139,20 @@ void gLineOverlayBar::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
x1 = jj * double(X - w.min_x);
x2 = jj * double(Y - w.min_x);
- x2 += (int(x1)==int(x2)) ? 1 : 0;
-
x2 = qMax(0.0, x2)+left;
x1 = qMin(width, x1)+left;
- painter.fillRect(QRect(x2, start_py, x1-x2, height), brush);
+ // x2 represents the begining of a span in pixels
+ // x1 represent the end of the span in pixels
+ // BUG HERE
+ //x2 += (int(x1)==int(x2)) ? 1 : 0;
+ // Fixed BY
+ int duration = x1-x2;
+ if (duration<2) duration=2; // display minial span with 2 pixels.
+ x2 =x1-duration;
+
+ painter.fillRect(QRect(x2, start_py, duration, height), brush);
+
}
}/* else if (m_flt == FT_Dot) {
////////////////////////////////////////////////////////////////////////////
diff --git a/oscar/SleepLib/common.cpp b/oscar/SleepLib/common.cpp
index 496259d2..55f460b4 100644
--- a/oscar/SleepLib/common.cpp
+++ b/oscar/SleepLib/common.cpp
@@ -332,8 +332,8 @@ void validateFont (QString which, int size, bool bold, bool italic) {
QFontDatabase fontdatabase;
if (installedFontFamilies.isEmpty()) {
installedFontFamilies = fontdatabase.families();
- qDebug() << "validateFont found" << installedFontFamilies.count() << "installed font families";
}
+ qDebug() << "validateFont found" << installedFontFamilies.count() << "installed font families";
QString prefPrefix = "Fonts_" + which + "_";
@@ -345,11 +345,19 @@ void validateFont (QString which, int size, bool bold, bool italic) {
if (p_pref->contains(prefPrefix + "Name")) {
// We already have a font, so it becomes desired font (if valid)
QString testFont = (*p_pref)[prefPrefix + "Name"].toString();
+ int prefSize = (*p_pref)[prefPrefix+"Size"].toInt();
// Is this a good font?
- if (testFont.length() > 0 && installedFontFamilies.indexOf(testFont) >= 0) {
- desiredFont = testFont;
- forceFont = false;
- }
+ if (testFont.length() > 0 ) {
+ qDebug() << which << "Preferences font is" << testFont;
+ if ( installedFontFamilies.indexOf(testFont) >= 0) {
+ desiredFont = testFont;
+ forceFont = false;
+ } else {
+ qDebug() << testFont << prefSize << "not found, substituting" << desiredFont << size;
+ for (int i = 0; i< installedFontFamilies.size(); i++)
+ qDebug() << installedFontFamilies.at(i);
+ }
+ }
}
#ifdef Q_OS_MAC
@@ -367,15 +375,20 @@ void validateFont (QString which, int size, bool bold, bool italic) {
(*p_pref)[prefPrefix + "Bold"] = bold;
(*p_pref)[prefPrefix + "Italic"] = italic;
}
+ qDebug() << which << "font set to" << desiredFont << "at size" << (*p_pref)[prefPrefix + "Size"];
}
void setApplicationFont () {
+ qDebug() << "Application font starts out as" << QApplication::font();
QFont font = QFont(((*p_pref)["Fonts_Application_Name"]).toString());
font.setPointSize(((*p_pref)["Fonts_Application_Size"]).toInt());
font.setWeight(((*p_pref)["Fonts_Application_Bold"]).toBool() ? QFont::Bold : QFont::Normal);
font.setItalic(((*p_pref)["Fonts_Application_Italic"]).toBool());
QApplication::setFont(font);
mainwin->menuBar()->setFont(font);
+ qDebug() << "Application font set to" << font;
+ qDebug() << "Application font reads back as" << QApplication::font();
+ qDebug() << "system font is" << QFontDatabase::systemFont(QFontDatabase::GeneralFont).family();
}
bool removeDir(const QString &path)
diff --git a/oscar/SleepLib/day.cpp b/oscar/SleepLib/day.cpp
index 691a1716..a752b6cf 100644
--- a/oscar/SleepLib/day.cpp
+++ b/oscar/SleepLib/day.cpp
@@ -202,11 +202,14 @@ EventDataType Day::countInsideSpan(ChannelID span, ChannelID code)
EventDataType Day::lookupValue(ChannelID code, qint64 time, bool square)
{
+ // Remove drift from CPAP graphs so we get the right value...
+ qint64 clockdrift = qint64(p_profile->cpap->clockDrift()) * 1000L;
for (auto & sess : sessions) {
if (sess->enabled()) {
- if ((time > sess->first()) && (time < sess->last())) {
+ qint64 drift = (sess->type() == MT_CPAP?clockdrift:0);
+ if ((time-drift > sess->first()) && (time-drift < sess->last())) {
if (sess->channelExists(code)) {
- return sess->SearchValue(code,time,square);
+ return sess->SearchValue(code,time-drift,square);
}
}
}
diff --git a/oscar/SleepLib/journal.cpp b/oscar/SleepLib/journal.cpp
index 56ddff0d..6e515497 100644
--- a/oscar/SleepLib/journal.cpp
+++ b/oscar/SleepLib/journal.cpp
@@ -214,6 +214,7 @@ void BackupJournal(QString filename)
stream.setAutoFormattingIndent(2);
stream.writeStartDocument();
+// stream.writeProcessingInstruction("xml", "version=\"1.0\" encoding=\"UTF-8\"");
stream.writeStartElement("OSCAR");
stream.writeStartElement("Journal");
stream.writeAttribute("username", p_profile->user->userName());
@@ -301,6 +302,8 @@ void BackupJournal(QString filename)
}
QTextStream ts(&file);
+ ts.setCodec("UTF-8");
+ ts.setGenerateByteOrderMark(true);
ts << outBuf;
file.close();
}
diff --git a/oscar/SleepLib/loader_plugins/cms50_loader.cpp b/oscar/SleepLib/loader_plugins/cms50_loader.cpp
index 2d79cfb8..b7b89888 100644
--- a/oscar/SleepLib/loader_plugins/cms50_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/cms50_loader.cpp
@@ -7,10 +7,11 @@
* for more details. */
//********************************************************************************************
-/// IMPORTANT!!!
-//********************************************************************************************
-// Please INCREMENT the cms50_data_version in cms50_loader.h when making changes to this loader
-// that change loader behaviour or modify channels.
+// Please only INCREMENT the cms50_data_version in cms50_loader.h when making changes
+// that change loader behaviour or modify channels in a manner that fixes old data imports.
+// Note that changing the data version will require a reimport of existing data for which OSCAR
+// does not keep a backup - so it should be avoided if possible.
+// i.e. there is no need to change the version when adding support for new devices
//********************************************************************************************
#include
diff --git a/oscar/SleepLib/loader_plugins/cms50f37_loader.cpp b/oscar/SleepLib/loader_plugins/cms50f37_loader.cpp
index 59b02928..fa88bc38 100644
--- a/oscar/SleepLib/loader_plugins/cms50f37_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/cms50f37_loader.cpp
@@ -7,10 +7,11 @@
* for more details. */
//********************************************************************************************
-/// IMPORTANT!!!
-//********************************************************************************************
-// Please INCREMENT the cms50_data_version in cms50_loader.h when making changes to this loader
-// that change loader behaviour or modify channels.
+// Please only INCREMENT the cms50f37_data_version in cms50f37_loader.h when making changes
+// that change loader behaviour or modify channels in a manner that fixes old data imports.
+// Note that changing the data version will require a reimport of existing data for which OSCAR
+// does not keep a backup - so it should be avoided if possible.
+// i.e. there is no need to change the version when adding support for new devices
//********************************************************************************************
// #include
diff --git a/oscar/SleepLib/loader_plugins/dreem_loader.cpp b/oscar/SleepLib/loader_plugins/dreem_loader.cpp
index 872dacca..a6c032d6 100644
--- a/oscar/SleepLib/loader_plugins/dreem_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/dreem_loader.cpp
@@ -7,10 +7,11 @@
* for more details. */
//********************************************************************************************
-// IMPORTANT!!!
-//********************************************************************************************
-// Please INCREMENT the dreem_data_version in dreem_loader.h when making changes to this loader
-// that change loader behaviour or modify channels.
+// Please only INCREMENT the dreem_data_version in dreem_loader.h when making changes
+// that change loader behaviour or modify channels in a manner that fixes old data imports.
+// Note that changing the data version will require a reimport of existing data for which OSCAR
+// does not keep a backup - so it should be avoided if possible.
+// i.e. there is no need to change the version when adding support for new devices
//********************************************************************************************
#include
diff --git a/oscar/SleepLib/loader_plugins/icon_loader.cpp b/oscar/SleepLib/loader_plugins/icon_loader.cpp
index 21d6a428..f2ad2963 100644
--- a/oscar/SleepLib/loader_plugins/icon_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/icon_loader.cpp
@@ -355,7 +355,9 @@ quint32 convertFLWDate(quint32 timestamp) // Bit format: hhhhhmmmmmmssssssYYYYYY
QDateTime dt = QDateTime(QDate(year, month, day), QTime(hour, minute, second), Qt::UTC);
if(!dt.isValid()){
- dt = QDateTime(QDate(2015,1,1), QTime(0,0,1));
+ // make this date too early, then change test later
+ // dt = QDateTime(QDate(2015,1,1), QTime(0,0,1));
+ dt = QDateTime(QDate(2010,1,1), QTime(0,0,0));
}
// Q NO!!! _ASSERT(dt.isValid());
// if ((year == 2013) && (month == 9) && (day == 18)) {
@@ -483,7 +485,7 @@ bool FPIconLoader::OpenFLW(Machine *mach, const QString & filename)
ts = convertFLWDate(t2);
- if (ts > QDateTime(QDate(2015,1,1), QTime(0,0,0)).toTime_t()) {
+ if (ts < QDateTime(QDate(2010,1,1), QTime(0,1,0)).toTime_t()) {
return false;
}
diff --git a/oscar/SleepLib/loader_plugins/intellipap_loader.cpp b/oscar/SleepLib/loader_plugins/intellipap_loader.cpp
index 0e9aa3eb..7c6cedb8 100644
--- a/oscar/SleepLib/loader_plugins/intellipap_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/intellipap_loader.cpp
@@ -3,6 +3,7 @@
* Notes: Intellipap DV54 requires the SmartLink attachment to access this data.
*
* Copyright (c) 2011-2018 Mark Watkins
+ * Copyright (c) 2020 The OSCAR Team
*
* This file is subject to the terms and conditions of the GNU General Public
* License. See the file COPYING in the main directory of the source code
@@ -12,6 +13,8 @@
#include "intellipap_loader.h"
+//#define DEBUG6
+
ChannelID INTP_SmartFlexMode, INTP_SmartFlexLevel;
Intellipap::Intellipap(Profile *profile, MachineID id)
@@ -591,22 +594,82 @@ int IntellipapLoader::OpenDV5(const QString & path)
return c;
}
-struct DV6_S_Record
+////////////////////////////////////////////////////////////////////////////
+// Devilbiss DV64 Notes
+// 1) High resolution data (flow and pressure) is kept on SD for only 100 hours
+// 1a) Flow graph for days without high resolution data is absent
+// 1b) Pressure graph is high resolution when high res data is available and
+// only 1 per minute when using low resolution data.
+// 2) Max and Average leak rates are as reported by DV64 machine but we're
+// not sure how those measures relate to other machine's data. Leak rate
+// seems to include the intentional mask leak.
+// 2a) Not sure how SmartLink calculates the pct of time of poor mask fit.
+// May be same as what we call large leak time for other machines?
+////////////////////////////////////////////////////////////////////////////
+
+class RollingFile
{
+public:
+ RollingFile () { }
+
+ ~RollingFile () {
+ if (data)
+ delete [] data;
+ data = nullptr;
+ }
+
+ bool open (QString fn); // Open the file
+ bool close(); // close the file
+ unsigned char * get(); // read the next record in the file
+
+ int numread () {return number_read;}; // Return number of records read
+ int recnum () {return record_number;}; // Return last-read record number
+
+private:
+ QString filename;
+ QFile file;
+ int record_length;
+ int wrap_record;
+ bool wrapping = false;
+
+ int number_read = 0; // Number of records read
+
+ int record_number = 0; // Number of record. First record in the file is #1. First record read is wrap_record;
+
+ unsigned char * data = nullptr;
+};
+
+struct DV6TestedModel
+{
+ QString model;
+ QString name;
+};
+
+static const DV6TestedModel testedModels[] = {
+ { "DV64D", "Blue StandardPlus" },
+ { "DV64E", "Blue AutoPlus" },
+ { "DV63E", "Blue (IntelliPAP 2) AutoPlus" },
+ { "", "unknown product" } // List stopper -- must be last entry
+};
+
+struct DV6_S_Data // Daily summary
+{
+/***
Session * sess;
unsigned char u1; //00 (position)
+***/
unsigned int start_time; //01
unsigned int stop_time; //05
unsigned int atpressure_time;//09
EventDataType hours; //13
- EventDataType meh; //14
+// EventDataType unknown14; //14
EventDataType pressureAvg; //15
EventDataType pressureMax; //16
EventDataType pressure50; //17 50th percentile
EventDataType pressure90; //18 90th percentile
EventDataType pressure95; //19 95th percentile
EventDataType pressureStdDev;//20 std deviation
- EventDataType u2; //21
+// EventDataType unknown_21; //21
EventDataType leakAvg; //22
EventDataType leakMax; //23
EventDataType leak50; //24 50th percentile
@@ -615,8 +678,8 @@ struct DV6_S_Record
EventDataType leakStdDev; //27 std deviation
EventDataType tidalVolume; //28 & 0x29
EventDataType avgBreathRate; //30
- EventDataType u3;
- EventDataType u4; //32 snores / hypopnea per minute
+ EventDataType unknown_31; //31
+ EventDataType snores; //32 snores / hypopnea per minute
EventDataType timeInExPuf; //33 Time in Expiratory Puff
EventDataType timeInFL; //34 Time in Flow Limitation
EventDataType timeInPB; //35 Time in Periodic Breathing
@@ -624,13 +687,11 @@ struct DV6_S_Record
EventDataType indexOA; //37 Obstructive
EventDataType indexCA; //38 Central index
EventDataType indexHyp; //39 Hypopnea Index
- EventDataType r0; //40 Reserved?
- EventDataType r1; //41 Reserved?
+ EventDataType unknown_40; //40 Reserved?
+ EventDataType unknown_41; //40 Reserved?
//42-48 unknown
EventDataType pressureSetMin; //49
EventDataType pressureSetMax; //50
-
- bool hasMaskPressure;
};
#ifdef _MSC_VER
@@ -640,82 +701,484 @@ struct DV6_S_Record
#endif
-PACK(struct SET_BIN_REC {
+// DV6_S_REC is the day structure in the S.BIN file
+PACK (struct DV6_S_REC{
+ unsigned char begin[4]; //0 Beginning of day
+ unsigned char end[4]; //4 End of day
+ unsigned char written[4]; //8 When this record was written??
+ unsigned char hours; //12 Hours in session * 10
+ unsigned char unknown_13; //13
+ unsigned char pressureAvg; //14 All pressure settings are * 10
+ unsigned char pressureMax; //15
+ unsigned char pressure50; //16 50th percentile
+ unsigned char pressure90; //17 90th percentile
+ unsigned char pressure95; //18 95th percentile
+ unsigned char pressureStdDev; //19 std deviation
+ unsigned char unknown_20; //20
+ unsigned char leakAvg; //21
+ unsigned char leakMax; //22
+ unsigned char leak50; //23 50th percentile
+ unsigned char leak90; //24 90th percentile
+ unsigned char leak95; //25 95th percentile
+ unsigned char leakStdDev; //26 std deviation
+ unsigned char tv1; //27 tidal volume = tv2 * 256 + tv1
+ unsigned char tv2; //28
+ unsigned char avgBreathRate; //29
+ unsigned char unknown_30; //30
+ unsigned char snores; //31 snores / hypopnea per minute
+ unsigned char timeInExPuf; //32 % Time in Expiratory Puff * 2
+ unsigned char timeInFL; //33 % Time in Flow Limitation * 2
+ unsigned char timeInPB; //34 % Time in Periodic Breathing * 2
+ unsigned char maskFit; //35 mask fit (or rather, not fit) percentage * 2
+ unsigned char indexOA; //36 Obstructive index * 4
+ unsigned char indexCA; //37 Central index * 4
+ unsigned char indexHyp; //38 Hypopnea Index * 4
+ unsigned char unknown_39; //39 Reserved?
+ unsigned char unknown_40; //40 Reserved?
+ unsigned char unknown_41; //41
+ unsigned char unknown_42; //42
+ unsigned char unknown_43; //43
+ unsigned char unknown_44; //44 % time snoring *4
+ unsigned char unknown_45; //45
+ unsigned char unknown_46; //46
+ unsigned char unknown_47; //47 (related to smartflex and flow rounding?)
+ unsigned char pressureSetMin; //48
+ unsigned char pressureSetMax; //49
+ unsigned char unknown_50; //50
+ unsigned char unknown_51; //51
+ unsigned char unknown_52; //52
+ unsigned char unknown_53; //53
+ unsigned char checksum; //54
+});
+
+// DV6 SET.BIN - structure of the entire file
+PACK (struct SET_BIN_REC {
char unknown_00; // assuming file version
char serial[11]; // null terminated
- unsigned short cap_flags; // capability flags
- unsigned short cpap_pressure;
- unsigned short max_pressure;
- unsigned short min_pressure;
+ unsigned char language;
+ unsigned char capabilities; // CPAP or APAP
+ unsigned char unknown_11;
+ unsigned char cpap_pressure;
+ unsigned char unknown_12;
+ unsigned char max_pressure;
+ unsigned char unknown_13;
+ unsigned char min_pressure;
unsigned char alg_apnea_threshhold; // always locked at 00
unsigned char alg_apnea_duration;
unsigned char alg_hypop_threshold;
unsigned char alg_hypop_duration;
- unsigned short ramp_pressure;
- unsigned short ramp_duration;
- unsigned char unknown_01[3];
- unsigned char smartflex;
- unsigned char unknown_02;
+ unsigned char ramp_pressure;
+ unsigned char unknown_01;
+ unsigned char ramp_duration;
+ unsigned char unknown_02[3];
+ unsigned char smartflex_setting;
+ unsigned char smartflex_when;
unsigned char inspFlowRounding;
unsigned char expFlowRounding;
unsigned char complianceHours;
- unsigned char unknown_03[9];
+ unsigned char unknown_03;
+ unsigned char tubing_diameter;
+ unsigned char autostart_setting;
+ unsigned char unknown_04;
+ unsigned char show_hide;
+ unsigned char unknown_05;
+ unsigned char lock_flags;
+ unsigned char unknown_06;
unsigned char humidifier_setting; // 0-5
- unsigned char unused[83];
+ unsigned char unknown_7;
+ unsigned char possible_alg_apnea;
+ unsigned char unknown_8[7];
+ unsigned char bacteria_filter;
+ unsigned char unused[73];
unsigned char checksum;
}); // http://digitalvampire.org/blog/index.php/2006/07/31/why-you-shouldnt-use-__attribute__packed/
-int IntellipapLoader::OpenDV6(const QString & path)
-{
- QString newpath = path + DV6_DIR;
+// Unless explicitly noted, all other DV6_x_REC are definitions for the repeating data structure that follows the header
+PACK (struct DV6_HEADER {
+ unsigned char unknown; // 0 always zero
+ unsigned char filetype; // 1 always "R"
+ unsigned char serial[11]; // 2 serial number
+ unsigned char numRecords[4]; // 13 Number of records in file (always 180,000)
+ unsigned char recordLength; // 17 Length of data record (always 117)
+ unsigned char recordStart[4]; // 18 First record in wrap-around buffer
+ unsigned char unknown_22[21]; // 22 Unknown values
+ unsigned char unknown_43[12]; // 43 Seems always to be zero
+ unsigned char checksum; // 55 Checksum
+});
- // Prime the machine database's info field with stuff relevant to this machine
- MachineInfo info = newInfo();
- info.series = "DV6";
- info.serial = "Unknown";
+// DV6 E.BIN - event data
+struct DV6_E_REC { // event log record
+ unsigned char begin[4];
+ unsigned char end[4];
+ unsigned char unknown_01;
+ unsigned char unknown_02;
+ unsigned char unknown_03;
+ unsigned char unknown_04;
+ unsigned char event_type;
+ unsigned char event_severity;
+ unsigned char value;
+ unsigned char reduction;
+ unsigned char duration;
+ unsigned char unknown[7];
+ unsigned char checksum;
+};
- int vmin=0, vmaj=0;
- EventDataType max_pressure=0, min_pressure=0; //, starting_pressure;
+// DV6 U.BIN - session start and stop times
+struct DV6_U_REC {
+ unsigned char begin[4];
+ unsigned char end[4];
+ unsigned char checksum; // possible checksum? Not really sure
+};
+
+// DV6 R.BIN - High resolution data (breath) and moderate resolution (pressure, flags)
+struct DV6_R_REC {
+ unsigned char timestamp[4];
+ qint16 breath[50]; // 50 breath flow records at 25 Hz
+ unsigned char pressure1; // pressure in first second of frame
+ unsigned char pressure2; // pressure in second second of frame
+ unsigned char unknown106;
+ unsigned char unknown107;
+ unsigned char flags1[4]; // flags for first second of frame
+ unsigned char flags2[4]; // flags for second second of frame
+ unsigned char checksum;
+};
+
+// DV6 L.BIN - Low resolution data
+PACK (struct DV6_L_REC {
+ unsigned char timestamp[4]; // 0 timestamp
+ unsigned char maxLeak; // 4 lpm
+ unsigned char avgLeak; // 5 lpm
+ unsigned char tidalVolume6; // 6
+ unsigned char tidalVolume7; // 7
+ unsigned char breathRate; // 8 breaths per minute
+ unsigned char unknown9; // 9
+ unsigned char avgPressure; // 10 pressure * 10
+ unsigned char unknown11; // 11 always zero?
+ unsigned char unknown12; // 12
+ unsigned char pressureLimitLow; // 13 pressure * 10
+ unsigned char pressureLimitHigh;// 14 pressure * 10
+ unsigned char timeSnoring; // 15
+ unsigned char snoringSeverity; // 16
+ unsigned char timeEP; // 17
+ unsigned char epSeverity; // 18
+ unsigned char timeX1; // 19 ??
+ unsigned char x1Severity; // 20 ??
+ unsigned char timeX2; // 21 ??
+ unsigned char x2Severity; // 22 ??
+ unsigned char timeX3; // 23 ??
+ unsigned char x3Severity; // 24 ??
+ unsigned char apSeverity; // 25
+ unsigned char TimeApnea; // 26
+ unsigned char noaSeverity; // 27
+ unsigned char timeNOA; // 28
+ unsigned char ukSeverity; // 29 ??
+ unsigned char timeUk; // 30 ??
+ unsigned char unknown31; // 31
+ unsigned char unknown32; // 32
+ unsigned char unknown33; // 33
+ unsigned char unknownFlag34; // 34
+ unsigned char unknownTime35; // 35
+ unsigned char unknownFlag36; // 36
+ unsigned char unknown37; // 37
+ unsigned char unknown38; // 38
+ unsigned char unknown39; // 39
+ unsigned char unknown40; // 40
+ unsigned char unknown41; // 41
+ unsigned char unknown42; // 42
+ unsigned char unknown43; // 43
+ unsigned char checksum; // 44
+});
+
+// Our structure for managing sessions
+struct DV6_SessionInfo {
+ Session * sess;
+ DV6_S_Data *dailyData;
+ SET_BIN_REC * dv6Settings;
+
+ unsigned int begin;
+ unsigned int end;
+ unsigned int written;
+ bool haveHighResData;
+ CPAPMode mode = MODE_UNKNOWN;
+};
+
+unsigned int ep = 0;
+
+// Convert a 4-character number in DV6 data file to a standard int
+unsigned int convertNum (unsigned char num[]) {
+ return ((num[3] << 24) + (num[2] << 16) + (num[1] << 8) + num[0]);
+}
+
+// Convert a timestamp in DV6 data file to a standard Unix epoch timestamp as used in OSCAR
+unsigned int convertTime (unsigned char time[]) {
+ if (ep == 0) {
+ QDateTime epoch(QDate(2002, 1, 1), QTime(0, 0, 0), Qt::UTC); // Intellipap Epoch
+ ep = epoch.toTime_t();
+ }
+ return ((time[3] << 24) + (time[2] << 16) + (time[1] << 8) + time[0]) + ep; // Time as Unix epoch time
+}
+
+bool RollingFile::open(QString fn) {
+
+ filename = fn;
+ file.setFileName(filename);
+
+ if (!file.open(QIODevice::ReadOnly)) {
+ qWarning() << "DV6 RollingFile could not open" << filename << "for reading, error code" << file.error() << file.errorString();
+ return false;
+ }
+
+ QByteArray dataBA = file.read(sizeof(DV6_HEADER));
+ DV6_HEADER * hdr = (DV6_HEADER *) dataBA.data();
+ record_length = hdr->recordLength;
+ wrap_record = convertNum(hdr->recordStart);
+ record_number = wrap_record;
+ number_read = 0;
+ wrapping = false;
+
+ data = new unsigned char[record_length];
+
+ if (!file.seek(sizeof(DV6_HEADER) + wrap_record * record_length)) {
+ qWarning() << "DV6 RollingFile unable to make initial seek to record" << wrap_record << "in" + filename << file.error() << file.errorString();
+ file.close();
+ return false;
+ }
+
+ qDebug() << "RollingFile opening" << filename << "at wrap record" << wrap_record;
+ return true;
+}
+
+bool RollingFile::close() {
+ file.close();
+ if (data != nullptr)
+ delete [] data;
+ data = nullptr;
+ return true;
+}
+
+unsigned char * RollingFile::get() {
+
+ record_number++;
+
+ // If we have found the wrap record again, we are done
+ if (wrapping && (record_number == wrap_record))
+ return nullptr;
+
+ // Hare we reached end of file and need to wrap around to beginning?
+ if (file.atEnd()) {
+ if (wrapping) {
+ qDebug() << "DV6 RollingFile wrap - second time through";
+ return nullptr;
+ }
+ qDebug() << "DV6 RollingFile wrapping to beginning of data in" << filename << "record number is" << record_number-1 << "records read" << number_read;
+ record_number = 0;
+ wrapping = true;
+ if (!file.seek(sizeof(DV6_HEADER))) {
+ file.close();
+ qWarning() << "DV6 RollingFile unable to seek to first data record in file";
+ return nullptr;
+ }
+ }
+
+ QByteArray dataBA;
+ dataBA=file.read(record_length); // read next record
+ if (dataBA.size() != record_length) {
+ qWarning() << "DV6 RollingFile record" << record_number << "wrong length";
+ file.close();
+ return nullptr;
+ }
+
+ number_read++;
+
+// qDebug() << "RollingFile read" << filename << "record number" << record_number << "of length" << record_length << "number read so far" << number_read;
+ memcpy (data, (unsigned char *) dataBA.data(), record_length);
+ return data;
+}
+
+MachineInfo info;
+Machine * mach = nullptr;
+
+bool rebuild_from_backups = false;
+
+QMap DailySummaries;
+QMap SessionData;
+SET_BIN_REC * settings;
+
+///////////////////////////////////////////////
+// U.BIN - Open and parse session list and create session data structures
+// with session start and stop times.
+///////////////////////////////////////////////
+
+bool load6Sessions (const QString & path) {
+
+ RollingFile rf;
+ unsigned int ts1,ts2;
+
+ SessionData.clear();
+
+ qDebug() << "Parsing U.BIN";
+
+ if (!rf.open(path+"/U.BIN")) {
+ qWarning() << "Unable to open U.BIN";
+ return false;
+ }
+
+ do {
+ DV6_U_REC * rec = (DV6_U_REC *) rf.get();
+ if (rec == nullptr)
+ break;
+ DV6_SessionInfo sinfo;
+ // big endian
+ ts1 = convertTime(rec->begin); // session start time (this is also the session id)
+ ts2 = convertTime(rec->end); // session end time
+#ifdef DEBUG6
+ qDebug() << "U.BIN Session" << QDateTime::fromTime_t(ts1).toString("MM/dd/yyyy hh:mm:ss") << ts1 << "to" << QDateTime::fromTime_t(ts2).toString("MM/dd/yyyy hh:mm:ss") << ts2;
+#endif
+ sinfo.sess = nullptr;
+ sinfo.dailyData = nullptr;
+ sinfo.begin = ts1;
+ sinfo.end = ts2;
+ sinfo.written = 0;
+ sinfo.haveHighResData = false;
+
+ SessionData[ts1] = sinfo;
+ } while (true);
+
+ rf.close();
+ qDebug() << "DV6 U.BIN processed" << rf.numread() << "records";
+
+ return true;
+}
+
+/////////////////////////////////////////////////////////////////////////////////
+// Parse SET.BIN settings file
+/////////////////////////////////////////////////////////////////////////////////
+
+bool load6Settings (const QString & path) {
+
+ QByteArray dataBA;
+
+ QFile f(path+"/"+SET_BIN);
- QByteArray str, dataBA;
- unsigned char *data = NULL;
- /////////////////////////////////////////////////////////////////////////////////
- // Parse SET.BIN settings file
- /////////////////////////////////////////////////////////////////////////////////
- QFile f(newpath+"/"+SET_BIN);
if (f.open(QIODevice::ReadOnly)) {
// Read and parse entire SET.BIN file
dataBA = f.readAll();
f.close();
- SET_BIN_REC *setbin = (SET_BIN_REC *)dataBA.data();
- info.serial = QString(setbin->serial);
- max_pressure = setbin->max_pressure;
- min_pressure = setbin->min_pressure;
+ settings = (SET_BIN_REC *)dataBA.data();
} else { // if f.open settings file
// Settings file open failed, return
- return -1;
+ qWarning() << "Unable to open SET.BIN file";
+ return false;
+ }
+ return true;
+}
+
+////////////////////////////////////////////////////////////////////////////////////////
+// S.BIN - Open and load day summary list
+////////////////////////////////////////////////////////////////////////////////////////
+
+bool load6DailySummaries (const QString & path) {
+ RollingFile rf;
+
+ DailySummaries.clear();
+
+ if (!rf.open(path+"/S.BIN")) {
+ qWarning() << "Unable to open S.BIN";
+ return false;
}
- ////////////////////////////////////////////////////////////////////////////////////////
- // Parser VER.BIN for model number
- ////////////////////////////////////////////////////////////////////////////////////////
- f.setFileName(newpath+"/VER.BIN");
+ qDebug() << "Reading S.BIN summaries";
+
+ do {
+ DV6_S_REC * rec = (DV6_S_REC *) rf.get();
+ if (rec == nullptr)
+ break;
+
+ DV6_S_Data dailyData;
+
+ dailyData.start_time = convertTime(rec->begin);
+ dailyData.stop_time = convertTime(rec->end);
+ dailyData.atpressure_time = convertTime(rec->written);
+
+ dailyData.hours = float(rec->hours) / 10.0F;
+ dailyData.pressureSetMin = float(rec->pressureSetMin) / 10.0F;
+ dailyData.pressureSetMax = float(rec->pressureSetMax) / 10.0F;
+
+ // The following stuff is not necessary to decode, but can be used to verify we are on the right track
+ dailyData.pressureAvg = float(rec->pressureAvg) / 10.0F;
+ dailyData.pressureMax = float(rec->pressureMax) / 10.0F;
+ dailyData.pressure50 = float(rec->pressure50) / 10.0F;
+ dailyData.pressure90 = float(rec->pressure90) / 10.0F;
+ dailyData.pressure95 = float(rec->pressure95) / 10.0F;
+ dailyData.pressureStdDev = float(rec->pressureStdDev) / 10.0F;
+
+ dailyData.leakAvg = float(rec->leakAvg) / 10.0F;
+ dailyData.leakMax = float(rec->leakMax) / 10.0F;
+ dailyData.leak50= float(rec->leak50) / 10.0F;
+ dailyData.leak90 = float(rec->leak90) / 10.0F;
+ dailyData.leak95 = float(rec->leak95) / 10.0F;
+ dailyData.leakStdDev = float(rec->leakStdDev) / 10.0F;
+
+ dailyData.tidalVolume = float(rec->tv1 | rec->tv2 << 8);
+ dailyData.avgBreathRate = float(rec->avgBreathRate);
+
+ dailyData.snores = float(rec->snores);
+ dailyData.timeInExPuf = float(rec->timeInExPuf) / 2.0F;
+ dailyData.timeInFL = float(rec->timeInFL) / 2.0F;
+ dailyData.timeInPB = float(rec->timeInPB) / 2.0F;
+ dailyData.maskFit = float(rec->maskFit) / 2.0F;
+ dailyData.indexOA = float(rec->indexOA) / 4.0F;
+ dailyData.indexCA = float(rec->indexCA) / 4.0F;
+ dailyData.indexHyp = float(rec->indexHyp) / 4.0F;
+
+ DailySummaries[dailyData.start_time] = dailyData;
+
+ } while (true);
+
+ rf.close();
+ qDebug() << "DV6 S.BIN processed" << rf.numread() << "records";
+
+ return true;
+}
+
+////////////////////////////////////////////////////////////////////////////////////////
+// Parse VER.BIN for model number, serial, etc.
+////////////////////////////////////////////////////////////////////////////////////////
+
+bool load6VersionInfo(const QString & path) {
+
+ QByteArray dataBA;
+ QByteArray str;
+
+ QFile f(path+"/VER.BIN");
+ info.series = "DV6";
+ info.brand = "DeVilbiss";
+
if (f.open(QIODevice::ReadOnly)) {
dataBA = f.readAll();
f.close();
int cnt = 0;
- data = (unsigned char *)dataBA.data();
for (int i=0; i< dataBA.size(); ++i) { // deliberately going one further to catch end condition
if ((dataBA.at(i) == 0) || (i >= dataBA.size()-1)) { // if null terminated or last byte
switch(cnt) {
case 1: // serial
+ info.serial = str;
break;
- case 2: // model
- info.model = str;
+ case 2: // modelnumber
+// info.model = str;
+ info.modelnumber = str;
+ info.modelnumber = info.modelnumber.trimmed();
+ for (int i = 0; i < (int)sizeof(testedModels); i++) {
+ if ( testedModels[i].model == info.modelnumber
+ || testedModels[i].model.isEmpty()) {
+ info.model = testedModels[i].name;
+ break;
+ }
+ }
break;
case 7: // ??? V025RN20170
break;
@@ -740,193 +1203,235 @@ int IntellipapLoader::OpenDV6(const QString & path)
str.append(dataBA[i]);
}
}
+ return true;
} else { // if (f.open(...)
// VER.BIN open failed
- return -1;
+ qWarning() << "Unable to open VER.BIN";
+ return false;
}
+}
- ////////////////////////////////////////////////////////////////////////////////////////
- // Creates Machine database record if it doesn't exist already
- ////////////////////////////////////////////////////////////////////////////////////////
- Machine *mach = p_profile->CreateMachine(info);
- if (mach == nullptr) {
- return -1;
- }
- qDebug() << "Opening DV6 (" << info.serial << ")" << "v" << vmaj << "." << vmin << "Min:" << min_pressure << "Max:" << max_pressure;
+////////////////////////////////////////////////////////////////////////////////////////
+// Create DV6_SessionInfo structures for each session and store in SessionData qmap
+////////////////////////////////////////////////////////////////////////////////////////
+
+int create6Sessions() {
+ SessionID sid = 0;
+ Session * sess;
+
+ for (auto sinfo=SessionData.begin(), end=SessionData.end(); sinfo != end; ++sinfo) {
+ sid = sinfo->begin;
+
+ if (mach->SessionExists(sid)) {
+ // skip already imported sessions..
+ qDebug() << "Session already exists" << QDateTime::fromTime_t(sid).toString("MM/dd/yyyy hh:mm:ss");
+
+ } else if (sinfo->sess == nullptr) {
+ // process new sessions
+ sess = new Session(mach, sid);
+#ifdef DEBUG6
+ qDebug() << "Creating session" << QDateTime::fromTime_t(sinfo->begin).toString("MM/dd/yyyy hh:mm:ss") << "to" << QDateTime::fromTime_t(sinfo->end).toString("MM/dd/yyyy hh:mm:ss");
+#endif
+
+ sinfo->sess = sess;
+ sinfo->dailyData = nullptr;
+ sinfo->written = 0;
+ sinfo->haveHighResData = false;
+
+ sess->really_set_first(quint64(sinfo->begin) * 1000L);
+ sess->really_set_last(quint64(sinfo->end) * 1000L);
+
+// rampstart[sid] = 0;
+// rampend[sid] = 0;
+ sess->SetChanged(true);
+
+ sess->AddEventList(INTELLIPAP_Unknown1, EVL_Event);
+ sess->AddEventList(INTELLIPAP_Unknown2, EVL_Event);
+
+ sess->AddEventList(CPAP_LeakTotal, EVL_Event);
+ sess->AddEventList(CPAP_MaxLeak, EVL_Event);
+ sess->AddEventList(CPAP_TidalVolume, EVL_Event);
+ sess->AddEventList(CPAP_MinuteVent, EVL_Event);
+ sess->AddEventList(CPAP_RespRate, EVL_Event);
+ sess->AddEventList(CPAP_Snore, EVL_Event);
+
+ sess->AddEventList(CPAP_Obstructive, EVL_Event);
+// sess->AddEventList(CPAP_VSnore, EVL_Event);
+ sess->AddEventList(CPAP_Hypopnea, EVL_Event);
+ sess->AddEventList(CPAP_NRI, EVL_Event);
+// sess->AddEventList(CPAP_LeakFlag, EVL_Event);
+ sess->AddEventList(CPAP_ExP, EVL_Event);
+ sess->AddEventList(CPAP_FlowLimit, EVL_Event);
- ////////////////////////////////////////////////////////////////////////////////////////
- // Open and parse session list and create a list of sessions to import
- ////////////////////////////////////////////////////////////////////////////////////////
-
-
- const int DV6_L_RecLength = 45;
- const int DV6_E_RecLength = 25;
- const int DV6_S_RecLength = 55;
- unsigned int ts1,ts2, lastts1;
-
- QMap summaryList; // QHash is faster, but QMap keeps order
-
- QDateTime epoch(QDate(2002, 1, 1), QTime(0, 0, 0), Qt::UTC); // Intellipap Epoch
- int ep = epoch.toTime_t();
-
-
- f.setFileName(newpath+"/S.BIN");
- if (f.open(QIODevice::ReadOnly)) {
- dataBA = f.readAll();
- f.close();
-
- data = (unsigned char *)dataBA.data();
-
- int records = dataBA.size() / DV6_S_RecLength;
-
- //data[0x11]; // Start of data block
- //data[0x12]; // Record count
- // First record is block header
- for (int r=1; rsessionlist.contains(ts1)) { // Check if already imported
- qDebug() << "Detected new Session" << ts1;
- R.sess = new Session(mach, ts1);
- R.sess->SetChanged(true);
-
- R.sess->really_set_first(qint64(ts1) * 1000L);
- R.sess->really_set_last(qint64(ts2) * 1000L);
-
- R.start_time = ts1;
- R.stop_time = ts2;
-
- R.atpressure_time = ((data[12] << 24) | (data[11] << 16) | (data[10] << 8) | data[9])+ep;
- R.hours = float(data[13]) / 10.0F;
- R.pressureSetMin = float(data[49]) / 10.0F;
- R.pressureSetMax = float(data[50]) / 10.0F;
-
- // The following stuff is not necessary to decode, but can be used to verify we are on the right track
- //data[14]... unknown
- R.pressureAvg = float(data[15]) / 10.0F;
- R.pressureMax = float(data[16]) / 10.0F;
- R.pressure50 = float(data[17]) / 10.0F;
- R.pressure90 = float(data[18]) / 10.0F;
- R.pressure95 = float(data[19]) / 10.0F;
- R.pressureStdDev = float(data[20]) / 10.0F;
- //data[21]... unknown
- R.leakAvg = float(data[22]) / 10.0F;
- R.leakMax = float(data[23]) / 10.0F;
- R.leak50= float(data[24]) / 10.0F;
- R.leak90 = float(data[25]) / 10.0F;
- R.leak95 = float(data[26]) / 10.0F;
- R.leakStdDev = float(data[27]) / 10.0F;
-
- R.tidalVolume = float(data[28] | data[29] << 8);
- R.avgBreathRate = float(data[30] | data[31] << 8);
-
- if (data[49] != data[50]) {
- R.sess->settings[CPAP_PressureMin] = R.pressureSetMin;
- R.sess->settings[CPAP_PressureMax] = R.pressureSetMax;
- R.sess->settings[CPAP_Mode] = MODE_APAP;
- } else {
- R.sess->settings[CPAP_Mode] = MODE_CPAP;
- R.sess->settings[CPAP_Pressure] = R.pressureSetMin;
- }
- R.hasMaskPressure = false;
- summaryList[ts1] = R;
- }
+ } else {
+ // If there is a duplicate session, null out the earlier session
+ // otherwise there will be a crash on shutdown.
+//?? for (int z = 0; z < SessionStart.size(); z++) {
+//?? if (SessionStart[z] == sid) {
+//?? SessionStart[z] = 0;
+//?? SessionEnd[z] = 0;
+//?? break;
+//?? }
+ qDebug() << sid << "has double ups" << QDateTime::fromTime_t(sid).toString("MM/dd/yyyy hh:mm:ss");
+
+ /*Session *sess=Sessions[sid];
+ Sessions.erase(Sessions.find(sid));
+ delete sess;
+ SessionStart[i]=0;
+ SessionEnd[i]=0; */
}
-
- } else { // if (f.open(...)
- // S.BIN open failed
- return -1;
}
+ qDebug() << "Created" << SessionData.size() << "sessions";
+ return SessionData.size();
+}
- QMap::iterator SR;
- const int DV6_R_RecLength = 117;
- const int DV6_R_HeaderSize = 55;
- f.setFileName(newpath+"/R.BIN");
- int numRrecs = (f.size()-DV6_R_HeaderSize) / DV6_R_RecLength;
- Session *sess = NULL;
- if (f.open(QIODevice::ReadOnly)) {
- // Let's not parse R all at once, it's huge
- dataBA = f.read(DV6_R_HeaderSize);
- if (dataBA.size() < DV6_R_HeaderSize) {
- // bit mean aborting on corrupt R file... but oh well
- return -1;
- }
+////////////////////////////////////////////////////////////////////////////////////////
+// Parse R.BIN for high resolution flow data
+////////////////////////////////////////////////////////////////////////////////////////
- sess = NULL;
- EventList * flow = NULL;
- EventList * pressure = NULL;
+bool load6HighResData (const QString & path) {
+
+ RollingFile rf;
+ Session *sess = nullptr;
+ unsigned int rec_ts1, previousRecBegin = 0;
+ bool inSession = false; // true if we are adding data to this session
+
+ if (!rf.open(path+"/R.BIN")) {
+ qWarning() << "DV6 Unable to open R.BIN";
+ return false;
+ }
+
+ qDebug() << "R.BIN starting at record" << rf.recnum();
+
+ sess = NULL;
+ EventList * flow = NULL;
+ EventList * pressure = NULL;
+ EventList * FLG = NULL;
+ EventList * snore = NULL;
// EventList * leak = NULL;
- EventList * OA = NULL;
- EventList * HY = NULL;
- EventList * NOA = NULL;
- EventList * EXP = NULL;
- EventList * FL = NULL;
- EventList * PB = NULL;
- EventList * VS = NULL;
- EventList * LL = NULL;
- EventList * RE = NULL;
- bool inOA = false, inH = false, inCA = false, inExP = false, inVS = false, inFL = false, inPB = false, inRE = false, inLL = false;
- qint64 OAstart = 0, OAend = 0;
- qint64 Hstart = 0, Hend = 0;
- qint64 CAstart = 0, CAend = 0;
- qint64 ExPstart = 0, ExPend = 0;
- qint64 VSstart = 0, VSend = 0;
- qint64 FLstart = 0, FLend = 0;
- qint64 PBstart = 0, PBend = 0;
- qint64 REstart =0, REend = 0;
- qint64 LLstart =0, LLend = 0;
- lastts1 = 0;
+/***
+ EventList * OA = NULL;
+ EventList * HY = NULL;
+ EventList * NOA = NULL;
+ EventList * EXP = NULL;
+ EventList * FL = NULL;
+ EventList * PB = NULL;
+ EventList * VS = NULL;
+ EventList * LL = NULL;
+ EventList * RE = NULL;
+ bool inOA = false, inH = false, inCA = false, inExP = false, inVS = false, inFL = false, inPB = false, inRE = false, inLL = false;
+ qint64 OAstart = 0, OAend = 0;
+ qint64 Hstart = 0, Hend = 0;
+ qint64 CAstart = 0, CAend = 0;
+ qint64 ExPstart = 0, ExPend = 0;
+ qint64 VSstart = 0, VSend = 0;
+ qint64 FLstart = 0, FLend = 0;
+ qint64 PBstart = 0, PBend = 0;
+ qint64 REstart =0, REend = 0;
+ qint64 LLstart =0, LLend = 0;
+// lastts1 = 0;
+***/
- SR = summaryList.begin();
- for (int r=0; r::iterator sinfo;
+ sinfo = SessionData.begin();
- DV6_S_Record *R = &SR.value();
+ do {
+ DV6_R_REC * R = (DV6_R_REC *) rf.get();
+ if (R == nullptr)
+ break;
- ts1 = ((data[4] << 24) | (data[3] << 16) | (data[2] << 8) | data[1]) + ep;
+ sess = sinfo->sess;
- if (flow && ((ts1 - lastts1) > 2)) {
+ // Get the timestamp from the record
+ rec_ts1 = convertTime(R->timestamp);
+
+ if (rec_ts1 < previousRecBegin) {
+ qWarning() << "R.BIN - Corruption/Out of sequence data found, skipping record" << rf.recnum() << ", prev"
+ << QDateTime::fromTime_t(previousRecBegin).toString("MM/dd/yyyy hh:mm:ss") << previousRecBegin
+ << "this"
+ << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss") << rec_ts1;
+ continue;
+ }
+
+ // Look for a gap in DV6_R records. They should be at two second intervals.
+ // If there is a gap, we are probably in a new session
+ if (inSession && ((rec_ts1 - previousRecBegin) > 2)) {
+ if (sess) {
sess->set_last(flow->last());
+ if (sess->first() == 0)
+ qWarning() << "R.BIN first = 0 - 1320";
+ EventDataType min = flow->Min();
+ EventDataType max = flow->Max();
+ sess->setMin(CPAP_FlowRate, min);
+ sess->setMax(CPAP_FlowRate, max);
+ sess->setPhysMax(CPAP_FlowRate, min); // not sure :/
+ sess->setPhysMin(CPAP_FlowRate, max);
+// sess->really_set_last(flow->last());
+ }
+ sess = nullptr;
+ flow = nullptr;
+ pressure = nullptr;
+ FLG = nullptr;
+ snore = nullptr;
+ inSession = false;
+ }
+
+ // Skip over sessions until we find one that this record is in
+ while (rec_ts1 > sinfo->end) {
+#ifdef DEBUG6
+ qDebug() << "R.BIN - skipping session" << QDateTime::fromTime_t(sinfo->begin).toString("MM/dd/yyyy hh:mm:ss")
+ << "looking for" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss")
+ << "record" << rf.recnum();
+#endif
+ if (inSession && sess) {
+ // update min and max
+ // then add to machine
+ if (sess->first() == 0)
+ qWarning() << "R.BIN first = 0 - 1284";
+ EventDataType min = flow->Min();
+ EventDataType max = flow->Max();
+ sess->setMin(CPAP_FlowRate, min);
+ sess->setMax(CPAP_FlowRate, max);
+ sess->setPhysMax(CPAP_FlowRate, min); // not sure :/
+ sess->setPhysMin(CPAP_FlowRate, max);
sess = nullptr;
flow = nullptr;
pressure = nullptr;
+ FLG = nullptr;
+ snore = nullptr;
+ inSession = false;
}
- lastts1=ts1;
-
- while (ts1 > R->stop_time) {
- if (flow && sess) {
- // update min and max
- // then add to machine
- sess = nullptr;
- flow = nullptr;
- pressure = nullptr;
- }
- SR++;
- if (SR == summaryList.end()) break;
- R = &SR.value();
- }
- if (SR == summaryList.end())
+ sinfo++;
+ if (sinfo == SessionData.end())
break;
+ }
- if (ts1 >= R->start_time) {
- if (!flow && R->sess) {
- flow = R->sess->AddEventList(CPAP_FlowRate, EVL_Waveform, 1.0f/60.0f, 0.0f, 0.0f, 0.0f, double(2000) / double(50));
- pressure = R->sess->AddEventList(CPAP_Pressure, EVL_Waveform, 0.1f, 0.0f, 0.0f, 0.0f, double(2000) / double(2));
- R->hasMaskPressure = true;
- //leak = R->sess->AddEventList(CPAP_Leak, EVL_Waveform, 1.0, 0.0, 0.0, 0.0, double(2000) / double(1));
+ previousRecBegin = rec_ts1;
+
+ // If we have data beyond last session, we are in trouble (for unknown reasons)
+ if (sinfo == SessionData.end()) {
+ qWarning() << "DV6 R.BIN import ran out of sessions to match flow data, record" << rf.recnum();
+ break;
+ }
+
+ // Check if record belongs in this session or a future session
+ if (!inSession && rec_ts1 <= sinfo->end) {
+ sess = sinfo->sess; // this is the Session we want
+ if (!inSession && sess) {
+ inSession = true;
+ flow = sess->AddEventList(CPAP_FlowRate, EVL_Waveform, 0.01f, 0.0f, 0.0f, 0.0f, double(2000) / double(50));
+ pressure = sess->AddEventList(CPAP_Pressure, EVL_Waveform, 0.1f, 0.0f, 0.0f, 0.0f, double(2000) / double(2));
+ FLG = sess->AddEventList(CPAP_FLG, EVL_Waveform, 1.0f, 0.0f, 0.0f, 0.0f, double(2000) / double(2));
+ snore = sess->AddEventList(CPAP_Snore, EVL_Waveform, 1.0f, 0.0f, 0.0f, 0.0f, double(2000) / double(2));
+ // sinfo->hasMaskPressure = true;
+ // leak = R->sess->AddEventList(CPAP_Leak, EVL_Waveform, 1.0, 0.0, 0.0, 0.0, double(2000) / double(1));
+ /***
OA = R->sess->AddEventList(CPAP_Obstructive, EVL_Event);
NOA = R->sess->AddEventList(CPAP_NRI, EVL_Event);
RE = R->sess->AddEventList(CPAP_RERA, EVL_Event);
@@ -936,318 +1441,334 @@ int IntellipapLoader::OpenDV6(const QString & path)
FL = R->sess->AddEventList(CPAP_FlowLimit, EVL_Event);
PB = R->sess->AddEventList(CPAP_PB, EVL_Event);
LL = R->sess->AddEventList(CPAP_LargeLeak, EVL_Event);
- }
- if (flow) {
- sess = R->sess;
- // starting at position 5 is 100 bytes, 16bit LE 25hz samples
- qint16 *wavedata = (qint16 *)(&data[5]);
-
- qint64 ti = qint64(ts1) * 1000;
-
- unsigned char d[2];
- d[0] = data[105];
- d[1] = data[106];
- flow->AddWaveform(ti+40000,wavedata,50,2000);
- pressure->AddWaveform(ti+40000, d, 2, 2000);
- // Fields data[107] && data[108] are bitfields default is 0x90, occasionally 0x98
-
- d[0] = data[107];
- d[1] = data[108];
-
- //leak->AddWaveform(ti+40000, d, 2, 2000);
-
-
- // Needs to track state to pull events out cleanly..
-
- //////////////////////////////////////////////////////////////////
- // High Leak
- //////////////////////////////////////////////////////////////////
-
- if (data[110] & 3) { // LL state 1st second
- if (!inLL) {
- LLstart = ti;
- inLL = true;
- }
- LLend = ti+1000L;
- } else {
- if (inLL) {
- inLL = false;
- LL->AddEvent(LLstart,(LLend-LLstart) / 1000L);
- LLstart = 0;
- }
- }
- if (data[114] & 3) {
- if (!inLL) {
- LLstart = ti+1000L;
- inLL = true;
- }
- LLend = ti+2000L;
-
- } else {
- if (inLL) {
- inLL = false;
- LL->AddEvent(LLstart,(LLend-LLstart) / 1000L);
- LLstart = 0;
- }
- }
-
-
- //////////////////////////////////////////////////////////////////
- // Obstructive Apnea
- //////////////////////////////////////////////////////////////////
-
- if (data[110] & 12) { // OA state 1st second
- if (!inOA) {
- OAstart = ti;
- inOA = true;
- }
- OAend = ti+1000L;
- } else {
- if (inOA) {
- inOA = false;
- OA->AddEvent(OAstart,(OAend-OAstart) / 1000L);
- OAstart = 0;
- }
- }
- if (data[114] & 12) {
- if (!inOA) {
- OAstart = ti+1000L;
- inOA = true;
- }
- OAend = ti+2000L;
-
- } else {
- if (inOA) {
- inOA = false;
- OA->AddEvent(OAstart,(OAend-OAstart) / 1000L);
- OAstart = 0;
- }
- }
-
-
- //////////////////////////////////////////////////////////////////
- // Hypopnea
- //////////////////////////////////////////////////////////////////
-
- if (data[110] & 192) {
- if (!inH) {
- Hstart = ti;
- inH = true;
- }
- Hend = ti + 1000L;
- } else {
- if (inH) {
- inH = false;
- HY->AddEvent(Hstart,(Hend-Hstart) / 1000L);
- Hstart = 0;
- }
- }
-
- if (data[114] & 192) {
- if (!inH) {
- Hstart = ti+1000L;
- inH = true;
- }
- Hend = ti + 2000L;
- } else {
- if (inH) {
- inH = false;
- HY->AddEvent(Hstart,(Hend-Hstart) / 1000L);
- Hstart = 0;
- }
- }
-
- //////////////////////////////////////////////////////////////////
- // Non Responding Apnea Event (Are these CA's???)
- //////////////////////////////////////////////////////////////////
- if (data[110] & 48) { // OA state 1st second
- if (!inCA) {
- CAstart = ti;
- inCA = true;
- }
- CAend = ti+1000L;
- } else {
- if (inCA) {
- inCA = false;
- NOA->AddEvent(CAstart,(CAend-CAstart) / 1000L);
- CAstart = 0;
- }
- }
- if (data[114] & 48) {
- if (!inCA) {
- CAstart = ti+1000L;
- inCA = true;
- }
- CAend = ti+2000L;
-
- } else {
- if (inCA) {
- inCA = false;
- NOA->AddEvent(CAstart,(CAend-CAstart) / 1000L);
- CAstart = 0;
- }
- }
-
- //////////////////////////////////////////////////////////////////
- // VSnore Event
- //////////////////////////////////////////////////////////////////
- if (data[109] & 3) { // OA state 1st second
- if (!inVS) {
- VSstart = ti;
- inVS = true;
- }
- VSend = ti+1000L;
- } else {
- if (inVS) {
- inVS = false;
- VS->AddEvent(VSstart,(VSend-VSstart) / 1000L);
- VSstart = 0;
- }
- }
- if (data[113] & 3) {
- if (!inVS) {
- VSstart = ti+1000L;
- inVS = true;
- }
- VSend = ti+2000L;
-
- } else {
- if (inVS) {
- inVS = false;
- VS->AddEvent(VSstart,(VSend-VSstart) / 1000L);
- VSstart = 0;
- }
- }
-
- //////////////////////////////////////////////////////////////////
- // Expiratory puff Event
- //////////////////////////////////////////////////////////////////
- if (data[109] & 12) { // OA state 1st second
- if (!inExP) {
- ExPstart = ti;
- inExP = true;
- }
- ExPend = ti+1000L;
- } else {
- if (inExP) {
- inExP = false;
- EXP->AddEvent(ExPstart,(ExPend-ExPstart) / 1000L);
- ExPstart = 0;
- }
- }
- if (data[113] & 12) {
- if (!inExP) {
- ExPstart = ti+1000L;
- inExP = true;
- }
- ExPend = ti+2000L;
-
- } else {
- if (inExP) {
- inExP = false;
- EXP->AddEvent(ExPstart,(ExPend-ExPstart) / 1000L);
- ExPstart = 0;
- }
- }
-
- //////////////////////////////////////////////////////////////////
- // Flow Limitation Event
- //////////////////////////////////////////////////////////////////
- if (data[109] & 48) { // OA state 1st second
- if (!inFL) {
- FLstart = ti;
- inFL = true;
- }
- FLend = ti+1000L;
- } else {
- if (inFL) {
- inFL = false;
- FL->AddEvent(FLstart,(FLend-FLstart) / 1000L);
- FLstart = 0;
- }
- }
- if (data[113] & 48) {
- if (!inFL) {
- FLstart = ti+1000L;
- inFL = true;
- }
- FLend = ti+2000L;
-
- } else {
- if (inFL) {
- inFL = false;
- FL->AddEvent(FLstart,(FLend-FLstart) / 1000L);
- FLstart = 0;
- }
- }
-
- //////////////////////////////////////////////////////////////////
- // Periodic Breathing Event
- //////////////////////////////////////////////////////////////////
- if (data[109] & 192) { // OA state 1st second
- if (!inPB) {
- PBstart = ti;
- inPB = true;
- }
- PBend = ti+1000L;
- } else {
- if (inPB) {
- inPB = false;
- PB->AddEvent(PBstart,(PBend-PBstart) / 1000L);
- PBstart = 0;
- }
- }
- if (data[113] & 192) {
- if (!inPB) {
- PBstart = ti+1000L;
- inPB = true;
- }
- PBend = ti+2000L;
-
- } else {
- if (inPB) {
- inPB = false;
- PB->AddEvent(PBstart,(PBend-PBstart) / 1000L);
- PBstart = 0;
- }
- }
-
- //////////////////////////////////////////////////////////////////
- // Respiratory Effort Related Arousal Event
- //////////////////////////////////////////////////////////////////
- if (data[111] & 48) { // OA state 1st second
- if (!inRE) {
- REstart = ti;
- inRE = true;
- }
- REend = ti+1000L;
- } else {
- if (inRE) {
- inRE = false;
- RE->AddEvent(REstart,(REend-REstart) / 1000L);
- REstart = 0;
- }
- }
- if (data[115] & 48) {
- if (!inRE) {
- REstart = ti+1000L;
- inRE = true;
- }
- REend = ti+2000L;
-
- } else {
- if (inRE) {
- inRE = false;
- RE->AddEvent(REstart,(REend-REstart) / 1000L);
- REstart = 0;
- }
- }
- }
+***/
}
-
-
}
- if (flow && sess) {
+ if (inSession) {
+ // Record breath and pressure waveforms
+ qint64 ti = qint64(rec_ts1) * 1000;
+ flow->AddWaveform(ti,R->breath,50,2000);
+ pressure->AddWaveform(ti, &R->pressure1, 2, 2000);
+ sinfo->haveHighResData = true;
+ if (sess->first() == 0)
+ qWarning() << "first = 0 - 1442";
+
+ //////////////////////////////////////////////////////////////////
+ // Show Flow Limitation Events as a graph
+ //////////////////////////////////////////////////////////////////
+ qint16 severity = (R->flags1[0] >> 4) & 0x03;
+ FLG->AddWaveform(ti, &severity, 1, 1000);
+ severity = (R->flags2[0] >> 4) & 0x03;
+ FLG->AddWaveform(ti+1000, &severity, 1, 1000);
+
+ //////////////////////////////////////////////////////////////////
+ // Show Snore Events as a graph
+ //////////////////////////////////////////////////////////////////
+ severity = R->flags1[0] & 0x03;
+ snore->AddWaveform(ti, &severity, 1, 1000);
+ severity = R->flags2[0] & 0x03;
+ snore->AddWaveform(ti+1000, &severity, 1, 1000);
+
+ /****
+ // Fields data[107] && data[108] are bitfields default is 0x90, occasionally 0x98
+
+ d[0] = data[107];
+ d[1] = data[108];
+
+ //leak->AddWaveform(ti+40000, d, 2, 2000);
+
+ // Needs to track state to pull events out cleanly..
+
+ //////////////////////////////////////////////////////////////////
+ // High Leak
+ //////////////////////////////////////////////////////////////////
+
+ if (data[110] & 3) { // LL state 1st second
+ if (!inLL) {
+ LLstart = ti;
+ inLL = true;
+ }
+ LLend = ti+1000L;
+ } else {
+ if (inLL) {
+ inLL = false;
+ LL->AddEvent(LLstart,(LLend-LLstart) / 1000L);
+ LLstart = 0;
+ }
+ }
+ if (data[114] & 3) {
+ if (!inLL) {
+ LLstart = ti+1000L;
+ inLL = true;
+ }
+ LLend = ti+2000L;
+
+ } else {
+ if (inLL) {
+ inLL = false;
+ LL->AddEvent(LLstart,(LLend-LLstart) / 1000L);
+ LLstart = 0;
+ }
+ }
+
+
+ //////////////////////////////////////////////////////////////////
+ // Obstructive Apnea
+ //////////////////////////////////////////////////////////////////
+
+ if (data[110] & 12) { // OA state 1st second
+ if (!inOA) {
+ OAstart = ti;
+ inOA = true;
+ }
+ OAend = ti+1000L;
+ } else {
+ if (inOA) {
+ inOA = false;
+ OA->AddEvent(OAstart,(OAend-OAstart) / 1000L);
+ OAstart = 0;
+ }
+ }
+ if (data[114] & 12) {
+ if (!inOA) {
+ OAstart = ti+1000L;
+ inOA = true;
+ }
+ OAend = ti+2000L;
+
+ } else {
+ if (inOA) {
+ inOA = false;
+ OA->AddEvent(OAstart,(OAend-OAstart) / 1000L);
+ OAstart = 0;
+ }
+ }
+
+
+ //////////////////////////////////////////////////////////////////
+ // Hypopnea
+ //////////////////////////////////////////////////////////////////
+
+ if (data[110] & 192) {
+ if (!inH) {
+ Hstart = ti;
+ inH = true;
+ }
+ Hend = ti + 1000L;
+ } else {
+ if (inH) {
+ inH = false;
+ HY->AddEvent(Hstart,(Hend-Hstart) / 1000L);
+ Hstart = 0;
+ }
+ }
+
+ if (data[114] & 192) {
+ if (!inH) {
+ Hstart = ti+1000L;
+ inH = true;
+ }
+ Hend = ti + 2000L;
+ } else {
+ if (inH) {
+ inH = false;
+ HY->AddEvent(Hstart,(Hend-Hstart) / 1000L);
+ Hstart = 0;
+ }
+ }
+
+ //////////////////////////////////////////////////////////////////
+ // Non Responding Apnea Event (Are these CA's???)
+ //////////////////////////////////////////////////////////////////
+ if (data[110] & 48) { // OA state 1st second
+ if (!inCA) {
+ CAstart = ti;
+ inCA = true;
+ }
+ CAend = ti+1000L;
+ } else {
+ if (inCA) {
+ inCA = false;
+ NOA->AddEvent(CAstart,(CAend-CAstart) / 1000L);
+ CAstart = 0;
+ }
+ }
+ if (data[114] & 48) {
+ if (!inCA) {
+ CAstart = ti+1000L;
+ inCA = true;
+ }
+ CAend = ti+2000L;
+
+ } else {
+ if (inCA) {
+ inCA = false;
+ NOA->AddEvent(CAstart,(CAend-CAstart) / 1000L);
+ CAstart = 0;
+ }
+ }
+
+ //////////////////////////////////////////////////////////////////
+ // VSnore Event
+ //////////////////////////////////////////////////////////////////
+ if (data[109] & 3) { // OA state 1st second
+ if (!inVS) {
+ VSstart = ti;
+ inVS = true;
+ }
+ VSend = ti+1000L;
+ } else {
+ if (inVS) {
+ inVS = false;
+ VS->AddEvent(VSstart,(VSend-VSstart) / 1000L);
+ VSstart = 0;
+ }
+ }
+ if (data[113] & 3) {
+ if (!inVS) {
+ VSstart = ti+1000L;
+ inVS = true;
+ }
+ VSend = ti+2000L;
+
+ } else {
+ if (inVS) {
+ inVS = false;
+ VS->AddEvent(VSstart,(VSend-VSstart) / 1000L);
+ VSstart = 0;
+ }
+ }
+
+ //////////////////////////////////////////////////////////////////
+ // Expiratory puff Event
+ //////////////////////////////////////////////////////////////////
+ if (data[109] & 12) { // OA state 1st second
+ if (!inExP) {
+ ExPstart = ti;
+ inExP = true;
+ }
+ ExPend = ti+1000L;
+ } else {
+ if (inExP) {
+ inExP = false;
+ EXP->AddEvent(ExPstart,(ExPend-ExPstart) / 1000L);
+ ExPstart = 0;
+ }
+ }
+ if (data[113] & 12) {
+ if (!inExP) {
+ ExPstart = ti+1000L;
+ inExP = true;
+ }
+ ExPend = ti+2000L;
+
+ } else {
+ if (inExP) {
+ inExP = false;
+ EXP->AddEvent(ExPstart,(ExPend-ExPstart) / 1000L);
+ ExPstart = 0;
+ }
+ }
+
+ //////////////////////////////////////////////////////////////////
+ // Flow Limitation Event
+ //////////////////////////////////////////////////////////////////
+ if (data[109] & 48) { // OA state 1st second
+ if (!inFL) {
+ FLstart = ti;
+ inFL = true;
+ }
+ FLend = ti+1000L;
+ } else {
+ if (inFL) {
+ inFL = false;
+ FL->AddEvent(FLstart,(FLend-FLstart) / 1000L);
+ FLstart = 0;
+ }
+ }
+ if (data[113] & 48) {
+ if (!inFL) {
+ FLstart = ti+1000L;
+ inFL = true;
+ }
+ FLend = ti+2000L;
+
+ } else {
+ if (inFL) {
+ inFL = false;
+ FL->AddEvent(FLstart,(FLend-FLstart) / 1000L);
+ FLstart = 0;
+ }
+ }
+
+ //////////////////////////////////////////////////////////////////
+ // Periodic Breathing Event
+ //////////////////////////////////////////////////////////////////
+ if (data[109] & 192) { // OA state 1st second
+ if (!inPB) {
+ PBstart = ti;
+ inPB = true;
+ }
+ PBend = ti+1000L;
+ } else {
+ if (inPB) {
+ inPB = false;
+ PB->AddEvent(PBstart,(PBend-PBstart) / 1000L);
+ PBstart = 0;
+ }
+ }
+ if (data[113] & 192) {
+ if (!inPB) {
+ PBstart = ti+1000L;
+ inPB = true;
+ }
+ PBend = ti+2000L;
+
+ } else {
+ if (inPB) {
+ inPB = false;
+ PB->AddEvent(PBstart,(PBend-PBstart) / 1000L);
+ PBstart = 0;
+ }
+ }
+
+ //////////////////////////////////////////////////////////////////
+ // Respiratory Effort Related Arousal Event
+ //////////////////////////////////////////////////////////////////
+ if (data[111] & 48) { // OA state 1st second
+ if (!inRE) {
+ REstart = ti;
+ inRE = true;
+ }
+ REend = ti+1000L;
+ } else {
+ if (inRE) {
+ inRE = false;
+ RE->AddEvent(REstart,(REend-REstart) / 1000L);
+ REstart = 0;
+ }
+ }
+ if (data[115] & 48) {
+ if (!inRE) {
+ REstart = ti+1000L;
+ inRE = true;
+ }
+ REend = ti+2000L;
+
+ } else {
+ if (inRE) {
+ inRE = false;
+ RE->AddEvent(REstart,(REend-REstart) / 1000L);
+ REstart = 0;
+ }
+ }
+***/
+ }
+
+ } while (true);
+
+ if (inSession && sess) {
+ /***
// Close event states if they are still open, and write event.
if (inH) HY->AddEvent(Hstart,(Hend-Hstart) / 1000L);
if (inOA) OA->AddEvent(OAstart,(OAend-OAstart) / 1000L);
@@ -1258,255 +1779,557 @@ int IntellipapLoader::OpenDV6(const QString & path)
if (inFL) FL->AddEvent(FLstart,(FLend-FLstart) / 1000L);
if (inPB) PB->AddEvent(PBstart,(PBend-PBstart) / 1000L);
if (inPB) RE->AddEvent(REstart,(REend-REstart) / 1000L);
+***/
+ // update min and max
+ // then add to machine
+ if (sess->first() == 0)
+ qWarning() << "R.BIN first = 0 - 1665";
+ EventDataType min = flow->Min();
+ EventDataType max = flow->Max();
+ sess->setMin(CPAP_FlowRate, min);
+ sess->setMax(CPAP_FlowRate, max);
- // update min and max
- // then add to machine
- EventDataType min = flow->Min();
- EventDataType max = flow->Max();
- sess->setMin(CPAP_FlowRate, min);
- sess->setMax(CPAP_FlowRate, max);
+ sess->setPhysMax(CPAP_FlowRate, min); // TODO: not sure :/
+ sess->setPhysMin(CPAP_FlowRate, max);
+ sess->really_set_last(flow->last());
- sess->setPhysMax(CPAP_FlowRate, min); // not sure :/
- sess->setPhysMin(CPAP_FlowRate, max);
- sess->really_set_last(flow->last());
-
- sess = NULL;
- flow = NULL;
- }
-
- f.close();
- data = (unsigned char *)dataBA.data();
- } else { // if (f.open(...)
- // L.BIN open failed
- return -1;
+ sess = nullptr;
+ inSession = false;
}
+ rf.close();
+ qDebug() << "DV6 R.BIN processed" << rf.numread() << "records";
+ return true;
+}
- /////////////////////////////////////////////////////////////////////////////////////////////
- /// Parse L.BIN and extract per-minute data.
- /////////////////////////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////////////////////
+// Parse L.BIN for per minute data
+////////////////////////////////////////////////////////////////////////////////////////
- EventList *leak = NULL;
- EventList *maxleak = NULL;
+bool load6PerMinute (const QString & path) {
+
+ RollingFile rf;
+ Session *sess = nullptr;
+ unsigned int rec_ts1, previousRecBegin = 0;
+ bool inSession = false; // true if we are adding data to this session
+
+ if (!rf.open(path+"/L.BIN")) {
+ qWarning() << "DV6 Unable to open L.BIN";
+ return false;
+ }
+
+ qDebug() << "L.BIN Minute Data starting at record" << rf.recnum();
+
+ EventList * leak = NULL;
+ EventList * maxleak = NULL;
EventList * RR = NULL;
EventList * Pressure = NULL;
EventList * TV = NULL;
EventList * MV = NULL;
+ QMap::iterator sinfo;
+ sinfo = SessionData.begin();
- sess = NULL;
- const int DV6_L_HeaderSize = 55;
- // Need to parse L.bin minute table to get graphs
- f.setFileName(newpath+"/L.BIN");
- if (f.open(QIODevice::ReadOnly)) {
- dataBA = f.readAll();
- if (dataBA.size() <= DV6_L_HeaderSize) {
- return -1;
+ // Walk through all the records
+ do {
+ DV6_L_REC * rec = (DV6_L_REC *) rf.get();
+ if (rec == nullptr)
+ break;
+
+ sess = sinfo->sess;
+
+ // Get the timestamp from the record
+ rec_ts1 = convertTime(rec->timestamp);
+
+ if (rec_ts1 < previousRecBegin) {
+ qWarning() << "L.BIN - Corruption/Out of sequence data found, skipping record" << rf.recnum() << ", prev"
+ << QDateTime::fromTime_t(previousRecBegin).toString("MM/dd/yyyy hh:mm:ss") << previousRecBegin
+ << "this"
+ << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss") << rec_ts1;
+ continue;
}
- f.close();
- data = (unsigned char *)dataBA.data()+DV6_L_HeaderSize;
- int numLrecs = (dataBA.size()-DV6_L_HeaderSize) / DV6_L_RecLength;
-
- SR = summaryList.begin();
-
- lastts1 = 0;
-
-
- if (SR != summaryList.end()) for (int r=0; r 60)) {
- sess->set_last(maxleak->last());
-
- sess = nullptr;
- leak = nullptr;
- maxleak = nullptr;
- MV = TV = RR = nullptr;
- Pressure = nullptr;
- }
- lastts1=ts1;
- while (ts1 > R->stop_time) {
- if (leak && sess) {
- // Close the open session and update the min and max
- sess->set_last(maxleak->last());
-
- sess = nullptr;
- leak = nullptr;
- maxleak = nullptr;
- MV = TV = RR = nullptr;
- Pressure = nullptr;
- }
- SR++;
- if (SR == summaryList.end()) break;
- R = &SR.value();
- }
- if (SR == summaryList.end())
- break;
-
- if (ts1 >= R->start_time) {
- if (!leak && R->sess) {
- qDebug() << "Adding Leak data for session" << R->sess->session() << "starting at" << ts1;
- leak = R->sess->AddEventList(CPAP_Leak, EVL_Event); // , 1.0, 0.0, 0.0, 0.0, double(60000) / double(1));
- maxleak = R->sess->AddEventList(CPAP_MaxLeak, EVL_Event);// , 1.0, 0.0, 0.0, 0.0, double(60000) / double(1));
- RR = R->sess->AddEventList(CPAP_RespRate, EVL_Event);
- MV = R->sess->AddEventList(CPAP_MinuteVent, EVL_Event);
- TV = R->sess->AddEventList(CPAP_TidalVolume, EVL_Event);
-
- if (!R->hasMaskPressure) {
- // Don't use this pressure if we have higher resolution available
- Pressure = R->sess->AddEventList(CPAP_Pressure, EVL_Event);
- }
- }
- if (leak) {
- sess = R->sess;
-
- qint64 ti = qint64(ts1) * 1000L;
-
- maxleak->AddEvent(ti, data[5]);
- leak->AddEvent(ti, data[6]);
- RR->AddEvent(ti, data[9]);
-
- if (Pressure) Pressure->AddEvent(ti, data[11] / 10.0f);
-
- unsigned tv = data[7] | data[8] << 8;
- MV->AddEvent(ti, data[10] );
- TV->AddEvent(ti, tv);
-
- if (!sess->channelExists(CPAP_FlowRate)) {
- // No flow rate, so lets grab this data...
- }
- }
-
- }
- } // for
- if (sess && leak) {
+ // Look for a gap in DV6_L records. They should be at one minute intervals.
+ // If there is a gap, we are probably in a new session
+ if (inSession && ((rec_ts1 - previousRecBegin) > 60)) {
+// qDebug() << "L.BIN record gap, current" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss")
+// << "previous" << QDateTime::fromTime_t(previousRecBegin).toString("MM/dd/yyyy hh:mm:ss");
sess->set_last(maxleak->last());
+ sess = nullptr;
+ leak = maxleak = MV = TV = RR = Pressure = nullptr;
+ inSession = false;
}
- } else { // if (f.open(...)
- // L.BIN open failed
- return -1;
+ // Skip over sessions until we find one that this record is in
+ while (rec_ts1 > sinfo->end) {
+#ifdef DEBUG6
+ qDebug() << "L.BIN - skipping session" << QDateTime::fromTime_t(sinfo->begin).toString("MM/dd/yyyy hh:mm:ss") << "looking for" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss");
+#endif
+ if (inSession && sess) {
+ // Close the open session and update the min and max
+ sess->set_last(maxleak->last());
+ sess = nullptr;
+ leak = maxleak = MV = TV = RR = Pressure = nullptr;
+ inSession = false;
+ }
+ sinfo++;
+ if (sinfo == SessionData.end())
+ break;
+ }
+
+ previousRecBegin = rec_ts1;
+
+ // If we have data beyond last session, we are in trouble (for unknown reasons)
+ if (sinfo == SessionData.end()) {
+ qWarning() << "DV6 L.BIN import ran out of sessions to match flow data";
+ break;
+ }
+
+ if (rec_ts1 < previousRecBegin) {
+ qWarning() << "L.BIN - Corruption/Out of sequence data found, stopping import, prev"
+ << QDateTime::fromTime_t(previousRecBegin).toString("MM/dd/yyyy hh:mm:ss")
+ << "this"
+ << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss");
+ break;
+ }
+
+ // Check if record belongs in this session or a future session
+ if (!inSession && rec_ts1 <= sinfo->end) {
+ sess = sinfo->sess; // this is the Session we want
+ if (!inSession && sess) {
+ leak = sess->AddEventList(CPAP_Leak, EVL_Event); // , 1.0, 0.0, 0.0, 0.0, double(60000) / double(1));
+ maxleak = sess->AddEventList(CPAP_LeakTotal, EVL_Event);// , 1.0, 0.0, 0.0, 0.0, double(60000) / double(1));
+ RR = sess->AddEventList(CPAP_RespRate, EVL_Event);
+ MV = sess->AddEventList(CPAP_MinuteVent, EVL_Event);
+ TV = sess->AddEventList(CPAP_TidalVolume, EVL_Event);
+
+ if (sess->last()/1000 > sinfo->end)
+ sinfo->end = sess->last()/1000;
+
+ if (!sinfo->haveHighResData) {
+ // Don't use this pressure if we already have higher resolution data
+ Pressure = sess->AddEventList(CPAP_Pressure, EVL_Event);
+ }
+ if (sinfo->mode == MODE_UNKNOWN) {
+ if (rec->pressureLimitLow != rec->pressureLimitHigh) {
+ sess->settings[CPAP_PressureMin] = rec->pressureLimitLow / 10.0f;
+ sess->settings[CPAP_PressureMax] = rec->pressureLimitHigh / 10.0f;
+// if available sess->settings[CPAP_PS) = ....
+ sess->settings[CPAP_Mode] = MODE_APAP;
+ sinfo->mode = MODE_APAP;
+ } else {
+ sess->settings[CPAP_Mode] = MODE_CPAP;
+ sess->settings[CPAP_Pressure] = rec->pressureLimitHigh / 10.0f;
+ sinfo->mode = MODE_CPAP;
+ }
+ inSession = true;
+ }
+ }
+ }
+ if (inSession) {
+ // Record breath and pressure waveforms
+ qint64 ti = qint64(rec_ts1) * 1000;
+ maxleak->AddEvent(ti, rec->maxLeak); //???
+ leak->AddEvent(ti, rec->avgLeak); //???
+ RR->AddEvent(ti, rec->breathRate);
+
+ if (Pressure) Pressure->AddEvent(ti, rec->avgPressure / 10.0f); // average pressure
+
+ unsigned tv = rec->tidalVolume6 + (rec->tidalVolume7 << 8);
+ MV->AddEvent(ti, rec->breathRate * tv / 1000.0 );
+ TV->AddEvent(ti, tv);
+
+ if (!sess->channelExists(CPAP_FlowRate)) {
+ // No flow rate, so lets grab this data...
+ }
+ }
+
+ } while (true);
+
+ if (sess && inSession) {
+ sess->set_last(maxleak->last());
}
+ rf.close();
+ qDebug() << "DV6 L.BIN processed" << rf.numread() << "records";
- // Now sessionList is populated with summary data, lets parse the Events list in E.BIN
+ return true;
+}
+////////////////////////////////////////////////////////////////////////////////////////
+// Parse E.BIN for event data
+////////////////////////////////////////////////////////////////////////////////////////
+
+bool load6EventData (const QString & path) {
+ RollingFile rf;
+
+ Session *sess = nullptr;
+ unsigned int rec_ts1, rec_ts2, previousRecBegin;
+ bool inSession = false; // true if we are adding data to this session
EventList * OA = nullptr;
EventList * CA = nullptr;
EventList * H = nullptr;
EventList * RE = nullptr;
- f.setFileName(newpath+"/E.BIN");
- const int DV6_E_HeaderSize = 55;
+ EventList * PB = nullptr;
+ EventList * LL = nullptr;
+ EventList * EP = nullptr;
+ EventList * SN = nullptr;
+ EventList * FL = nullptr;
- if (f.open(QIODevice::ReadOnly)) {
- dataBA = f.readAll();
- if (dataBA.size() == 0) {
- return -1;
- }
- f.close();
- data = (unsigned char *)dataBA.data()+DV6_E_HeaderSize;
- int numErecs = (dataBA.size()-DV6_E_HeaderSize) / DV6_E_RecLength;
+ if (!rf.open(path+"/E.BIN")) {
+ qWarning() << "DV6 Unable to open E.BIN";
+ return false;
+ }
- SR = summaryList.begin();
+ qDebug() << "Processing E.BIN starting at record" << rf.recnum();
- for (int r=0; r::iterator sinfo;
+ sinfo = SessionData.begin();
- if (SR == summaryList.end())
- break;
+ // Walk through all the records
+ do {
+ DV6_E_REC * rec = (DV6_E_REC *) rf.get();
+ if (rec == nullptr)
+ break;
+ sess = sinfo->sess;
- DV6_S_Record *R = &SR.value();
- while (ts1 > R->stop_time) {
- if (OA && sess) {
- // Close the open session and update the min and max
+ // Get the timestamp from the record
+ rec_ts1 = convertTime(rec->begin);
+ rec_ts2 = convertTime(rec->end);
+
+ // Skip over sessions until we find one that this record is in
+ while (rec_ts1 > sinfo->end) {
+#ifdef DEBUG6
+ qDebug() << "E.BIN - skipping session" << QDateTime::fromTime_t(sinfo->begin).toString("MM/dd/yyyy hh:mm:ss") << "looking for" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss");
+#endif
+ if (inSession) {
+ // Close the open session and update the min and max
+ if (OA->last() > 0)
sess->set_last(OA->last());
+ if (CA->last() > 0)
sess->set_last(CA->last());
+ if (H->last() > 0)
sess->set_last(H->last());
+ if (RE->last() > 0)
sess->set_last(RE->last());
+ if (PB->last() > 0)
+ sess->set_last(PB->last());
+ if (LL->last() > 0)
+ sess->set_last(LL->last());
+ if (EP->last() > 0)
+ sess->set_last(EP->last());
+ if (SN->last() > 0)
+ sess->set_last(SN->last());
+ if (FL->last() > 0)
+ sess->set_last(FL->last());
- sess = nullptr;
- H = CA = RE = OA = nullptr;
- }
- SR++;
- if (SR == summaryList.end()) break;
- R = &SR.value();
+ sess = nullptr;
+ H = CA = RE = OA = PB = LL = EP = SN = FL = nullptr;
+ inSession = false;
}
- if (SR == summaryList.end())
+ sinfo++;
+ if (sinfo == SessionData.end())
break;
-
- if (ts1 >= R->start_time) {
- if (!OA && R->sess) {
- qDebug() << "Adding Event data for session" << R->sess->session() << "starting at" << ts1;
- OA = R->sess->AddEventList(CPAP_Obstructive, EVL_Event);
- H = R->sess->AddEventList(CPAP_Hypopnea, EVL_Event);
- RE = R->sess->AddEventList(CPAP_RERA, EVL_Event);
- CA = R->sess->AddEventList(CPAP_ClearAirway, EVL_Event);
- }
- if (OA) {
- sess = R->sess;
-
- qint64 ti = qint64(ts1) * 1000L;
- int code = data[13];
- switch (code) {
- case 1:
- CA->AddEvent(ti, data[17]);
- break;
- case 2:
- OA->AddEvent(ti, data[17]);
- break;
- case 4:
- H->AddEvent(ti, data[17]);
- break;
- case 5:
- RE->AddEvent(ti, data[17]);
- break;
- default:
- break;
- }
- }
-
-
- } // for
}
- if (sess && OA) {
- sess->set_last(OA->last());
- sess->set_last(CA->last());
- sess->set_last(H->last());
- sess->set_last(RE->last());
+ previousRecBegin = rec_ts1;
+
+ // If we have data beyond last session, we are in trouble (for unknown reasons)
+ if (sinfo == SessionData.end()) {
+ qWarning() << "DV6 E.BIN import ran out of sessions to match flow data";
+ break;
}
- } else { // if (f.open(...)
- // E.BIN open failed
+
+ if (rec_ts1 < previousRecBegin) {
+ qWarning() << "E.BIN - Corruption/Out of sequence data found, stopping import, prev"
+ << QDateTime::fromTime_t(previousRecBegin).toString("MM/dd/yyyy hh:mm:ss")
+ << "this"
+ << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss");
+ break;
+ }
+
+ // Check if record belongs in this session or a future session
+ if (!inSession && rec_ts1 <= sinfo->end) {
+ sess = sinfo->sess; // this is the Session we want
+ if (!inSession && sess) {
+ OA = sess->AddEventList(CPAP_Obstructive, EVL_Event);
+ H = sess->AddEventList(CPAP_Hypopnea, EVL_Event);
+ RE = sess->AddEventList(CPAP_RERA, EVL_Event);
+ CA = sess->AddEventList(CPAP_ClearAirway, EVL_Event);
+ PB = sess->AddEventList(CPAP_PB, EVL_Event);
+ LL = sess->AddEventList(CPAP_LargeLeak, EVL_Event);
+ EP = sess->AddEventList(CPAP_ExP, EVL_Event);
+// SN = sess->AddEventList(CPAP_VSnore, EVL_Event);
+ FL = sess->AddEventList(CPAP_FlowLimit, EVL_Event);
+
+ SN = sess->AddEventList(CPAP_Snore, EVL_Waveform, 1.0f, 0.0f, 0.0f, 0.0f, double(2000) / double(2));
+ inSession = true;
+ }
+ }
+ if (inSession) {
+ qint64 duration = rec_ts2 - rec_ts1;
+ // We make an ad hoc adjustment to the start time so that the event lines up better with the flow graph
+ // TODO: We don't know what is really going on here. Is it sloppiness on the part of the DV6 in recording time stamps?
+ qint64 ti = qint64(rec_ts1 - (duration/2)) * 1000L;
+ if (duration < 0) {
+ qDebug() << "E.BIN at" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss")
+ << "reports duration of" << duration
+ << "ending" << QDateTime::fromTime_t(rec_ts2).toString("MM/dd/yyyy hh:mm:ss");
+ }
+ int code = rec->event_type;
+ //////////////////////////////////////////////////////////////////
+ // Show Snore Events as a graph
+ //////////////////////////////////////////////////////////////////
+ if (code == 9) {
+ qint16 severity = rec->event_severity;
+ SN->AddWaveform(ti, &severity, 1, duration*1000);
+ }
+ if (rec->event_severity >= 3)
+ switch (code) {
+ case 1:
+ CA->AddEvent(ti, duration);
+ break;
+ case 2:
+ OA->AddEvent(ti, duration);
+// qDebug() << "E.BIN - OA" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss") << "duration" << duration << "r" << rf.recnum();
+ break;
+ case 4:
+ H->AddEvent(ti, duration);
+ break;
+ case 5:
+ RE->AddEvent(ti, duration);
+// qDebug() << "E.BIN - RERA" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss") << "duration" << duration << "r" << rf.recnum();
+ break;
+ case 8: // snore
+ SN->AddEvent(ti, duration);
+// qDebug() << "E.BIN - Snore" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss") << "duration" << duration << "r" << rf.recnum();
+ break;
+ case 9: // expiratory puff
+ EP->AddEvent(ti, duration);
+// qDebug() << "E.BIN - exhale puff" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss") << "duration" << duration << "r" << rf.recnum();
+ break;
+ case 10: // flow limitation
+ FL->AddEvent(ti, duration);
+// qDebug() << "E.BIN - flow limit" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss") << "duration" << duration << "r" << rf.recnum();
+ break;
+ case 11: // periodic breathing
+ PB->AddEvent(ti, duration);
+ break;
+ case 12: // large leaks
+ LL->AddEvent(ti, duration);
+// qDebug() << "E.BIN - large leak" << QDateTime::fromTime_t(rec_ts1).toString("MM/dd/yyyy hh:mm:ss") << "duration" << duration << "r" << rf.recnum();
+ break;
+ case 13: // pressure change
+ break;
+ case 14: // start of session
+ break;
+ default:
+ break;
+ }
+ }
+ } while (true);
+
+ rf.close();
+ qDebug() << "DV6 E.BIN processed" << rf.numread() << "records";
+
+ return true;
+}
+
+////////////////////////////////////////////////////////////////////////////////////////
+// Finalize data and add to database
+////////////////////////////////////////////////////////////////////////////////////////
+
+int addSessions() {
+
+ for (auto si=SessionData.begin(), end=SessionData.end(); si != end; ++si) {
+ Session * sess = si.value().sess;
+
+ if (sess) {
+ if ( ! mach->AddSession(sess) ) {
+ qWarning() << "Session" << sess->session() << "was not addded";
+ }
+#ifdef DEBUG6
+ else
+ qDebug() << "Added session" << sess->session() << QDateTime::fromTime_t(sess->session()).toString("MM/dd/yyyy hh:mm:ss");;
+#endif
+
+ // Update indexes, process waveform and perform flagging
+ sess->UpdateSummaries();
+
+ // Save is not threadsafe
+ sess->Store(mach->getDataPath());
+
+ // Unload them from memory
+ sess->TrashEvents();
+ } else
+ qWarning() << "addSessions: session pointer is null";
+ }
+
+ return SessionData.size();
+
+}
+
+// Returns empty QByteArray() on failure.
+QByteArray fileChecksum(const QString &fileName,
+ QCryptographicHash::Algorithm hashAlgorithm)
+{
+ QFile f(fileName);
+ if (f.open(QFile::ReadOnly)) {
+ QCryptographicHash hash(hashAlgorithm);
+ if (hash.addData(&f)) {
+ return hash.result();
+ }
+ }
+ return QByteArray();
+}
+
+/****
+// Return the OSCAR date that the last data was written.
+// This will be considered to be the last day for which we have any data.
+// Adjust to get the correct date for sessions starting after midnight.
+QDate getLastDate () {
+ return QDate();
+}
+
+// Return date used within OSCAR, assuming day ends at noon
+QDate getOscarDate (QDateTime dt) {
+ QDate d = dt.date();
+ QTime tm = dt.time();
+ if (tm.hour() < 11)
+ d = d.addDays(-1);
+ return d;
+}
+***/
+
+////////////////////////////////////////////////////////////////////////////////////////
+// Create backup of input files
+// Create dated backup files when necesaary
+////////////////////////////////////////////////////////////////////////////////////////
+
+bool backup6 (const QString & path) {
+
+ // Are backups enabled?
+ if (!p_profile->session->backupCardData())
+ return true;
+
+ QString backup_path = mach->getBackupPath();
+ QString history_path = backup_path + "/DV6/HISTORY";
+
+ // Compare QDirs rather than QStrings because separators may be different, especially on Windows.
+ // We want to check whether import and backup paths are the same, regardless of variations in the string representations.
+ QDir ipath(path);
+ QDir bpath(backup_path);
+
+ if (ipath == bpath) {
+ // Don't create backups if importing from backup folder
+ rebuild_from_backups = true;
+ return true;
+ }
+
+ if ( ! bpath.exists()) {
+ if ( ! bpath.mkpath(backup_path) ) {
+ qWarning() << "Could not create DV6 backup directory" << backup_path;
+ return false;
+ }
+ }
+
+ // Copy input data to backup location
+ copyPath(ipath.absolutePath(), bpath.absolutePath());
+
+ // Create history directory for dated backups
+ QDir hpath(history_path);
+ if ( ! hpath.exists())
+ if ( ! hpath.mkpath(history_path)) {
+ qWarning() << "Could not create DV6 archive directory" << history_path;
+ return false;
+ }
+
+ // Create archive of settings file if needed (SET.BIN)
+ bool backup_settings = true;
+
+ QStringList filters;
+ filters << "set_*.bin";
+ hpath.setNameFilters(filters);
+ hpath.setFilter(QDir::Files);
+ QDir::Name | QDir::Reversed;
+ QStringList fileNames = hpath.entryList(); // Get list of files
+ if (! fileNames.isEmpty()) {
+ QString lastFile = fileNames.first();
+ QString newFile = ipath.absolutePath() + "/set.bin";
+ qDebug() << "last settings file is" << lastFile << "new file is" << newFile;
+ QByteArray newMD5 = fileChecksum(newFile, QCryptographicHash::Md5);
+ QByteArray oldMD5 = fileChecksum(lastFile, QCryptographicHash::Md5);
+ if (newMD5 == oldMD5)
+ backup_settings = false;
+ }
+
+ if (backup_settings) {
+ QString newFile = hpath.absolutePath() + "/set-" + "1234" + ".bin";
+ qDebug() << "history filename is" << newFile;
+ }
+
+ // We're done!
+ return true;
+}
+
+////////////////////////////////////////////////////////////////////////////////////////
+// Open a DV6 SD card, parse everything, add to OSCAR database
+////////////////////////////////////////////////////////////////////////////////////////
+
+int IntellipapLoader::OpenDV6(const QString & path)
+{
+ QString newpath = path + DV6_DIR;
+
+ // Prime the machine database's info field with stuff relevant to this machine
+ info = newInfo();
+
+ // VER.BIN - Parse model number, serial, etc.
+ if (!load6VersionInfo(newpath))
+ return -1;
+
+ // Now, create Machine database record if it doesn't exist already
+ mach = p_profile->CreateMachine(info);
+ if (mach == nullptr) {
+ qWarning() << "Could not create Machine data structure";
return -1;
}
- QMap::iterator it;
+ // SET.BIN - Parse settings file (which is only the latest settings)
+ if (!load6Settings(newpath))
+ return -1;
- for (it=summaryList.begin(); it!= summaryList.end(); ++it) {
- Session * sess = it.value().sess;
+ // S.BIN - Open and parse day summary list and create a list of days
+ if (!load6DailySummaries(newpath))
+ return -1;
- mach->AddSession(sess);
+ // Back up data files (must do after parsing VER.BIN, S.BIN, and creating Machine)
+ if (!backup6(path))
+ return -1;
- // Update indexes, process waveform and perform flagging
- sess->UpdateSummaries();
+ // U.BIN - Open and parse session list and create a list of session times
+ // (S.BIN must already be loaded)
+ if (!load6Sessions(newpath))
+ return -1;
- // Save is not threadsafe
- sess->Store(mach->getDataPath());
+ // Create OSCAR session list from session times and summary data
+ if (create6Sessions() <= 0)
+ return -1;
- // Unload them from memory
- sess->TrashEvents();
+ // R.BIN - Open and parse flow data
+ if (!load6HighResData(newpath))
+ return -1;
- }
+ // L.BIN - Open and parse per minute data
+ if (!load6PerMinute(newpath))
+ return -1;
+ // E.BIN - Open and parse event data
+ if (!load6EventData(newpath))
+ return -1;
- return summaryList.size();
+ // Finalize input
+ return addSessions();
}
int IntellipapLoader::Open(const QString & dirpath)
@@ -1560,11 +2383,11 @@ void IntellipapLoader::initChannels()
bool intellipap_initialized = false;
void IntellipapLoader::Register()
{
- if (intellipap_initialized) { return; }
-
- qDebug() << "Registering IntellipapLoader";
- RegisterLoader(new IntellipapLoader());
- //InitModelMap();
- intellipap_initialized = true;
-
+ if (!intellipap_initialized) {
+ qDebug() << "Registering IntellipapLoader";
+ RegisterLoader(new IntellipapLoader());
+ //InitModelMap();
+ intellipap_initialized = true;
+ }
+ return;
}
diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.cpp b/oscar/SleepLib/loader_plugins/prs1_loader.cpp
index 7c9e6f95..758ff8e0 100644
--- a/oscar/SleepLib/loader_plugins/prs1_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/prs1_loader.cpp
@@ -314,6 +314,7 @@ static const PRS1TestedModel s_PRS1TestedModels[] = {
{ "1030X150", 3, 6, "DreamStation BiPAP S/T 30 with AAM" },
{ "1130X110", 3, 6, "DreamStation BiPAP AVAPS 30" },
{ "1131X150", 3, 6, "DreamStation BiPAP AVAPS 30 AE" },
+ { "1130X200", 3, 6, "DreamStation BiPAP AVAPS 30" },
{ "", 0, 0, "" },
};
@@ -6213,7 +6214,7 @@ bool PRS1DataChunk::ParseSettingsF3V6(const unsigned char* data, int size)
case 2: // Breath Rate (fixed BPM)
breath_rate = data[pos+1];
timed_inspiration = data[pos+2];
- if (breath_rate < 9 || breath_rate > 13) UNEXPECTED_VALUE(breath_rate, "9-13");
+ if (breath_rate < 9 || breath_rate > 15) UNEXPECTED_VALUE(breath_rate, "9-15");
if (timed_inspiration < 8 || timed_inspiration > 20) UNEXPECTED_VALUE(timed_inspiration, "8-20");
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_BACKUP_BREATH_MODE, PRS1Backup_Fixed));
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_BACKUP_BREATH_RATE, breath_rate));
@@ -6248,17 +6249,22 @@ bool PRS1DataChunk::ParseSettingsF3V6(const unsigned char* data, int size)
// Rise time
if (data[pos] < 1 || data[pos] > 6) UNEXPECTED_VALUE(data[pos], "1-6"); // 1-6 have been seen
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_RISE_TIME, data[pos]));
+ } else {
+ UNEXPECTED_VALUE(flexmode, "BiFlex or RiseTime");
}
// Timed inspiration specified in the backup breath rate.
break;
- case 0x2f: // Rise Time lock? (was flex lock on F0V6, 0x80 for locked)
+ case 0x2f: // Flex / Rise Time lock
CHECK_VALUE(len, 1);
- if (cpapmode == PRS1_MODE_S) {
+ if (flexmode == FLEX_BiFlex) {
+ CHECK_VALUE(cpapmode, PRS1_MODE_S);
CHECK_VALUES(data[pos], 0, 0x80); // Bi-Flex Lock
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_FLEX_LOCK, data[pos] != 0));
+ } else if (flexmode == FLEX_RiseTime) {
+ CHECK_VALUES(data[pos], 0, 0x80); // Rise Time Lock
+ this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_RISE_TIME_LOCK, data[pos] != 0));
} else {
- CHECK_VALUE(data[pos], 0); // Rise Time Lock? not yet observed on F3V6
- //this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_RISE_TIME_LOCK, data[pos] != 0));
+ UNEXPECTED_VALUE(flexmode, "BiFlex or RiseTime");
}
break;
case 0x35: // Humidifier setting
@@ -7147,7 +7153,7 @@ void PRS1DataChunk::ParseHumidifierSettingV3(unsigned char byte1, unsigned char
} else if (humidadaptive) {
// All humidity levels seen.
} else if (humidfixed) {
- if (humidlevel == 0) UNEXPECTED_VALUE(humidlevel, "1-5");
+ // All humidity levels seen.
}
}
}
diff --git a/oscar/SleepLib/loader_plugins/resmed_loader.cpp b/oscar/SleepLib/loader_plugins/resmed_loader.cpp
index 33fd4798..2183be01 100644
--- a/oscar/SleepLib/loader_plugins/resmed_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/resmed_loader.cpp
@@ -181,6 +181,7 @@ void ResmedLoader::initChannels()
chan->addOption(0, STR_TR_Off);
chan->addOption(1, STR_TR_On);
+ chan->addOption(2, QObject::tr("Auto"));
// Setup ResMeds signal name translation map
setupResMedTranslationMap();
diff --git a/oscar/SleepLib/loader_plugins/somnopose_loader.cpp b/oscar/SleepLib/loader_plugins/somnopose_loader.cpp
index 04a56765..40f45de0 100644
--- a/oscar/SleepLib/loader_plugins/somnopose_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/somnopose_loader.cpp
@@ -7,10 +7,11 @@
* for more details. */
//********************************************************************************************
-// IMPORTANT!!!
-//********************************************************************************************
-// Please INCREMENT the somnopose_data_version in somnopose_loader.h when making changes to this loader
-// that change loader behaviour or modify channels.
+// Please only INCREMENT the somnopose_data_version in somnopose_loader.h when making changes
+// that change loader behaviour or modify channels in a manner that fixes old data imports.
+// Note that changing the data version will require a reimport of existing data for which OSCAR
+// does not keep a backup - so it should be avoided if possible.
+// i.e. there is no need to change the version when adding support for new devices
//********************************************************************************************
#include
@@ -68,12 +69,24 @@ int SomnoposeLoader::OpenFile(const QString & filename)
// Read header line and determine order of fields
QString hdr = ts.readLine();
QStringList headers = hdr.split(",");
+ QString model = "";
+ QString serial = "";
- int col_inclination = -1, col_orientation = -1, col_timestamp = -1;
+ int col_inclination = -1, col_orientation = -1, col_timestamp = -1, col_movement = -1;
int hdr_size = headers.size();
for (int i = 0; i < hdr_size; i++) {
+ // Optional header model=
+ if (headers.at(i).startsWith("model=", Qt::CaseInsensitive)) {
+ model=headers.at(i).split("=")[1];
+ }
+
+ // Optional header serial=
+ if (headers.at(i).startsWith("serial=", Qt::CaseInsensitive)) {
+ serial=headers.at(i).split("=")[1];
+ }
+
if (headers.at(i).compare("timestamp", Qt::CaseInsensitive) == 0) {
col_timestamp = i;
}
@@ -85,11 +98,20 @@ int SomnoposeLoader::OpenFile(const QString & filename)
if (headers.at(i).compare("orientation", Qt::CaseInsensitive) == 0) {
col_orientation = i;
}
+
+ if (headers.at(i).compare("movement", Qt::CaseInsensitive) == 0) {
+ col_movement = i;
+ }
}
// Check we have all fields available
- if ((col_timestamp < 0) || (col_inclination < 0) || (col_orientation < 0)) {
- qDebug() << "Header missing one of timestamp, inclination, or orientation";
+ if (col_timestamp < 0) {
+ qDebug() << "Header missing timestamp";
+ return 0;
+ }
+
+ if ((col_inclination < 0) && (col_orientation < 0) && (col_movement < 0)) {
+ qDebug() << "Header missing all of inclination, orientation, movement (at least one must be present)";
return 0;
}
@@ -97,18 +119,20 @@ int SomnoposeLoader::OpenFile(const QString & filename)
qint64 ep = qint64(epoch.toTime_t()+epoch.offsetFromUtc()) * 1000, time;
qDebug() << "Epoch starts at" << epoch.toString();
- double timestamp, orientation, inclination;
+ double timestamp, orientation=0, inclination=0, movement=0;
QString data;
QStringList fields;
- bool ok;
+ bool ok, orientation_ok, inclination_ok, movement_ok;
bool first = true;
MachineInfo info = newInfo();
+ info.model = model;
+ info.serial = serial;
Machine *mach = p_profile->CreateMachine(info);
Session *sess = nullptr;
SessionID sid;
- EventList *ev_orientation = nullptr, *ev_inclination = nullptr;
+ EventList *ev_orientation = nullptr, *ev_inclination = nullptr, *ev_movement = nullptr;
while (!(data = ts.readLine()).isEmpty()) {
fields = data.split(",");
@@ -121,14 +145,21 @@ int SomnoposeLoader::OpenFile(const QString & filename)
if (!ok) { continue; }
- orientation = fields[col_orientation].toDouble(&ok);
+ if (col_orientation >= 0) {
+ orientation = fields[col_orientation].toDouble(&orientation_ok);
+ }
- if (!ok) { continue; }
+ if (col_inclination >= 0) {
+ inclination = fields[col_inclination].toDouble(&inclination_ok);
+ }
- inclination = fields[col_inclination].toDouble(&ok);
-
- if (!ok) { continue; }
+ if (col_movement >= 0) {
+ movement = fields[col_movement].toDouble(&movement_ok);
+ }
+ if (!orientation_ok && !inclination_ok && !movement_ok) {
+ continue;
+ }
// convert to milliseconds since epoch
time = (timestamp * 1000.0) + ep;
@@ -137,35 +168,58 @@ int SomnoposeLoader::OpenFile(const QString & filename)
qDebug() << "First timestamp is" << QDateTime::fromMSecsSinceEpoch(time).toString();
if (mach->SessionExists(sid)) {
- return 0; // Already imported
+ qDebug() << "File " << filename << " already loaded... skipping";
+ return -1; // Already imported
}
sess = new Session(mach, sid);
sess->really_set_first(time);
- ev_orientation = sess->AddEventList(POS_Orientation, EVL_Event, 1, 0, 0, 0);
- ev_inclination = sess->AddEventList(POS_Inclination, EVL_Event, 1, 0, 0, 0);
+ if (col_orientation >= 0) {
+ ev_orientation = sess->AddEventList(POS_Orientation, EVL_Event, 1, 0, 0, 0);
+ }
+ if (col_inclination >= 0) {
+ ev_inclination = sess->AddEventList(POS_Inclination, EVL_Event, 1, 0, 0, 0);
+ }
+ if (col_movement >= 0) {
+ ev_movement = sess->AddEventList(POS_Movement, EVL_Event, 1, 0, 0, 0);
+ }
first = false;
}
sess->set_last(time);
- ev_orientation->AddEvent(time, orientation);
- ev_inclination->AddEvent(time, inclination);
+ if (ev_orientation && orientation_ok) {
+ ev_orientation->AddEvent(time, orientation);
+ }
+ if (ev_inclination && inclination_ok) {
+ ev_inclination->AddEvent(time, inclination);
+ }
+ if (ev_movement && movement_ok) {
+ ev_movement->AddEvent(time, movement);
+ }
}
if (sess) {
- if (ev_orientation && ev_inclination) {
+ if (ev_orientation) {
sess->setMin(POS_Orientation, ev_orientation->Min());
sess->setMax(POS_Orientation, ev_orientation->Max());
+ }
+ if (ev_inclination) {
sess->setMin(POS_Inclination, ev_inclination->Min());
sess->setMax(POS_Inclination, ev_inclination->Max());
}
-
+ if (ev_movement) {
+ sess->setMin(POS_Movement, ev_movement->Min());
+ sess->setMax(POS_Movement, ev_movement->Max());
+ }
sess->really_set_last(time);
sess->SetChanged(true);
mach->AddSession(sess);
mach->Save();
+ // Adding these to hopefully make data persistent...
+ mach->SaveSummaryCache();
+ p_profile->StoreMachines();
}
return true;
diff --git a/oscar/SleepLib/loader_plugins/viatom_loader.cpp b/oscar/SleepLib/loader_plugins/viatom_loader.cpp
index 1f61a8c3..19147c4e 100644
--- a/oscar/SleepLib/loader_plugins/viatom_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/viatom_loader.cpp
@@ -2,16 +2,18 @@
*
* Copyright (c) 2019-2020 The OSCAR Team
* (Initial importer written by dave madden )
+ * Modified 02/21/2021 to allow for CheckMe device data files by Crimson Nape
*
* This file is subject to the terms and conditions of the GNU General Public
* License. See the file COPYING in the main directory of the source code
* for more details. */
//********************************************************************************************
-// IMPORTANT!!!
-//********************************************************************************************
-// Please INCREMENT the viatom_data_version in viatom_loader.h when making changes to this loader
-// that change loader behaviour or modify channels.
+// Please only INCREMENT the viatom_data_version in viatom_loader.h when making changes
+// that change loader behaviour or modify channels in a manner that fixes old data imports.
+// Note that changing the data version will require a reimport of existing data for which OSCAR
+// does not keep a backup - so it should be avoided if possible.
+// i.e. there is no need to change the version when adding support for new devices
//********************************************************************************************
#include
@@ -39,7 +41,7 @@ ViatomLoader::Open(const QString & dirpath)
int imported = 0;
int found = 0;
s_unexpectedMessages.clear();
-
+
if (QFileInfo(dirpath).isDir()) {
QDir dir(dirpath);
dir.setFilter(QDir::NoDotAndDotDot | QDir::Files | QDir::Hidden);
@@ -95,7 +97,7 @@ ViatomLoader::Open(const QString & dirpath)
bool ViatomLoader::OpenFile(const QString & filename)
{
Machine* mach = nullptr;
-
+
Session* sess = ParseFile(filename);
if (sess) {
SaveSessionToDatabase(sess);
@@ -124,7 +126,7 @@ Session* ViatomLoader::ParseFile(const QString & filename)
QString foldername = QFileInfo(filename).dir().dirName();
if (foldername.length() >= 9) {
bool numeric;
- foldername.right(4).toInt(&numeric);
+ foldername.rightRef(4).toInt(&numeric);
if (numeric) {
info.serial = foldername;
}
@@ -167,7 +169,7 @@ Session* ViatomLoader::ParseFile(const QString & filename)
void ViatomLoader::SaveSessionToDatabase(Session* sess)
{
Machine* mach = sess->machine();
-
+
sess->SetChanged(true);
mach->AddSession(sess);
}
@@ -192,7 +194,7 @@ void ViatomLoader::EndEventList(ChannelID channel, qint64 /*t*/)
// The below would be needed for square charts if the first sample represents
// the 4 seconds following the starting timestamp:
//C->AddEvent(t, m_importLastValue[channel]);
-
+
// Mark this channel's event list as ended.
m_importChannels[channel] = nullptr;
}
@@ -248,6 +250,7 @@ static QString dur(qint64 msecs)
#define CHECK_VALUES(SRC, VAL1, VAL2) if ((SRC) != (VAL1) && (SRC) != (VAL2)) UNEXPECTED_VALUE(SRC, #VAL1 " or " #VAL2)
// for more than 2 values, just write the test manually and use UNEXPECTED_VALUE if it fails
+
ViatomFile::ViatomFile(QFile & file) : m_file(file)
{
}
@@ -270,10 +273,23 @@ bool ViatomFile::ParseHeader()
int min = header[7];
int sec = header[8];
+
+ /* CN - Changed the if statement to a switch to accomdate additional Viatom/Wellue signatures in the future
if (sig != 0x0003) {
qDebug() << m_file.fileName() << "invalid signature for Viatom data file" << sig;
return false;
}
+ CN */
+ switch (sig){
+ case 0x0003:
+ case 0x0005:
+ break;
+ default:
+ qDebug() << m_file.fileName() << "invalid signature for Viatom data file" << sig;
+ return false;
+ break;
+ }
+
if ((year < 2015 || year > 2059) || (month < 1 || month > 12) || (day < 1 || day > 31) ||
(hour > 23) || (min > 59) || (sec > 59)) {
qDebug() << m_file.fileName() << "invalid timestamp in Viatom data file";
@@ -338,8 +354,8 @@ bool ViatomFile::ParseHeader()
CHECK_VALUE(filesize, m_file.size());
}
CHECK_VALUES(m_resolution, 2000, 4000);
- CHECK_VALUE(datasize % RECORD_SIZE, 0);
- CHECK_VALUE(m_duration % m_record_count, 0);
+// CHECK_VALUE(datasize % RECORD_SIZE, 0); CN - Commented out these 2 lines because CheckMe can record odd number of entries
+// CHECK_VALUE(m_duration % m_record_count, 0);
//qDebug().noquote() << m_file.fileName() << ts(m_timestamp) << dur(m_duration * 1000L) << ":" << m_record_count << "records @" << m_resolution << "ms";
@@ -351,21 +367,22 @@ QList ViatomFile::ReadData()
QByteArray data = m_file.readAll();
QDataStream in(data);
in.setByteOrder(QDataStream::LittleEndian);
-
+ int iCheckMeAdj; // Allows for an odd number in the CheckMe duration/# of records return
QList records;
-
// Read all Pulse, SPO2 and Motion data
do {
ViatomFile::Record rec;
in >> rec.spo2 >> rec.hr >> rec.oximetry_invalid >> rec.motion >> rec.vibration;
- CHECK_VALUES(rec.oximetry_invalid, 0, 0xFF);
- CHECK_VALUES(rec.vibration, 0, 0x80); // 0x80 when vibration is triggered
+ CHECK_VALUES(rec.oximetry_invalid, 0, 0xFF); //If it doesn't have one of these 2 values, it's bad
+ if (rec.vibration == 0x40) rec.vibration = 0x80; //0x040 (64) or 0x80 (128) when vibration is triggered
+ CHECK_VALUES(rec.vibration, 0, 0x80); // 0x80 (128) when vibration is triggered
if (rec.oximetry_invalid == 0xFF) {
CHECK_VALUE(rec.spo2, 0xFF);
- CHECK_VALUE(rec.hr, 0xFF);
+ CHECK_VALUE(rec.hr, 0xFF); // if all 3 have 0xFF, then end of data
}
records.append(rec);
- } while (!in.atEnd());
+ } while (records.size() < m_record_count); // CN Changed to allow for an incomlpete record values
+// CN } while (!in.atEnd());
// It turns out 2s files are actually just double-reported samples!
if (m_resolution == 2000) {
@@ -392,7 +409,11 @@ QList ViatomFile::ReadData()
records = dedup;
}
}
- CHECK_VALUE(duration() / records.size(), 4); // We've only seen 4s true resolution so far.
+ iCheckMeAdj = duration() / records.size();
+ if(iCheckMeAdj == 3) iCheckMeAdj = 4; // CN - Sanity check for CheckMe devices since their files do not always terminate on an even number.
+
+ CHECK_VALUE(iCheckMeAdj, 4); // Crimson Nape - Changed to accomadate the CheckMe data files.
+ // CHECK_VALUE(duration() / records.size(), 4); // We've only seen 4s true resolution so far.
return records;
}
diff --git a/oscar/SleepLib/loader_plugins/viatom_loader.h b/oscar/SleepLib/loader_plugins/viatom_loader.h
index 9e2dfc9b..7b326ac4 100644
--- a/oscar/SleepLib/loader_plugins/viatom_loader.h
+++ b/oscar/SleepLib/loader_plugins/viatom_loader.h
@@ -2,6 +2,7 @@
*
* Copyright (c) 2019-2020 The OSCAR Team
* (Initial importer written by dave madden )
+ * Modified 02/21/2021 to allow for CheckMe device data files by Crimson Nape
*
* This file is subject to the terms and conditions of the GNU General Public
* License. See the file COPYING in the main directory of the source code
@@ -38,7 +39,7 @@ class ViatomLoader : public MachineLoader
virtual MachineInfo newInfo() {
return MachineInfo(MT_OXIMETER, 0, viatom_class_name, QObject::tr("Viatom"), QString(), QString(), QString(), QObject::tr("Viatom Software"), QDateTime::currentDateTime(), viatom_data_version);
}
-
+
QStringList getNameFilter();
//Machine *CreateMachine();
diff --git a/oscar/SleepLib/loader_plugins/zeo_loader.cpp b/oscar/SleepLib/loader_plugins/zeo_loader.cpp
index e282d075..bdaf4c4d 100644
--- a/oscar/SleepLib/loader_plugins/zeo_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/zeo_loader.cpp
@@ -8,10 +8,11 @@
* for more details. */
//********************************************************************************************
-// IMPORTANT!!!
-//********************************************************************************************
-// Please INCREMENT the zeo_data_version in zel_loader.h when making changes to this loader
-// that change loader behaviour or modify channels.
+// Please only INCREMENT the zeo_data_version in zeo_loader.h when making changes
+// that change loader behaviour or modify channels in a manner that fixes old data imports.
+// Note that changing the data version will require a reimport of existing data for which OSCAR
+// does not keep a backup - so it should be avoided if possible.
+// i.e. there is no need to change the version when adding support for new devices
//********************************************************************************************
#include
diff --git a/oscar/daily.cpp b/oscar/daily.cpp
index 00e5ea61..98e2e8d4 100644
--- a/oscar/daily.cpp
+++ b/oscar/daily.cpp
@@ -509,6 +509,9 @@ Daily::Daily(QWidget *parent,gGraphView * shared)
connect(GraphView, SIGNAL(updateCurrentTime(double)), this, SLOT(on_LineCursorUpdate(double)));
connect(GraphView, SIGNAL(updateRange(double,double)), this, SLOT(on_RangeUpdate(double,double)));
connect(GraphView, SIGNAL(GraphsChanged()), this, SLOT(updateGraphCombo()));
+
+ // Watch for focusOut events on the JournalNotes widget
+ ui->JournalNotes->installEventFilter(this);
// qDebug() << "Finished making new Daily object";
// sleep(3);
}
@@ -521,9 +524,11 @@ Daily::~Daily()
disconnect(sessionbar, SIGNAL(sessionClicked(Session*)), this, SLOT(doToggleSession(Session*)));
disconnect(webView,SIGNAL(anchorClicked(QUrl)),this,SLOT(Link_clicked(QUrl)));
+ ui->JournalNotes->removeEventFilter(this);
- if (previous_date.isValid())
+ if (previous_date.isValid()) {
Unload(previous_date);
+ }
// Save graph orders and pin status, etc...
GraphView->SaveSettings("Daily");
@@ -1221,8 +1226,9 @@ QString Daily::getOximeterInformation(Day * day)
html+=" |
";
html+=""+oxi->brand()+" "+oxi->model()+" |
\n";
html+=" |
";
- html+=QString("%1: %2 (%3%) |
").arg(tr("SpO2 Desaturations")).arg(day->count(OXI_SPO2Drop)).arg((100.0/day->hours(MT_OXIMETER)) * (day->sum(OXI_SPO2Drop)/3600.0),0,'f',2);
- html+=QString("%1: %2 (%3%) |
").arg(tr("Pulse Change events")).arg(day->count(OXI_PulseChange)).arg((100.0/day->hours(MT_OXIMETER)) * (day->sum(OXI_PulseChange)/3600.0),0,'f',2);
+ // Include SpO2 and PC drops per hour of Oximetry data in case CPAP data is missing
+ html+=QString("%1: %2 (%3%) %4/h |
").arg(tr("SpO2 Desaturations")).arg(day->count(OXI_SPO2Drop)).arg((100.0/day->hours(MT_OXIMETER)) * (day->sum(OXI_SPO2Drop)/3600.0),0,'f',2).arg((day->count(OXI_SPO2Drop)/day->hours(MT_OXIMETER)),0,'f',2);
+ html+=QString("%1: %2 (%3%) %4/h |
").arg(tr("Pulse Change events")).arg(day->count(OXI_PulseChange)).arg((100.0/day->hours(MT_OXIMETER)) * (day->sum(OXI_PulseChange)/3600.0),0,'f',2).arg((day->count(OXI_PulseChange)/day->hours(MT_OXIMETER)),0,'f',2);
html+=QString("%1: %2% |
").arg(tr("SpO2 Baseline Used")).arg(day->settings_wavg(OXI_SPO2Drop),0,'f',2); // CHECKME: Should this value be wavg OXI_SPO2 isntead?
html+="\n";
html+="
\n";
@@ -1295,7 +1301,7 @@ QString Daily::getStatisticsInfo(Day * day)
.arg(STR_TR_Min)
.arg(midname)
.arg(tr("%1%2").arg(percentile*100,0,'f',0).arg(STR_UNIT_Percentage))
- .arg(STR_TR_Max);
+ .arg(ST_max == ST_MAX?STR_TR_Max:tr("99.5%"));
ChannelID chans[]={
CPAP_Pressure,CPAP_PressureSet,CPAP_EPAP,CPAP_EPAPSet,CPAP_IPAP,CPAP_IPAPSet,CPAP_PS,CPAP_PTB,
@@ -1529,6 +1535,11 @@ QVariant MyTextBrowser::loadResource(int type, const QUrl &url)
void Daily::Load(QDate date)
{
+ qDebug() << "Daily::Load called for" << date.toString() << "using" << QApplication::font().toString();
+
+ qDebug() << "Setting App font in Daily::Load";
+ setApplicationFont();
+
dateDisplay->setText(""+date.toString(Qt::SystemLocaleLongDate)+"");
previous_date=date;
@@ -1643,6 +1654,11 @@ void Daily::Load(QDate date)
if (hours>0) {
htmlLeftAHI="\n";
+
+ // Show application font, for debugging font issues
+ // QString appFont = QApplication::font().toString();
+ // htmlLeftAHI+=QString("%1 |
").arg(appFont);
+
htmlLeftAHI+="";
if (!isBrick) {
ChannelID ahichan=CPAP_AHI;
@@ -1651,10 +1667,10 @@ void Daily::Load(QDate date)
ahichan=CPAP_RDI;
ahiname=STR_TR_RDI;
}
- htmlLeftAHI+=QString("%3 %5 | \n")
+ htmlLeftAHI+=QString("%3 %5 | \n")
.arg("#F88017").arg(COLOR_Text.name()).arg(ahiname).arg(schema::channel[ahichan].fullname()).arg(ahi,0,'f',2);
} else {
- htmlLeftAHI+=QString("%2 | \n")
+ htmlLeftAHI+=QString("%2 | \n")
.arg("#F88017").arg(tr("BRICK! :("));
}
htmlLeftAHI+="
\n";
@@ -1680,14 +1696,23 @@ void Daily::Load(QDate date)
schema::Channel & chan = schema::channel[code];
// if (!chan.enabled()) continue;
QString data;
+ float channelHours = hours;
+ if (chan.machtype() != MT_CPAP) {
+ // Use machine type hours (if available) rather than CPAP hours, since
+ // might have Oximetry (for example) longer or shorter than CPAP
+ channelHours = day->hours(chan.machtype());
+ if (channelHours <= 0) {
+ channelHours = hours;
+ }
+ }
if (chan.type() == schema::SPAN) {
- val = (100.0 / hours)*(day->sum(code)/3600.0);
+ val = (100.0 / channelHours)*(day->sum(code)/3600.0);
data = QString("%1%").arg(val,0,'f',2);
} else if (code == CPAP_VSnore2) { // TODO: This should be generalized rather than special-casing a single channel here.
- val = day->sum(code) / hours;
+ val = day->sum(code) / channelHours;
data = QString("%1").arg(val,0,'f',2);
} else {
- val = day->count(code) / hours;
+ val = day->count(code) / channelHours;
data = QString("%1").arg(val,0,'f',2);
}
// TODO: percentage would be another useful option here for things like
@@ -2186,6 +2211,9 @@ void Daily::on_JournalNotesUnderline_clicked()
void Daily::on_prevDayButton_clicked()
{
+ if (previous_date.isValid()) {
+ Unload(previous_date);
+ }
if (!p_profile->ExistsAndTrue("SkipEmptyDays")) {
LoadDate(previous_date.addDays(-1));
} else {
@@ -2200,8 +2228,23 @@ void Daily::on_prevDayButton_clicked()
}
}
+bool Daily::eventFilter(QObject *object, QEvent *event)
+{
+ if (object == ui->JournalNotes && event->type() == QEvent::FocusOut) {
+ // Trigger immediate save of journal when we focus out from it so we never
+ // lose any journal entry text...
+ if (previous_date.isValid()) {
+ Unload(previous_date);
+ }
+ }
+ return false;
+}
+
void Daily::on_nextDayButton_clicked()
{
+ if (previous_date.isValid()) {
+ Unload(previous_date);
+ }
if (!p_profile->ExistsAndTrue("SkipEmptyDays")) {
LoadDate(previous_date.addDays(1));
} else {
@@ -2232,6 +2275,9 @@ void Daily::on_calButton_toggled(bool checked)
void Daily::on_todayButton_clicked()
{
+ if (previous_date.isValid()) {
+ Unload(previous_date);
+ }
// QDate d=QDate::currentDate();
// if (d > p_profile->LastDay()) {
QDate lastcpap = p_profile->LastDay(MT_CPAP);
@@ -2404,21 +2450,10 @@ void Daily::on_bookmarkTable_itemChanged(QTableWidgetItem *item)
void Daily::on_weightSpinBox_valueChanged(double arg1)
{
- // Update the BMI display
- double kg;
- if (p_profile->general->unitSystem()==US_English) {
- kg=((arg1*pound_convert) + (ui->ouncesSpinBox->value()*ounce_convert)) / 1000.0;
- } else kg=arg1;
- double height=p_profile->user->height()/100.0;
- if ((height>0) && (kg>0)) {
- double bmi=kg/(height * height);
- ui->BMI->display(bmi);
- ui->BMI->setVisible(true);
- ui->BMIlabel->setVisible(true);
- } else {
- ui->BMI->setVisible(false);
- ui->BMIlabel->setVisible(false);
- }
+ // This is called if up/down arrows are used, in which case editingFinished is
+ // never called. So always call editingFinished instead
+ Q_UNUSED(arg1);
+ this->on_weightSpinBox_editingFinished();
}
void Daily::on_weightSpinBox_editingFinished()
@@ -2437,7 +2472,25 @@ void Daily::on_weightSpinBox_editingFinished()
} else {
kg=arg1;
}
- journal->settings[Journal_Weight]=kg;
+ if (journal->settings.contains(Journal_Weight)) {
+ QVariant old = journal->settings[Journal_Weight];
+ if (old == kg && kg > 0) {
+ // No change to weight - skip
+ return;
+ }
+ } else if (kg == 0) {
+ // Still zero - skip
+ return;
+ }
+ if (kg > 0) {
+ journal->settings[Journal_Weight]=kg;
+ } else {
+ // Weight now zero - remove from journal
+ auto jit = journal->settings.find(Journal_Weight);
+ if (jit != journal->settings.end()) {
+ journal->settings.erase(jit);
+ }
+ }
gGraphView *gv=mainwin->getOverview()->graphView();
gGraph *g;
if (gv) {
@@ -2450,66 +2503,35 @@ void Daily::on_weightSpinBox_editingFinished()
ui->BMI->setVisible(true);
ui->BMIlabel->setVisible(true);
journal->settings[Journal_BMI]=bmi;
- if (gv) {
- g=gv->findGraph(STR_GRAPH_BMI);
- if (g) g->setDay(nullptr);
- }
} else {
+ // BMI now zero - remove it
+ auto jit = journal->settings.find(Journal_BMI);
+ if (jit != journal->settings.end()) {
+ journal->settings.erase(jit);
+ }
+ // And make it invisible
ui->BMI->setVisible(false);
ui->BMIlabel->setVisible(false);
}
+ if (gv) {
+ g=gv->findGraph(STR_GRAPH_BMI);
+ if (g) g->setDay(nullptr);
+ }
journal->SetChanged(true);
}
void Daily::on_ouncesSpinBox_valueChanged(int arg1)
{
- // just update for BMI display
- double height=p_profile->user->height()/100.0;
- double kg=((ui->weightSpinBox->value()*pound_convert) + (arg1*ounce_convert)) / 1000.0;
- if ((height>0) && (kg>0)) {
- double bmi=kg/(height * height);
- ui->BMI->display(bmi);
- ui->BMI->setVisible(true);
- ui->BMIlabel->setVisible(true);
- } else {
- ui->BMI->setVisible(false);
- ui->BMIlabel->setVisible(false);
- }
+ // This is called if up/down arrows are used, in which case editingFinished is
+ // never called. So always call editingFinished instead
+ Q_UNUSED(arg1);
+ this->on_weightSpinBox_editingFinished();
}
void Daily::on_ouncesSpinBox_editingFinished()
{
- double arg1=ui->ouncesSpinBox->value();
- Session *journal=GetJournalSession(previous_date);
- if (!journal) {
- journal=CreateJournalSession(previous_date);
- }
- double height=p_profile->user->height()/100.0;
- double kg=((ui->weightSpinBox->value()*pound_convert) + (arg1*ounce_convert)) / 1000.0;
- journal->settings[Journal_Weight]=kg;
-
- gGraph *g;
- if (mainwin->getOverview()) {
- g=mainwin->getOverview()->graphView()->findGraph(STR_GRAPH_Weight);
- if (g) g->setDay(nullptr);
- }
-
- if ((height>0) && (kg>0)) {
- double bmi=kg/(height * height);
- ui->BMI->display(bmi);
- ui->BMI->setVisible(true);
- ui->BMIlabel->setVisible(true);
-
- journal->settings[Journal_BMI]=bmi;
- if (mainwin->getOverview()) {
- g=mainwin->getOverview()->graphView()->findGraph(STR_GRAPH_BMI);
- if (g) g->setDay(nullptr);
- }
- } else {
- ui->BMI->setVisible(false);
- ui->BMIlabel->setVisible(false);
- }
- journal->SetChanged(true);
+ // This is functionally identical to the weightSpinBox_editingFinished, so just call that
+ this->on_weightSpinBox_editingFinished();
}
QString Daily::GetDetailsText()
diff --git a/oscar/daily.h b/oscar/daily.h
index 35e9d4cb..94aad87a 100644
--- a/oscar/daily.h
+++ b/oscar/daily.h
@@ -304,6 +304,8 @@ private:
*/
void UpdateEventsTree(QTreeWidget * tree,Day *day);
+ virtual bool eventFilter(QObject *object, QEvent *event) override;
+
void updateCube();
diff --git a/oscar/main.cpp b/oscar/main.cpp
index 44077282..29739096 100644
--- a/oscar/main.cpp
+++ b/oscar/main.cpp
@@ -564,8 +564,7 @@ int main(int argc, char *argv[]) {
if (testFile.exists())
testFile.remove();
if (!testFile.open(QFile::ReadWrite)) {
- QString errMsg = QObject::tr("Unable to write to OSCAR data directory") + " " + GetAppData() + "\n" +
- GetAppData() + "\n" +
+ QString errMsg = QObject::tr("Unable to write to OSCAR data directory") + " " + GetAppData() + "\n\n" +
QObject::tr("Error code") + ": " + QString::number(testFile.error()) + " - " + testFile.errorString() + "\n\n" +
QObject::tr("OSCAR cannot continue and is exiting.") + "\n";
qCritical() << errMsg;
@@ -592,6 +591,7 @@ int main(int argc, char *argv[]) {
AppSetting->setLanguage(language);
// Set fonts from preferences file
+ qDebug() << "App font before Prefs setting" << QApplication::font();
validateAllFonts();
setApplicationFont();
diff --git a/oscar/mainwindow.cpp b/oscar/mainwindow.cpp
index 0393f405..390fed99 100644
--- a/oscar/mainwindow.cpp
+++ b/oscar/mainwindow.cpp
@@ -2394,16 +2394,46 @@ void MainWindow::on_actionImport_Somnopose_Data_triggered()
w.setNameFilters(QStringList("Somnopause CSV File (*.csv)"));
SomnoposeLoader somno;
+ // Display progress if we have more than 1 file to load...
+ ProgressDialog progress(this);
if (w.exec() == QFileDialog::Accepted) {
- QString filename = w.selectedFiles()[0];
+ int i, skipped = 0;
+ int size = w.selectedFiles().size();
+ if (size > 1) {
+ progress.setMessage(QObject::tr("Importing Sessions..."));
+ progress.setProgressMax(size);
+ progress.setProgressValue(0);
+ progress.setWindowModality(Qt::ApplicationModal);
+ progress.open();
+ QCoreApplication::processEvents();
+ }
+ for (i=0; i < size; i++) {
+ QString filename = w.selectedFiles()[i];
- if (!somno.OpenFile(filename)) {
- Notify(tr("There was a problem opening Somnopose Data File: ") + filename);
- return;
+ int res = somno.OpenFile(filename);
+ if (!res) {
+ if (i == 0) {
+ Notify(tr("There was a problem opening Somnopose Data File: ") + filename);
+ return;
+ } else {
+ Notify(tr("Somnopause Data Import of %1 file(s) complete").arg(i) + "\n\n" +
+ tr("There was a problem opening Somnopose Data File: ") + filename,
+ tr("Somnopose Import Partial Success"));
+ break;
+ }
+ }
+ if (res < 0) {
+ // Should we report on skipped count?
+ skipped++;
+ }
+ progress.setProgressValue(i+1);
+ QCoreApplication::processEvents();
}
- Notify(tr("Somnopause Data Import complete"));
+ if (i == size) {
+ Notify(tr("Somnopause Data Import complete"));
+ }
PopulatePurgeMenu();
if (overview) overview->ReloadGraphs();
if (welcome) welcome->refreshPage();
diff --git a/oscar/mainwindow.ui b/oscar/mainwindow.ui
index 9b797bae..1b68d8c9 100644
--- a/oscar/mainwindow.ui
+++ b/oscar/mainwindow.ui
@@ -1211,8 +1211,8 @@ QToolBox::tab:selected {
0
0
- 178
- 685
+ 175
+ 690
@@ -1669,8 +1669,8 @@ border: 2px solid #56789a; border-radius: 30px;
0
0
- 178
- 685
+ 175
+ 690
@@ -2713,8 +2713,8 @@ border-radius: 10px;
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd">
<html><head><meta name="qrichtext" content="1" /><style type="text/css">
p, li { white-space: pre-wrap; }
-</style></head><body style=" font-family:'MS Shell Dlg 2'; font-size:7.84158pt; font-weight:400; font-style:normal;">
-<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8.25pt;"><br /></p></body></html>
+</style></head><body style=" font-family:'MS Shell Dlg 2'; font-size:8.25pt; font-weight:400; font-style:normal;">
+<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><br /></p></body></html>
false
@@ -2728,8 +2728,8 @@ p, li { white-space: pre-wrap; }
0
0
- 178
- 685
+ 175
+ 690
@@ -2770,8 +2770,8 @@ p, li { white-space: pre-wrap; }
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd">
<html><head><meta name="qrichtext" content="1" /><style type="text/css">
p, li { white-space: pre-wrap; }
-</style></head><body style=" font-family:'MS Shell Dlg 2'; font-size:7.84158pt; font-weight:400; font-style:normal;">
-<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8.25pt;"><br /></p></body></html>
+</style></head><body style=" font-family:'MS Shell Dlg 2'; font-size:8.25pt; font-weight:400; font-style:normal;">
+<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><br /></p></body></html>
false
@@ -2791,7 +2791,7 @@ p, li { white-space: pre-wrap; }
0
0
1023
- 23
+ 18
@@ -3157,7 +3157,7 @@ p, li { white-space: pre-wrap; }
- Import &Viatom Data
+ Import &Viatom/Wellue Data
diff --git a/oscar/overview.cpp b/oscar/overview.cpp
index de2b930b..4260b8c3 100644
--- a/oscar/overview.cpp
+++ b/oscar/overview.cpp
@@ -236,8 +236,20 @@ void Overview::CreateAllGraphs() {
} // for chit
WEIGHT = createGraph(STR_GRAPH_Weight, STR_TR_Weight, STR_TR_Weight, YT_Weight);
+ weight = new SummaryChart("Weight", GT_LINE);
+ weight->setMachineType(MT_JOURNAL);
+ weight->addSlice(Journal_Weight, QColor("black"), ST_SETAVG);
+ WEIGHT->AddLayer(weight);
BMI = createGraph(STR_GRAPH_BMI, STR_TR_BMI, tr("Body\nMass\nIndex"));
+ bmi = new SummaryChart("BMI", GT_LINE);
+ bmi->setMachineType(MT_JOURNAL);
+ bmi->addSlice(Journal_BMI, QColor("black"), ST_SETAVG);
+ BMI->AddLayer(bmi);
ZOMBIE = createGraph(STR_GRAPH_Zombie, STR_TR_Zombie, tr("How you felt\n(0-10)"));
+ zombie = new SummaryChart("Zombie", GT_LINE);
+ zombie->setMachineType(MT_JOURNAL);
+ zombie->addSlice(Journal_ZombieMeter, QColor("black"), ST_SETAVG);
+ ZOMBIE->AddLayer(zombie);
}
// Recalculates Overview chart info