From be1d3d878b42cafc85dbf0bc654d7149c02402a2 Mon Sep 17 00:00:00 2001
From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com>
Date: Thu, 4 Mar 2021 17:34:05 -0500
Subject: [PATCH 01/28] Add 1130X200 to the list of tested machines.
Also update warnings based on new test data and correctly
label the rise time lock setting on some 1030X sessions.
---
oscar/SleepLib/loader_plugins/prs1_loader.cpp | 18 ++++++++++++------
1 file changed, 12 insertions(+), 6 deletions(-)
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.
}
}
}
From 562cd9cc37af2e997533f00a4b095b75c98ac2fe Mon Sep 17 00:00:00 2001
From: LoudSnorer
Date: Sun, 21 Mar 2021 14:15:47 -0400
Subject: [PATCH 02/28] fix display issues for short span events
---
oscar/Graphs/gFlagsLine.cpp | 16 +++++++++-------
oscar/Graphs/gLineOverlay.cpp | 19 ++++++++++++++-----
2 files changed, 23 insertions(+), 12 deletions(-)
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/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) {
////////////////////////////////////////////////////////////////////////////
From 9777ecd2ab15d5de1f6aeb439ba2eba2c33e52e9 Mon Sep 17 00:00:00 2001
From: LoudSnorer
Date: Sun, 21 Mar 2021 16:21:48 -0400
Subject: [PATCH 03/28] This allows the Overview Custon range to be saved and
resurvive a reloaded of OSCAR. a New feature has been been added but can not
be activated until a UI file is changed - another submission. This new
feature will utilize the current displayed range as the custom range. Other
wise the Overview custom range uses the values from the calendar. This
changes does not require and data or translations.
---
oscar/SleepLib/profiles.h | 6 ++++
oscar/overview.cpp | 72 +++++++++++++++++++++++++--------------
oscar/overview.h | 5 +--
3 files changed, 56 insertions(+), 27 deletions(-)
diff --git a/oscar/SleepLib/profiles.h b/oscar/SleepLib/profiles.h
index 4c218082..6acf9b6a 100644
--- a/oscar/SleepLib/profiles.h
+++ b/oscar/SleepLib/profiles.h
@@ -369,6 +369,8 @@ const QString STR_US_PrefCalcMax = "PrefCalcMax";
const QString STR_US_ShowUnknownFlags = "ShowUnknownFlags";
const QString STR_US_StatReportMode = "StatReportMode";
const QString STR_US_LastOverviewRange = "LastOverviewRange";
+const QString STR_US_CustomOverviewRangeStart = "CustomOverviewRangeStart";
+const QString STR_US_CustomOverviewRangeEnd = "CustomOverviewRangeEnd";
// Values for StatReportMode
const int STAT_MODE_STANDARD = 0;
@@ -742,6 +744,8 @@ class UserSettings : public PrefSettings
int statReportMode() const { return getPref(STR_US_StatReportMode).toInt(); }
inline bool showUnknownFlags() const { return m_showUnownFlags; }
int lastOverviewRange() const { return getPref(STR_US_LastOverviewRange).toInt(); }
+ QDate customOverviewRangeStart () const { return getPref(STR_US_CustomOverviewRangeStart).toDate(); }
+ QDate customOverviewRangeEnd () const { return getPref(STR_US_CustomOverviewRangeEnd).toDate(); }
void setUnitSystem(UnitSystem us) { setPref(STR_US_UnitSystem, (int)us); }
void setEventWindowSize(double size) { setPref(STR_US_EventWindowSize, size); }
@@ -754,6 +758,8 @@ class UserSettings : public PrefSettings
void setStatReportMode(int i) { setPref(STR_US_StatReportMode, i); }
void setShowUnknownFlags(bool b) { setPref(STR_US_ShowUnknownFlags, m_showUnownFlags=b); }
void setLastOverviewRange(int i) { setPref(STR_US_LastOverviewRange, i); }
+ void setCustomOverviewRangeStart(QDate i) { setPref(STR_US_CustomOverviewRangeStart, i); }
+ void setCustomOverviewRangeEnd(QDate i) { setPref(STR_US_CustomOverviewRangeEnd, i); }
bool m_calculateRDI, m_showUnownFlags, m_skipEmptyDays;
int m_prefCalcMiddle, m_prefCalcMax;
diff --git a/oscar/overview.cpp b/oscar/overview.cpp
index de2b930b..755f4b05 100644
--- a/oscar/overview.cpp
+++ b/oscar/overview.cpp
@@ -151,11 +151,6 @@ Overview::~Overview()
disconnect(ui->dateEnd->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateEnd_currentPageChanged(int, int)));
disconnect(ui->dateStart->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateStart_currentPageChanged(int, int)));
- // Don't save custom date range. Default to last 3 months
- if (p_profile->general->lastOverviewRange() == 8) {
- p_profile->general->setLastOverviewRange(4);
- }
-
// Save graph orders and pin status, etc...
GraphView->SaveSettings("Overview");//no trans
@@ -325,9 +320,6 @@ void Overview::updateGraphCombo()
{
ui->graphCombo->clear();
gGraph *g;
- // ui->graphCombo->addItem("Show All Graphs");
- // ui->graphCombo->addItem("Hide All Graphs");
- // ui->graphCombo->addItem("---------------");
for (int i = 0; i < GraphView->size(); i++) {
g = (*GraphView)[i];
@@ -345,6 +337,7 @@ void Overview::updateGraphCombo()
updateCube();
}
+#if 0
void Overview::ResetGraphs()
{
QDate start = ui->dateStart->date();
@@ -366,6 +359,7 @@ void Overview::ResetGraph(QString name)
g->setDay(nullptr);
GraphView->redraw();
}
+#endif
void Overview::RedrawGraphs()
{
@@ -430,6 +424,9 @@ void Overview::on_dateEnd_dateChanged(const QDate &date)
qint64 d2 = qint64(QDateTime(date, QTime(23, 0, 0)/*, Qt::UTC*/).toTime_t()) * 1000L;
GraphView->SetXBounds(d1, d2);
ui->dateStart->setMaximumDate(date);
+ if (customMode) {
+ p_profile->general->setCustomOverviewRangeEnd(date);
+ }
}
void Overview::on_dateStart_dateChanged(const QDate &date)
@@ -438,6 +435,10 @@ void Overview::on_dateStart_dateChanged(const QDate &date)
qint64 d2 = qint64(QDateTime(ui->dateEnd->date(), QTime(23, 0, 0)/*, Qt::UTC*/).toTime_t()) * 1000L;
GraphView->SetXBounds(d1, d2);
ui->dateEnd->setMinimumDate(date);
+ if (customMode) {
+ p_profile->general->setCustomOverviewRangeStart(date);
+ }
+
}
// Zoom to 100% button clicked or called back from 100% zoom in popup menu
@@ -463,7 +464,6 @@ void Overview::ResetGraphOrder(int type)
// Process new range selection from combo button
void Overview::on_rangeCombo_activated(int index)
{
- p_profile->general->setLastOverviewRange(index); // type of range in last use
ui->dateStart->setMinimumDate(p_profile->FirstDay()); // first and last dates for ANY machine type
ui->dateEnd->setMaximumDate(p_profile->LastDay());
@@ -474,22 +474,6 @@ void Overview::on_rangeCombo_activated(int index)
end = max(end, p_profile->LastDay(MT_SLEEPSTAGE));
QDate start;
- if (index == 8) { // Custom
- ui->dateStartLabel->setEnabled(true);
- ui->dateEndLabel->setEnabled(true);
- ui->dateEnd->setEnabled(true);
- ui->dateStart->setEnabled(true);
-
- ui->dateStart->setMaximumDate(ui->dateEnd->date());
- ui->dateEnd->setMinimumDate(ui->dateStart->date());
- p_profile->general->setLastOverviewRange(8);
- return;
- }
-
- ui->dateEnd->setEnabled(false);
- ui->dateStart->setEnabled(false);
- ui->dateStartLabel->setEnabled(false);
- ui->dateEndLabel->setEnabled(false);
if (index == 0) {
start = end.addDays(-6);
@@ -507,10 +491,48 @@ void Overview::on_rangeCombo_activated(int index)
start = end.addYears(-1).addDays(1);
} else if (index == 7) { // Everything
start = p_profile->FirstDay();
+ } else if (index == 8 || index == 9) { // Custom
+ // Validate save Overview Custom Range for first access.
+ if (!p_profile->general->customOverviewRangeStart().isValid()
+ || (!p_profile->general->customOverviewRangeEnd().isValid() )
+ || (index==9 /* New Coustom mode - to reset custom range to displayed date range*/)
+ ) {
+ // Reset Custom Range to current range displayed
+ // on first initialization of this version of OSCAR
+ // or on new custom Mode to reset range.
+ qint64 istart,iend;
+ GraphView->GetXBounds(istart , iend);
+ start = QDateTime::fromMSecsSinceEpoch( istart ).date();
+ end = QDateTime::fromMSecsSinceEpoch( iend ).date();
+ p_profile->general->setCustomOverviewRangeStart(start);
+ p_profile->general->setCustomOverviewRangeEnd(end);
+ index=8;
+ ui->rangeCombo->setCurrentIndex(index);
+ } else if (customMode) { // last mode was custom.
+ // Reset Custom Range to current range in calendar widget
+ // Custom mode MUST be initialized to false when the Custom Instance is created.
+ start = ui->dateStart->date();
+ end = ui->dateEnd->date();
+ p_profile->general->setCustomOverviewRangeStart(start);
+ p_profile->general->setCustomOverviewRangeEnd(end);
+ } else {
+ // have a change in RangeCombo selection. Use last saved values.
+ start = p_profile->general->customOverviewRangeStart() ;
+ end = p_profile->general->customOverviewRangeEnd() ;
+ }
}
if (start < p_profile->FirstDay()) { start = p_profile->FirstDay(); }
+ customMode = (index == 8) ;
+ ui->dateStartLabel->setEnabled(customMode);
+ ui->dateEndLabel->setEnabled(customMode);
+ ui->dateEnd->setEnabled(customMode);
+ ui->dateStart->setEnabled(customMode);
+
+
+ p_profile->general->setLastOverviewRange(index); // type of range in last use
+
// Ensure that all summary files are available and update version numbers if required
int size = start.daysTo(end);
qDebug() << "Overview range combo from" << start << "to" << end << "with" << size << "days";
diff --git a/oscar/overview.h b/oscar/overview.h
index 5a434b18..5513c329 100644
--- a/oscar/overview.h
+++ b/oscar/overview.h
@@ -50,7 +50,7 @@ class Overview : public QWidget
void ResetFont();
//! \brief Recalculates Overview chart info, but keeps the date set
- void ResetGraphs();
+ //void ResetGraphs();
//! \brief Reset graphs to uniform heights
void ResetGraphLayout();
@@ -80,7 +80,7 @@ class Overview : public QWidget
//! \brief List of SummaryCharts shown on the overview page
QVector OverviewCharts;
- void ResetGraph(QString name);
+ //void ResetGraph(QString name);
void RebuildGraphs(bool reset = true);
@@ -128,6 +128,7 @@ class Overview : public QWidget
QIcon *icon_on;
QIcon *icon_off;
MyLabel *dateLabel;
+ bool customMode=false;
//! \brief Updates the calendar highlighting for the calendar object for this date.
void UpdateCalendarDay(QDateEdit *calendar, QDate date);
From 2d502f6a6804c764b90a6a8197e35d7f08bc7130 Mon Sep 17 00:00:00 2001
From: kappa44 <6469032-kappa44@users.noreply.gitlab.com>
Date: Tue, 23 Mar 2021 09:42:47 +1100
Subject: [PATCH 04/28] Revert viatom data version to avoid purge
---
oscar/SleepLib/loader_plugins/viatom_loader.h | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/oscar/SleepLib/loader_plugins/viatom_loader.h b/oscar/SleepLib/loader_plugins/viatom_loader.h
index 97697daf..7b326ac4 100644
--- a/oscar/SleepLib/loader_plugins/viatom_loader.h
+++ b/oscar/SleepLib/loader_plugins/viatom_loader.h
@@ -14,7 +14,7 @@
#include "SleepLib/machine_loader.h"
const QString viatom_class_name = "Viatom";
-const int viatom_data_version = 3; //CN increased from 2
+const int viatom_data_version = 2;
/*! \class ViatomLoader
From 00ad97ff70184d56af7f6fde360943b8957f5c49 Mon Sep 17 00:00:00 2001
From: kappa44 <6469032-kappa44@users.noreply.gitlab.com>
Date: Tue, 23 Mar 2021 09:43:14 +1100
Subject: [PATCH 05/28] Update loader version change comment
---
oscar/SleepLib/loader_plugins/cms50_loader.cpp | 9 +++++----
oscar/SleepLib/loader_plugins/cms50f37_loader.cpp | 9 +++++----
oscar/SleepLib/loader_plugins/dreem_loader.cpp | 9 +++++----
oscar/SleepLib/loader_plugins/somnopose_loader.cpp | 9 +++++----
oscar/SleepLib/loader_plugins/viatom_loader.cpp | 9 +++++----
oscar/SleepLib/loader_plugins/zeo_loader.cpp | 9 +++++----
6 files changed, 30 insertions(+), 24 deletions(-)
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/somnopose_loader.cpp b/oscar/SleepLib/loader_plugins/somnopose_loader.cpp
index 605a3a89..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
diff --git a/oscar/SleepLib/loader_plugins/viatom_loader.cpp b/oscar/SleepLib/loader_plugins/viatom_loader.cpp
index 7c2f3e44..19147c4e 100644
--- a/oscar/SleepLib/loader_plugins/viatom_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/viatom_loader.cpp
@@ -9,10 +9,11 @@
* 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
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
From 37483de62a432d2183de85d4c3822c9cf20ca3cc Mon Sep 17 00:00:00 2001
From: kappa44 <6469032-kappa44@users.noreply.gitlab.com>
Date: Sun, 28 Mar 2021 14:15:10 +1100
Subject: [PATCH 06/28] Display Weight/BMI/Zombie graphs
---
oscar/overview.cpp | 12 ++++++++++++
1 file changed, 12 insertions(+)
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
From 2331bbba267cc795f4d08af042456f77ed905edb Mon Sep 17 00:00:00 2001
From: kappa44 <6469032-kappa44@users.noreply.gitlab.com>
Date: Sun, 28 Mar 2021 14:20:19 +1100
Subject: [PATCH 07/28] Don't ResetBounds when setDay is called to avoid
changing overview graphs to full date range on BMI/Weight change
---
oscar/Graphs/gGraph.cpp | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
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)
From a0b9488aa96d5e63bc611afe6332eb293f597d76 Mon Sep 17 00:00:00 2001
From: kappa44 <6469032-kappa44@users.noreply.gitlab.com>
Date: Sun, 28 Mar 2021 14:22:31 +1100
Subject: [PATCH 08/28] Save notes immediately when focus leaves notes widget.
Fix saving of weight changes when using up/down arrows
---
oscar/daily.cpp | 130 ++++++++++++++++++++++++------------------------
1 file changed, 66 insertions(+), 64 deletions(-)
diff --git a/oscar/daily.cpp b/oscar/daily.cpp
index 84b8a86f..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");
@@ -2206,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 {
@@ -2220,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 {
@@ -2252,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);
@@ -2424,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()
@@ -2457,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) {
@@ -2470,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()
From 00225103fd4313096b7eeb2653a21752d62c0c85 Mon Sep 17 00:00:00 2001
From: kappa44 <6469032-kappa44@users.noreply.gitlab.com>
Date: Sun, 28 Mar 2021 14:22:57 +1100
Subject: [PATCH 09/28] Save notes immediately when focus leaves notes widget.
---
oscar/daily.h | 2 ++
1 file changed, 2 insertions(+)
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();
From 3362fa4a4d1cf858c9dab91d04ad1560b9900391 Mon Sep 17 00:00:00 2001
From: Arie Klerk
Date: Mon, 29 Mar 2021 20:54:36 +0200
Subject: [PATCH 10/28] We have a new translator for BULGARIAN! Here's his
first update.
---
Translations/Bulgarian.bg.ts | 43 ++++++++++++++++++------------------
1 file changed, 22 insertions(+), 21 deletions(-)
diff --git a/Translations/Bulgarian.bg.ts b/Translations/Bulgarian.bg.ts
index 2e3b91be..d777a896 100644
--- a/Translations/Bulgarian.bg.ts
+++ b/Translations/Bulgarian.bg.ts
@@ -11,7 +11,7 @@
&About
- &Относно
+ За &приложение
@@ -22,12 +22,13 @@
Credits
-
+ ЗаслугиGPL License
-
+ As a whole this actually should read "Общ публичен лиценз на ГНУ", but that is a bit long on a tab. I think it would be acceptable to just say "license GPL" and then the explanation is written in the tab. The Bulgarian wikipedia page for the GPL gives a couple of translation options.
+ лиценз GPL
@@ -37,27 +38,27 @@
Show data folder
-
+ Покажи папката на даннитеAbout OSCAR %1
-
+ За предложението OSCAR %1Sorry, could not locate About file.
-
+ За съжаление, файлът За приложение не се намери.Sorry, could not locate Credits file.
-
+ За съжаление, файлът Заслуги не се намери.Sorry, could not locate Release Notes.
-
+ За съжаление, Бележки по изданието не се намери.
@@ -72,12 +73,12 @@
As this is a pre-release version, it is recommended that you <b>back up your data folder manually</b> before proceeding, because attempting to roll back later may break things.
-
+ Тъй като това е предварително издание, препоръчано е <b>да направите ръчно архивиране на своята папка с данни</b> преди да продължите, защото е възможно при по-късен опит за връщане назад нещата да се повредят.To see if the license text is available in your language, see %1.
-
+ За да проверите дали съществува превода на лиценз на Вашия език, вижте %1.
@@ -85,12 +86,12 @@
Could not find the oximeter file:
-
+ Файлът на оксиметър не се намери:Could not open the oximeter file:
-
+ Не може да се отвори файлът на оксиметър:
@@ -108,12 +109,12 @@
Could not find the oximeter file:
-
+ Файлът на оксиметър не се намери:Could not open the oximeter file:
-
+ Не може да се отвори файлът на оксиметър:
@@ -121,7 +122,7 @@
Checking for newer OSCAR versions
-
+ Проверяваме за за нова версия на OSCAR
@@ -191,12 +192,12 @@
I'm feeling ...
-
+ Чувствам се ...If height is greater than zero in Preferences Dialog, setting weight here will show Body Mass Index (BMI) value
-
+ Ако в настройки ръст е над нула, задаване на тегло тук ще се покажи стойността на индекса на телесната маса (ИТМ)
@@ -211,7 +212,7 @@
Show/hide available graphs.
-
+ Покажи или скрий достъпни графики.
@@ -361,7 +362,7 @@
Unable to display Pie Chart on this system
-
+
@@ -391,7 +392,7 @@
Sorry, this machine only provides compliance data.
-
+ За съжаление, тази машина предоставя само данни за съответствие.
@@ -441,7 +442,7 @@
<b>Please Note:</b> All settings shown below are based on assumptions that nothing has changed since previous days.
-
+ <b>Моля, Забележете:</b>Всичките настройки, които са показани надолу, се основават на предположения, че нищо не се е променило от предишните дни.
From 85b4013b2d681fcccf1ae4aec0761e39d57918f4 Mon Sep 17 00:00:00 2001
From: kappa44 <6469032-kappa44@users.noreply.gitlab.com>
Date: Thu, 1 Apr 2021 11:32:58 +1100
Subject: [PATCH 11/28] Fix stage & position session toggle
---
oscar/daily.cpp | 23 +++++++++++++++++++----
1 file changed, 19 insertions(+), 4 deletions(-)
diff --git a/oscar/daily.cpp b/oscar/daily.cpp
index 98e2e8d4..8511a362 100644
--- a/oscar/daily.cpp
+++ b/oscar/daily.cpp
@@ -576,6 +576,7 @@ void Daily::Link_clicked(const QUrl &url)
// webView->page()->mainFrame()->setScrollBarValue(Qt::Vertical, webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-i);
} else if (code=="toggleoxisession") { // Enable/Disable Oximetry session
day=p_profile->GetDay(previous_date,MT_OXIMETER);
+ if (!day) return;
Session *sess=day->find(sid, MT_OXIMETER);
if (!sess)
return;
@@ -585,6 +586,20 @@ void Daily::Link_clicked(const QUrl &url)
// Reload day
LoadDate(previous_date);
// webView->page()->mainFrame()->setScrollBarValue(Qt::Vertical, webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-i);
+ } else if (code=="togglestagesession") { // Enable/Disable Sleep Stage session
+ day=p_profile->GetDay(previous_date,MT_SLEEPSTAGE);
+ if (!day) return;
+ Session *sess=day->find(sid, MT_SLEEPSTAGE);
+ if (!sess) return;
+ sess->setEnabled(!sess->enabled());
+ LoadDate(previous_date);
+ } else if (code=="togglepositionsession") { // Enable/Disable Position session
+ day=p_profile->GetDay(previous_date,MT_POSITION);
+ if (!day) return;
+ Session *sess=day->find(sid, MT_POSITION);
+ if (!sess) return;
+ sess->setEnabled(!sess->enabled());
+ LoadDate(previous_date);
} else if (code=="cpap") {
day=p_profile->GetDay(previous_date,MT_CPAP);
if (day) {
@@ -1010,7 +1025,7 @@ QString Daily::getSessionInformation(Day * day)
case MT_SLEEPSTAGE: type="stage";
html+=tr("Sleep Stage Sessions");
break;
- case MT_POSITION: type="stage";
+ case MT_POSITION: type="position";
html+=tr("Position Sensor Sessions");
break;
@@ -1535,10 +1550,10 @@ 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() << "Daily::Load called for" << date.toString() << "using" << QApplication::font().toString();
- qDebug() << "Setting App font in Daily::Load";
- setApplicationFont();
+ qDebug() << "Setting App font in Daily::Load";
+ setApplicationFont();
dateDisplay->setText(""+date.toString(Qt::SystemLocaleLongDate)+"");
previous_date=date;
From e358d31f269dd01bf63a420267e9d9a23f387ac1 Mon Sep 17 00:00:00 2001
From: Guy Scharf
Date: Thu, 1 Apr 2021 21:39:06 -0700
Subject: [PATCH 12/28] Add logic to read rolling files created by DeVilbiss
BLUE CPAPs
---
.../loader_plugins/intellipap_loader.cpp | 469 +++++++++++++-----
1 file changed, 332 insertions(+), 137 deletions(-)
diff --git a/oscar/SleepLib/loader_plugins/intellipap_loader.cpp b/oscar/SleepLib/loader_plugins/intellipap_loader.cpp
index 7c6cedb8..e4a98e87 100644
--- a/oscar/SleepLib/loader_plugins/intellipap_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/intellipap_loader.cpp
@@ -607,38 +607,6 @@ int IntellipapLoader::OpenDV5(const QString & path)
// 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;
@@ -658,9 +626,9 @@ 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
+ unsigned int start_time; //01 Start time for date
+ unsigned int stop_time; //05 End time
+ unsigned int written; //09 timestamp when this record was written
EventDataType hours; //13
// EventDataType unknown14; //14
EventDataType pressureAvg; //15
@@ -796,13 +764,14 @@ PACK (struct SET_BIN_REC {
// 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 filetype; // 1 e.g. "R" for a R.BIN file
unsigned char serial[11]; // 2 serial number
- unsigned char numRecords[4]; // 13 Number of records in file (always 180,000)
+ unsigned char numRecords[4]; // 13 Number of records in file (always fixed, 180,000 for R.BIN)
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 unknown_43[8]; // 43 Seems always to be zero
+ unsigned char lasttime[4]; // 51 OSCAR only: Last timestamp, in history files only
unsigned char checksum; // 55 Checksum
});
@@ -902,6 +871,20 @@ struct DV6_SessionInfo {
CPAPMode mode = MODE_UNKNOWN;
};
+QString card_path;
+QString backup_path;
+QString history_path;
+
+MachineInfo info;
+Machine * mach = nullptr;
+
+bool rebuild_from_backups = false;
+bool create_backups = false;
+
+QMap DailySummaries;
+QMap SessionData;
+SET_BIN_REC * settings;
+
unsigned int ep = 0;
// Convert a 4-character number in DV6 data file to a standard int
@@ -918,41 +901,199 @@ unsigned int convertTime (unsigned char time[]) {
return ((time[3] << 24) + (time[2] << 16) + (time[1] << 8) + time[0]) + ep; // Time as Unix epoch time
}
-bool RollingFile::open(QString fn) {
+class RollingBackup
+{
+public:
+ RollingBackup () {}
+ ~RollingBackup () {
+ }
- filename = fn;
- file.setFileName(filename);
+ bool open (const QString filetype, DV6_HEADER * newhdr); // Open the file
+ bool close(); // close the file
+ bool save(QByteArray dataBA); // save the next record in the file
+
+private:
+ DV6_HEADER hdr; // file header
+ QString filetype;
+ QFile hFile;
+
+ int record_length; // Length of record block in incoming file
+ const int maxHistFileSize = 20*10e6; // Maximum size of file before we create a new file
+
+ int numWritten; // Number of records written
+ quint32 lastTimestamp;
+};
+
+bool RollingBackup::open (const QString filetype, DV6_HEADER * newhdr) {
+ if (!create_backups)
+ return true;
+
+ this->filetype = filetype;
+
+ QDir hpath(history_path);
+ QStringList filters;
+
+ numWritten = 0;
+
+ filters.append(filetype);
+ filters[0].insert(1, "_*");
+ hpath.setNameFilters(filters);
+ hpath.setFilter(QDir::Files);
+ hpath.setSorting(QDir::Name | QDir::Reversed);
+
+ QStringList fileNames = hpath.entryList(); // Get list of files
+ QFile histfile(fileNames.first());
+
+// bool needNewFile = false;
+
+ // Handle first time a history file is being created
+ if (fileNames.isEmpty()) {
+ memcpy (&hdr, newhdr, sizeof(DV6_HEADER));
+ for (int i = 0; i < 4; i++) {
+ hdr.recordStart[i] = 0;
+ hdr.lasttime[i] = 0;
+ }
+ record_length = hdr.recordLength;
+ }
+
+ // We have an existing history record
+ if (! fileNames.isEmpty()) {
+ // See if this file is large enough that we want to create a new file
+ if (histfile.size() > maxHistFileSize) {
+ memcpy (&hdr, newhdr, sizeof(DV6_HEADER));
+ for (int i = 0; i < 4; i++)
+ hdr.recordStart[i] = 0;
+
+ if (!histfile.open(QIODevice::ReadOnly)) {
+ qWarning() << "DV6 RollingBackup could not open" << fileNames.first() << "for reading, error code" << histfile.error() << histfile.errorString();
+ return false;
+ }
+ record_length = hdr.recordLength;
+
+#ifdef ROLLBACKUP
+ wrap_record = convertNum(hdr.recordStart);
+ if (!histfile.seek(sizeof(DV6_HEADER) + (wrap_record-1) * record_length)) {
+ qWarning() << "DV6 RollingBackup unable to make initial seek to record" << wrap_record << "in" + filename << file.error() << file.errorString();
+ file.close();
+ return false;
+ }
+#endif
+
+ }
+ }
+
+ return true;
+}
+
+bool RollingBackup::close() {
+ if (!create_backups)
+ return true;
+ return true;
+}
+
+bool RollingBackup::save(QByteArray dataBA) {
+ Q_UNUSED(dataBA)
+ if (!create_backups)
+ return true;
+ return true;
+}
+
+class RollingFile
+{
+public:
+ RollingFile () { }
+
+ ~RollingFile () {
+ if (data)
+ delete [] data;
+ data = nullptr;
+ if (hdr)
+ delete hdr;
+ hdr = 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
+
+ RollingBackup rb;
+
+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;
+
+ DV6_HEADER * hdr; // file header
+
+ unsigned char * data = nullptr; // record pointer
+};
+
+bool RollingFile::open(QString filetype) {
+
+ filename = filetype;
+ file.setFileName(card_path + "/" +filetype);
if (!file.open(QIODevice::ReadOnly)) {
qWarning() << "DV6 RollingFile could not open" << filename << "for reading, error code" << file.error() << file.errorString();
return false;
}
+ // Save header for use in making backups of data
+ hdr = new DV6_HEADER;
QByteArray dataBA = file.read(sizeof(DV6_HEADER));
- DV6_HEADER * hdr = (DV6_HEADER *) dataBA.data();
+ memcpy (hdr, dataBA.data(), sizeof(DV6_HEADER));
+
+ // Extract control information from header
record_length = hdr->recordLength;
wrap_record = convertNum(hdr->recordStart);
record_number = wrap_record;
number_read = 0;
wrapping = false;
+ // Create buffer to hold each record as it is read
data = new unsigned char[record_length];
+ // Seek to first data record in file
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;
}
+#ifdef ROLLBACKUP
+ if (!rb.open(filetype, hdr)) {
+ qWarning() << "DV6 RollingBackup failed";
+ file.close();
+ return false;
+ }
+#endif
- qDebug() << "RollingFile opening" << filename << "at wrap record" << wrap_record;
+ qDebug() << "DV6 RollingFile opening" << filename << "at wrap record" << wrap_record;
return true;
}
bool RollingFile::close() {
file.close();
- if (data != nullptr)
+
+#ifdef ROLLBACKUP
+ rb.close();
+#endif
+
+ if (data)
delete [] data;
data = nullptr;
+ if (hdr)
+ delete hdr;
+ hdr = nullptr;
+
return true;
}
@@ -987,6 +1128,11 @@ unsigned char * RollingFile::get() {
file.close();
return nullptr;
}
+#ifdef ROLLBACKUP
+ if (!rb.save(dataBA)) {
+ qWarning() << "DV6 RollingBackup failed";
+ }
+#endif
number_read++;
@@ -995,21 +1141,51 @@ unsigned char * RollingFile::get() {
return data;
}
-MachineInfo info;
-Machine * mach = nullptr;
+// Returns empty QByteArray() on failure.
+QByteArray fileChecksum(const QString &fileName,
+ QCryptographicHash::Algorithm hashAlgorithm)
+{
+ QFile f(fileName);
+ if (f.open(QFile::ReadOnly)) {
+ QCryptographicHash hash(hashAlgorithm);
+ bool res = hash.addData(&f);
+ f.close();
+ if (res) {
+ return hash.result();
+ }
+ }
+ return QByteArray();
+}
-bool rebuild_from_backups = false;
+/***
+// 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();
+}
+***/
-QMap DailySummaries;
-QMap SessionData;
-SET_BIN_REC * settings;
+// Return date used within OSCAR, assuming day ends at split time in preferences (usually noon)
+QDate getNominalDate (QDateTime dt) {
+ QDate d = dt.date();
+ QTime tm = dt.time();
+ QTime daySplitTime = p_profile->session->getPref(STR_IS_DaySplitTime).toTime();
+ if (tm < daySplitTime)
+ d = d.addDays(-1);
+ return d;
+}
+QDate getNominalDate (unsigned int dt) {
+ QDateTime xdt = QDateTime::fromSecsSinceEpoch(dt);
+ return getNominalDate(xdt);
+}
///////////////////////////////////////////////
// U.BIN - Open and parse session list and create session data structures
// with session start and stop times.
///////////////////////////////////////////////
-bool load6Sessions (const QString & path) {
+bool load6Sessions () {
RollingFile rf;
unsigned int ts1,ts2;
@@ -1018,7 +1194,7 @@ bool load6Sessions (const QString & path) {
qDebug() << "Parsing U.BIN";
- if (!rf.open(path+"/U.BIN")) {
+ if (!rf.open("U.BIN")) {
qWarning() << "Unable to open U.BIN";
return false;
}
@@ -1079,12 +1255,12 @@ bool load6Settings (const QString & path) {
// S.BIN - Open and load day summary list
////////////////////////////////////////////////////////////////////////////////////////
-bool load6DailySummaries (const QString & path) {
+bool load6DailySummaries () {
RollingFile rf;
DailySummaries.clear();
- if (!rf.open(path+"/S.BIN")) {
+ if (!rf.open("S.BIN")) {
qWarning() << "Unable to open S.BIN";
return false;
}
@@ -1100,7 +1276,13 @@ bool load6DailySummaries (const QString & path) {
dailyData.start_time = convertTime(rec->begin);
dailyData.stop_time = convertTime(rec->end);
- dailyData.atpressure_time = convertTime(rec->written);
+ dailyData.written = convertTime(rec->written);
+
+#ifdef DEBUG6
+ qDebug() << "DV6 S.BIN start" << dailyData.start_time
+ << "stop" << dailyData.stop_time
+ << "at pressure?" << dailyData.atpressure_time;
+#endif
dailyData.hours = float(rec->hours) / 10.0F;
dailyData.pressureSetMin = float(rec->pressureSetMin) / 10.0F;
@@ -1135,6 +1317,26 @@ bool load6DailySummaries (const QString & path) {
DailySummaries[dailyData.start_time] = dailyData;
+/**** Previous loader did this:
+ if (!mach->sessionlist.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);
+
+ 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;
+***/
+
} while (true);
rf.close();
@@ -1293,14 +1495,14 @@ int create6Sessions() {
// Parse R.BIN for high resolution flow data
////////////////////////////////////////////////////////////////////////////////////////
-bool load6HighResData (const QString & path) {
+bool load6HighResData () {
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")) {
+ if (!rf.open("R.BIN")) {
qWarning() << "DV6 Unable to open R.BIN";
return false;
}
@@ -1806,14 +2008,14 @@ bool load6HighResData (const QString & path) {
// Parse L.BIN for per minute data
////////////////////////////////////////////////////////////////////////////////////////
-bool load6PerMinute (const QString & path) {
+bool load6PerMinute () {
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")) {
+ if (!rf.open("L.BIN")) {
qWarning() << "DV6 Unable to open L.BIN";
return false;
}
@@ -1960,7 +2162,7 @@ bool load6PerMinute (const QString & path) {
// Parse E.BIN for event data
////////////////////////////////////////////////////////////////////////////////////////
-bool load6EventData (const QString & path) {
+bool load6EventData () {
RollingFile rf;
Session *sess = nullptr;
@@ -1977,7 +2179,7 @@ bool load6EventData (const QString & path) {
EventList * SN = nullptr;
EventList * FL = nullptr;
- if (!rf.open(path+"/E.BIN")) {
+ if (!rf.open("E.BIN")) {
qWarning() << "DV6 Unable to open E.BIN";
return false;
}
@@ -2170,63 +2372,20 @@ int addSessions() {
}
-// 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
+// Create dated backup of settings file if changed
////////////////////////////////////////////////////////////////////////////////////////
bool backup6 (const QString & path) {
- // Are backups enabled?
- if (!p_profile->session->backupCardData())
+ if (rebuild_from_backups || !create_backups)
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 cpath(card_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;
@@ -2249,67 +2408,103 @@ bool backup6 (const QString & path) {
bool backup_settings = true;
QStringList filters;
- filters << "set_*.bin";
+
+ QFile settingsFile;
+ QString inputFile = cpath.absolutePath() + "/SET.BIN";
+ settingsFile.setFileName(inputFile);
+
+ filters << "SET_*.BIN";
hpath.setNameFilters(filters);
hpath.setFilter(QDir::Files);
- QDir::Name | QDir::Reversed;
+ hpath.setSorting(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);
+ qDebug() << "last settings file is" << lastFile << "new file is" << settingsFile;
+ QByteArray newMD5 = fileChecksum(settingsFile.fileName(), QCryptographicHash::Md5);
+ QByteArray oldMD5 = fileChecksum(hpath.absolutePath()+"/"+lastFile, QCryptographicHash::Md5);
if (newMD5 == oldMD5)
backup_settings = false;
}
- if (backup_settings) {
- QString newFile = hpath.absolutePath() + "/set-" + "1234" + ".bin";
- qDebug() << "history filename is" << newFile;
+ if (backup_settings && !DailySummaries.isEmpty()) {
+ DV6_S_Data ds = DailySummaries.last();
+ QString newFile = hpath.absolutePath() + "/SET_" + getNominalDate(ds.start_time).toString("yyyyMMdd") + ".BIN";
+ if (!settingsFile.copy(inputFile, newFile)) {
+ qWarning() << "DV6 backup could not copy" << inputFile << "to" << newFile << ", error code" << settingsFile.error() << settingsFile.errorString();
+ }
}
// We're done!
return true;
}
+////////////////////////////////////////////////////////////////////////////////////////
+// Initialize DV6 environment
+////////////////////////////////////////////////////////////////////////////////////////
+
+bool init6Environment (const QString & path) {
+
+ // Create Machine database record if it doesn't exist already
+ mach = p_profile->CreateMachine(info);
+ if (mach == nullptr) {
+ qWarning() << "Could not create DV6 Machine data structure";
+ return false;
+ }
+
+ backup_path = mach->getBackupPath();
+ history_path = backup_path + "/HISTORY";
+
+ // Compare QDirs rather than QStrings because separators may be different, especially on Windows.
+ QDir ipath(path);
+ QDir bpath(backup_path);
+
+ if (ipath == bpath) {
+ // Don't create backups if importing from backup folder
+ rebuild_from_backups = true;
+ create_backups = false;
+ } else {
+ rebuild_from_backups = false;
+ create_backups = p_profile->session->backupCardData();
+ }
+
+ return true;
+}
+
////////////////////////////////////////////////////////////////////////////////////////
// Open a DV6 SD card, parse everything, add to OSCAR database
////////////////////////////////////////////////////////////////////////////////////////
int IntellipapLoader::OpenDV6(const QString & path)
{
- QString newpath = path + DV6_DIR;
+ card_path = path + DV6_DIR;
- // Prime the machine database's info field with stuff relevant to this machine
+ // 1. Prime the machine database's info field with this machine
info = newInfo();
- // VER.BIN - Parse model number, serial, etc.
- if (!load6VersionInfo(newpath))
+ // 2. VER.BIN - Parse model number, serial, etc. into info structure
+ if (!load6VersionInfo(card_path))
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;
- }
-
- // SET.BIN - Parse settings file (which is only the latest settings)
- if (!load6Settings(newpath))
+ // 3. Initialize rest of the DV6 loader environment
+ if (!init6Environment (path))
return -1;
- // S.BIN - Open and parse day summary list and create a list of days
- if (!load6DailySummaries(newpath))
+ // 4. SET.BIN - Parse settings file (which is only the latest settings)
+ if (!load6Settings(card_path))
return -1;
- // Back up data files (must do after parsing VER.BIN, S.BIN, and creating Machine)
+ // 5. S.BIN - Open and parse day summary list and create a list of days
+ if (!load6DailySummaries())
+ return -1;
+
+ // 6. Back up data files (must do after parsing VER.BIN, S.BIN, and creating Machine)
if (!backup6(path))
return -1;
- // U.BIN - Open and parse session list and create a list of session times
+ // 7. U.BIN - Open and parse session list and create a list of session times
// (S.BIN must already be loaded)
- if (!load6Sessions(newpath))
+ if (!load6Sessions())
return -1;
// Create OSCAR session list from session times and summary data
@@ -2317,15 +2512,15 @@ int IntellipapLoader::OpenDV6(const QString & path)
return -1;
// R.BIN - Open and parse flow data
- if (!load6HighResData(newpath))
+ if (!load6HighResData())
return -1;
// L.BIN - Open and parse per minute data
- if (!load6PerMinute(newpath))
+ if (!load6PerMinute())
return -1;
// E.BIN - Open and parse event data
- if (!load6EventData(newpath))
+ if (!load6EventData())
return -1;
// Finalize input
From e699ea182d435794ca1633102b3c7f71c2b1ef50 Mon Sep 17 00:00:00 2001
From: Guy Scharf
Date: Fri, 2 Apr 2021 17:17:50 -0700
Subject: [PATCH 13/28] Update version checking to allow QT 6
---
OSCAR_QT.pro | 14 ++++++++++----
oscar/oscar.pro | 16 +++++++++++-----
2 files changed, 21 insertions(+), 9 deletions(-)
diff --git a/OSCAR_QT.pro b/OSCAR_QT.pro
index df4e72bb..a9aa515c 100644
--- a/OSCAR_QT.pro
+++ b/OSCAR_QT.pro
@@ -1,7 +1,13 @@
-lessThan(QT_MAJOR_VERSION,5)|lessThan(QT_MINOR_VERSION,9) {
- message("You need to Qt 5.9 or newer to build OSCAR with Help Pages")
- lessThan(QT_MAJOR_VERSION,5)|lessThan(QT_MINOR_VERSION,7) {
- error("You need Qt 5.7 or newer to build OSCAR")
+lessThan(QT_MAJOR_VERSION,5) {
+ error("You need Qt 5.7 or newer to build OSCAR");
+}
+
+if (equals(QT_MAJOR_VERSION,5)) {
+ lessThan(QT_MINOR_VERSION,9) {
+ message("You need Qt 5.9 to build OSCAR with Help Pages")
+ }
+ lessThan(QT_MINOR_VERSION,7) {
+ error("You need Qt 5.7 or newer to build OSCAR");
}
}
diff --git a/oscar/oscar.pro b/oscar/oscar.pro
index eb704b56..b2f8e977 100644
--- a/oscar/oscar.pro
+++ b/oscar/oscar.pro
@@ -6,14 +6,20 @@
message(Platform is $$QMAKESPEC )
-lessThan(QT_MAJOR_VERSION,5)|lessThan(QT_MINOR_VERSION,9) {
- message("You need Qt 5.9 to build OSCAR with Help Pages")
- DEFINES += helpless
-}
-lessThan(QT_MAJOR_VERSION,5)|lessThan(QT_MINOR_VERSION,7) {
+lessThan(QT_MAJOR_VERSION,5) {
error("You need Qt 5.7 or newer to build OSCAR");
}
+if (equals(QT_MAJOR_VERSION,5)) {
+ lessThan(QT_MINOR_VERSION,9) {
+ message("You need Qt 5.9 to build OSCAR with Help Pages")
+ DEFINES += helpless
+ }
+ lessThan(QT_MINOR_VERSION,7) {
+ error("You need Qt 5.7 or newer to build OSCAR");
+ }
+}
+
# get rid of the help browser, at least for now
DEFINES += helpless
From e3a87a31481d36e3a7365ddbb6bc9c2fe9c46d6c Mon Sep 17 00:00:00 2001
From: Guy Scharf
Date: Sun, 4 Apr 2021 04:56:06 -0700
Subject: [PATCH 14/28] Update release notes with recent changes.
---
Htmldocs/release_notes.html | 10 +++++++++-
1 file changed, 9 insertions(+), 1 deletion(-)
diff --git a/Htmldocs/release_notes.html b/Htmldocs/release_notes.html
index e2dd1c9b..304a8ca6 100644
--- a/Htmldocs/release_notes.html
+++ b/Htmldocs/release_notes.html
@@ -12,16 +12,20 @@
http://www.apneaboard.com/wiki/index.php/OSCAR_Release_Notes