diff --git a/oscar/Graphs/gGraphView.cpp b/oscar/Graphs/gGraphView.cpp index 2b3d94ab..301b8778 100644 --- a/oscar/Graphs/gGraphView.cpp +++ b/oscar/Graphs/gGraphView.cpp @@ -3513,13 +3513,20 @@ void gGraphView::SaveDefaultSettings() { m_default_graphs = m_graphs; } -const quint32 gvmagic = 0x41756728; +const quint32 gvmagic = 0x41756728; //'Aug(' const quint16 gvversion = 4; -void gGraphView::SaveSettings(QString title) +QString gGraphView::settingsFilename (QString title,QString folderName, QString ext) { + if (folderName.size()==0) { + folderName = p_profile->Get("{DataFolder}/"); + } + return folderName+title.toLower()+ext; +} + +void gGraphView::SaveSettings(QString title,QString folderName) { qDebug() << "Saving" << title << "settings"; - QString filename = p_profile->Get("{DataFolder}/") + title.toLower() + ".shg"; + QString filename=settingsFilename(title,folderName) ; QFile f(filename); f.open(QFile::WriteOnly); QDataStream out(&f); @@ -3571,9 +3578,10 @@ template inline void hashMerge(T & a, const T & b) } -bool gGraphView::LoadSettings(QString title) +bool gGraphView::LoadSettings(QString title,QString folderName) { - QString filename = p_profile->Get("{DataFolder}/") + title.toLower() + ".shg"; + //qDebug() << "Loading" << title << "settings"; + QString filename=settingsFilename (title,folderName) ; QFile f(filename); if (!f.exists()) { diff --git a/oscar/Graphs/gGraphView.h b/oscar/Graphs/gGraphView.h index 1e1bfeea..e0a6f12c 100644 --- a/oscar/Graphs/gGraphView.h +++ b/oscar/Graphs/gGraphView.h @@ -396,11 +396,13 @@ class gGraphView //! \brief Supplies time range to all graph objects in linked group, refreshing if requested void SetXBounds(qint64 minx, qint64 maxx, short group = 0, bool refresh = true); + QString settingsFilename (QString title,QString folderName="" ,QString ext=".shg"); + //! \brief Saves the current graph order, heights, min & Max Y values to disk - void SaveSettings(QString title); + void SaveSettings(QString title,QString folderName=""); //! \brief Loads the current graph order, heights, min & max Y values from disk - bool LoadSettings(QString title); + bool LoadSettings(QString title,QString folderName=""); //! \brief Saves the current (initial) graph order, heights, min & Max Y values for future recovery void SaveDefaultSettings(); diff --git a/oscar/Graphs/gLineChart.cpp b/oscar/Graphs/gLineChart.cpp index be4bfd6e..4755cd33 100644 --- a/oscar/Graphs/gLineChart.cpp +++ b/oscar/Graphs/gLineChart.cpp @@ -1086,8 +1086,12 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion) double sum = 0; int cnt = 0; - // Draw the linechart overlays - if (m_day && (AppSetting->lineCursorMode() || (m_codes[0]==CPAP_FlowRate))) { + //Draw the linechart overlays (Event flags) independant of line Cursor mode + //The problem was that turning lineCUrsor mode off (or Control L) also stopped flag event on most daily graphs. + // The user didn't know what trigger the problem. Best quess is that Control L was typed by mistable. + // this fix allows flag events to be normally displayed when the line Cursor mode is off. + //was if (m_day /*&& (AppSetting->lineCursorMode() || (m_codes[0]==CPAP_FlowRate))*/) + if (m_day) { bool blockhover = false; for (auto fit=flags.begin(), end=flags.end(); fit != end; ++fit) { code = fit.key(); diff --git a/oscar/Resources.qrc b/oscar/Resources.qrc index 95050bd4..9b3ab92a 100644 --- a/oscar/Resources.qrc +++ b/oscar/Resources.qrc @@ -58,5 +58,14 @@ icons/fp_icon.png icons/up-down.png icons/warning.png + icons/exit.png + icons/plus.png + icons/rename.png + icons/restore.png + icons/trash_can.png + icons/update.png + icons/cog.png + icons/question_mark.png + icons/return.png diff --git a/oscar/SleepLib/loader_plugins/viatom_loader.cpp b/oscar/SleepLib/loader_plugins/viatom_loader.cpp index 9d0fb2fa..9dbb6b66 100644 --- a/oscar/SleepLib/loader_plugins/viatom_loader.cpp +++ b/oscar/SleepLib/loader_plugins/viatom_loader.cpp @@ -239,8 +239,8 @@ QStringList ViatomLoader::getNameFilter() // Sometimes the files have a SleepU_ or O2Ring_ prefix. // Sometimes they have punctuation in the timestamp. // Note that ":" is not allowed on macOS, so Mac users will need to rename their files in order to select and import them. - return QStringList({"*20[0-5][0-9][01][0-9][0-3][0-9][012][0-9][0-5][0-9][0-5][0-9]", - "*20[0-5][0-9]-[01][0-9]-[0-3][0-9] [012][0-9]:[0-5][0-9]:[0-5][0-9]" + return QStringList({"*20[0-5][0-9][01][0-9][0-3][0-9][012][0-9][0-5][0-9][0-5][0-9]*", + "*20[0-5][0-9]-[01][0-9]-[0-3][0-9] [012][0-9]:[0-5][0-9]:[0-5][0-9]*" }); } diff --git a/oscar/SleepLib/profiles.cpp b/oscar/SleepLib/profiles.cpp index 64217a31..0253b068 100644 --- a/oscar/SleepLib/profiles.cpp +++ b/oscar/SleepLib/profiles.cpp @@ -7,6 +7,9 @@ * License. See the file COPYING in the main directory of the source code * for more details. */ +#define TEST_MACROS_ENABLEDoff +#include + #include #include #include @@ -2101,7 +2104,7 @@ void Profile::loadChannels() if (chan->isNull()) { qDebug() << "loadChannels has no idea about channel" << name; - if (in.atEnd()) return; + if (in.atEnd()) break; continue; } chan->setEnabled(enabled); @@ -2122,8 +2125,14 @@ void Profile::loadChannels() chan->setUpperThresholdColor(upperThresholdColor); chan->setShowInOverview(showOverview); - if (in.atEnd()) return; + if (in.atEnd()) break; } - f.close(); + refrehOxiChannelsPref(); } + +void Profile::refrehOxiChannelsPref() { + schema::channel[OXI_Pulse].setLowerThreshold(oxi->flagPulseBelow()); + schema::channel[OXI_Pulse].setUpperThreshold(oxi->flagPulseAbove()); + schema::channel[OXI_SPO2].setLowerThreshold(oxi->oxiDesaturationThreshold()); +}; diff --git a/oscar/SleepLib/profiles.h b/oscar/SleepLib/profiles.h index 44daf68d..2d0f127a 100644 --- a/oscar/SleepLib/profiles.h +++ b/oscar/SleepLib/profiles.h @@ -216,6 +216,7 @@ class Profile : public Preferences void loadChannels(); void saveChannels(); + void refrehOxiChannelsPref(); bool is_first_day; @@ -299,7 +300,11 @@ const QString STR_OS_SPO2DropDuration = "SPO2DropDuration"; const QString STR_OS_SPO2DropPercentage = "SPO2DropPercentage"; const QString STR_OS_PulseChangeDuration = "PulseChangeDuration"; const QString STR_OS_PulseChangeBPM = "PulseChangeBPM"; + const QString STR_OS_SkipOxiIntroScreen = "SkipOxiIntroScreen"; +const QString STR_OS_oxiDesaturationThreshold = "oxiDesaturationThreshold"; +const QString STR_OS_flagPulseAbove = "flagPulseAbove"; +const QString STR_OS_flagPulseBelow = "flagPulseBelow"; // CPAPSettings Strings @@ -489,6 +494,10 @@ class OxiSettings : public PrefSettings initPref(STR_OS_PulseChangeDuration, 8.0); initPref(STR_OS_PulseChangeBPM, 5.0); initPref(STR_OS_SkipOxiIntroScreen, false); + + initPref(STR_OS_oxiDesaturationThreshold, 88); + initPref(STR_OS_flagPulseAbove, 130); + initPref(STR_OS_flagPulseBelow, 40); } bool oximetryEnabled() const { return getPref(STR_OS_EnableOximetry).toBool(); } @@ -502,6 +511,10 @@ class OxiSettings : public PrefSettings double pulseChangeBPM() const { return getPref(STR_OS_PulseChangeBPM).toDouble(); } bool skipOxiIntroScreen() const { return getPref(STR_OS_SkipOxiIntroScreen).toBool(); } + double oxiDesaturationThreshold() const { return getPref(STR_OS_oxiDesaturationThreshold).toDouble(); } + double flagPulseAbove() const { return getPref(STR_OS_flagPulseAbove).toDouble(); } + double flagPulseBelow() const { return getPref(STR_OS_flagPulseBelow).toDouble(); } + void setOximetryEnabled(bool enabled) { setPref(STR_OS_EnableOximetry, enabled); } void setDefaultDevice(QString name) { setPref(STR_OS_DefaultDevice, name); } @@ -517,6 +530,9 @@ class OxiSettings : public PrefSettings void setPulseChangeDuration(double duration) { setPref(STR_OS_PulseChangeDuration, duration); } + void setOxiDesaturationThreshold(double value) { setPref(STR_OS_oxiDesaturationThreshold, value); } + void setFlagPulseAbove(double value) { setPref(STR_OS_flagPulseAbove, value); } + void setFlagPulseBelow(double value) { setPref(STR_OS_flagPulseBelow, value); } }; /*! \class CPAPSettings diff --git a/oscar/SleepLib/schema.cpp b/oscar/SleepLib/schema.cpp index d2676d01..2fabb88e 100644 --- a/oscar/SleepLib/schema.cpp +++ b/oscar/SleepLib/schema.cpp @@ -203,12 +203,9 @@ void init() // Oximetry schema::channel.add(GRP_OXI, new Channel(OXI_Pulse = 0x1800, WAVEFORM, MT_OXIMETER, SESSION, STR_GRAPH_Oxi_Pulse, QObject::tr("Pulse Rate"), QObject::tr("Heart rate in beats per minute"), QObject::tr("Pulse Rate"), STR_UNIT_BPM, DEFAULT, QColor("red"))); - schema::channel[OXI_Pulse].setLowerThreshold(40); - schema::channel[OXI_Pulse].setUpperThreshold(130); schema::channel.add(GRP_OXI, new Channel(OXI_SPO2 = 0x1801, WAVEFORM, MT_OXIMETER, SESSION, STR_GRAPH_Oxi_SPO2, QString("SpO2 %"), QObject::tr("Blood-oxygen saturation percentage"), QString("SpO2"), STR_UNIT_Percentage, DEFAULT, QColor("blue"))); - schema::channel[OXI_SPO2].setLowerThreshold(88); schema::channel.add(GRP_OXI, new Channel(OXI_Plethy = 0x1802, WAVEFORM, MT_OXIMETER, SESSION, STR_GRAPH_Oxi_Plethy, QObject::tr("Plethysomogram"), QObject::tr("An optical Photo-plethysomogram showing heart rhythm"), QObject::tr("Plethy"), STR_UNIT_Hz, DEFAULT, QColor("#404040"))); @@ -396,6 +393,12 @@ void done() schema::channel.channels.clear(); schema::channel.groups.clear(); + // ahiChannels did not get cleared since day1 OSCAR when a reset was required. + // when reset occured then the Overview AHI graph would should an addtional set of channels. + // this fix just clears the variable that stores ahi data. + // probelm #59 https://gitlab.com/pholy/OSCAR-code/-/issues/59 + ahiChannels.clear(); + schema_initialized = false; } diff --git a/oscar/daily.cpp b/oscar/daily.cpp index 1ffac8d4..3ca612aa 100644 --- a/oscar/daily.cpp +++ b/oscar/daily.cpp @@ -541,6 +541,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared) ui->JournalNotes->installEventFilter(this); // qDebug() << "Finished making new Daily object"; // sleep(3); + saveGraphLayoutSettings=nullptr; } Daily::~Daily() @@ -563,6 +564,7 @@ Daily::~Daily() delete ui; delete icon_on; delete icon_off; + if (saveGraphLayoutSettings!=nullptr) delete saveGraphLayoutSettings; } void Daily::showEvent(QShowEvent *) @@ -600,6 +602,7 @@ void Daily::Link_clicked(const QUrl &url) // Reload day LoadDate(previous_date); + mainwin->getOverview()->graphView()->dataChanged(); // 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); @@ -612,6 +615,7 @@ void Daily::Link_clicked(const QUrl &url) // Reload day LoadDate(previous_date); + mainwin->getOverview()->graphView()->dataChanged(); // 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); @@ -620,6 +624,7 @@ void Daily::Link_clicked(const QUrl &url) if (!sess) return; sess->setEnabled(!sess->enabled()); LoadDate(previous_date); + mainwin->getOverview()->graphView()->dataChanged(); } else if (code=="togglepositionsession") { // Enable/Disable Position session day=p_profile->GetDay(previous_date,MT_POSITION); if (!day) return; @@ -627,6 +632,7 @@ void Daily::Link_clicked(const QUrl &url) if (!sess) return; sess->setEnabled(!sess->enabled()); LoadDate(previous_date); + mainwin->getOverview()->graphView()->dataChanged(); } else if (code=="cpap") { day=p_profile->GetDay(previous_date,MT_CPAP); if (day) { @@ -1681,9 +1687,10 @@ void Daily::Load(QDate date) ui->eventsCombo->addItem(*icon_up_down, tr("10 of 10 Event Types"), 0); // Translation used only for spacing for (int i=0; i < available.size(); ++i) { ChannelID code = available.at(i); + int comboxBoxIndex = i+1; schema::Channel & chan = schema::channel[code]; ui->eventsCombo->addItem(chan.enabled() ? *icon_on : * icon_off, chan.label(), code); - ui->eventsCombo->setItemData(i, chan.fullname(), Qt::ToolTipRole); + ui->eventsCombo->setItemData(comboxBoxIndex, chan.fullname(), Qt::ToolTipRole); } setFlagText(); @@ -2849,3 +2856,13 @@ void Daily::on_splitter_2_splitterMoved(int, int) // qDebug() << "Left Panel width set to " << size; AppSetting->setDailyPanelWidth(size); } + +void Daily::on_layout_clicked() { + if (!saveGraphLayoutSettings) { + saveGraphLayoutSettings= new SaveGraphLayoutSettings("daily",this); + } + if (saveGraphLayoutSettings) { + saveGraphLayoutSettings->menu(GraphView); + } +} + diff --git a/oscar/daily.h b/oscar/daily.h index 71198971..5cac0477 100644 --- a/oscar/daily.h +++ b/oscar/daily.h @@ -28,6 +28,7 @@ #include "Graphs/gLineChart.h" #include "sessionbar.h" #include "mytextbrowser.h" +#include "saveGraphLayoutSettings.h" namespace Ui { @@ -272,6 +273,8 @@ private slots: void on_splitter_2_splitterMoved(int pos, int index); + void on_layout_clicked(); + protected: virtual void showEvent(QShowEvent *); @@ -360,6 +363,7 @@ private: #endif bool BookmarksChanged; + SaveGraphLayoutSettings* saveGraphLayoutSettings=nullptr; }; #endif // DAILY_H diff --git a/oscar/daily.ui b/oscar/daily.ui index 6199b948..9dd0ad79 100644 --- a/oscar/daily.ui +++ b/oscar/daily.ui @@ -1557,6 +1557,20 @@ QToolButton:pressed { + + + + Layout + + + + :/icons/cog.png:/icons/cog.png + + + Save and Restore Graph Layout Settings + + + diff --git a/oscar/icons/cog.png b/oscar/icons/cog.png new file mode 100644 index 00000000..bb8aef89 Binary files /dev/null and b/oscar/icons/cog.png differ diff --git a/oscar/icons/exit.png b/oscar/icons/exit.png new file mode 100644 index 00000000..976c77d9 Binary files /dev/null and b/oscar/icons/exit.png differ diff --git a/oscar/icons/plus.png b/oscar/icons/plus.png new file mode 100644 index 00000000..fdc3b603 Binary files /dev/null and b/oscar/icons/plus.png differ diff --git a/oscar/icons/question_mark.png b/oscar/icons/question_mark.png new file mode 100644 index 00000000..ddd4b9d1 Binary files /dev/null and b/oscar/icons/question_mark.png differ diff --git a/oscar/icons/rename.png b/oscar/icons/rename.png new file mode 100644 index 00000000..0adf88d0 Binary files /dev/null and b/oscar/icons/rename.png differ diff --git a/oscar/icons/restore.png b/oscar/icons/restore.png new file mode 100644 index 00000000..d0cdac5f Binary files /dev/null and b/oscar/icons/restore.png differ diff --git a/oscar/icons/return.png b/oscar/icons/return.png new file mode 100644 index 00000000..ca57b820 Binary files /dev/null and b/oscar/icons/return.png differ diff --git a/oscar/icons/trash_can.png b/oscar/icons/trash_can.png new file mode 100644 index 00000000..5e745bcd Binary files /dev/null and b/oscar/icons/trash_can.png differ diff --git a/oscar/icons/update.png b/oscar/icons/update.png new file mode 100644 index 00000000..8f4d0e9c Binary files /dev/null and b/oscar/icons/update.png differ diff --git a/oscar/mainwindow.cpp b/oscar/mainwindow.cpp index bcc8cfb2..846c43bb 100644 --- a/oscar/mainwindow.cpp +++ b/oscar/mainwindow.cpp @@ -7,6 +7,10 @@ * License. See the file COPYING in the main directory of the source code * for more details. */ +#define TEST_MACROS_ENABLED +#include + + #include #include #include @@ -132,6 +136,7 @@ QString MainWindow::getMainWindowTitle() void MainWindow::SetupGUI() { setupRunning = true; + setWindowTitle(getMainWindowTitle()); #ifdef Q_OS_MAC @@ -233,11 +238,17 @@ void MainWindow::SetupGUI() ui->action_Sidebar_Toggle->setChecked(b); ui->toolBox->setVisible(b); + bool oldState = ui->actionShowPersonalData->blockSignals(true); ui->actionShowPersonalData->setChecked(AppSetting->showPersonalData()); + ui->actionShowPersonalData->blockSignals(oldState); + oldState = ui->actionPie_Chart->blockSignals(true); ui->actionPie_Chart->setChecked(AppSetting->showPieChart()); + ui->actionPie_Chart->blockSignals(oldState); + oldState = ui->actionDaily_Calendar->blockSignals(true); ui->actionDaily_Calendar->setChecked(AppSetting->calendarVisible()); + ui->actionDaily_Calendar->blockSignals(oldState); on_tabWidget_currentChanged(0); diff --git a/oscar/oscar.pro b/oscar/oscar.pro index 2f908712..12ce9a97 100644 --- a/oscar/oscar.pro +++ b/oscar/oscar.pro @@ -255,6 +255,8 @@ lessThan(QT_MAJOR_VERSION,5)|lessThan(QT_MINOR_VERSION,12) { SOURCES += \ checkupdates.cpp \ + saveGraphLayoutSettings.cpp \ + overview.cpp \ common_gui.cpp \ cprogressbar.cpp \ daily.cpp \ @@ -262,7 +264,6 @@ SOURCES += \ main.cpp \ mainwindow.cpp \ newprofile.cpp \ - overview.cpp \ preferencesdialog.cpp \ # psettings.cpp \ reports.cpp \ @@ -361,13 +362,14 @@ QMAKE_EXTRA_COMPILERS += optimize HEADERS += \ checkupdates.h \ + saveGraphLayoutSettings.h \ + overview.h \ common_gui.h \ cprogressbar.h \ daily.h \ exportcsv.h \ mainwindow.h \ newprofile.h \ - overview.h \ preferencesdialog.h \ # psettings.h \ reports.h \ diff --git a/oscar/overview.cpp b/oscar/overview.cpp index a2e6907a..661c154c 100644 --- a/oscar/overview.cpp +++ b/oscar/overview.cpp @@ -171,6 +171,7 @@ Overview::Overview(QWidget *parent, gGraphView *shared) : connect(GraphView, SIGNAL(updateRange(double,double)), this, SLOT(on_RangeUpdate(double,double))); connect(GraphView, SIGNAL(GraphsChanged()), this, SLOT(updateGraphCombo())); connect(GraphView, SIGNAL(XBoundsChanged(qint64 ,qint64)), this, SLOT(on_XBoundsChanged(qint64 ,qint64))); + saveGraphLayoutSettings=nullptr; } Overview::~Overview() @@ -192,6 +193,7 @@ Overview::~Overview() delete icon_off ; delete icon_up_down ; delete icon_warning ; + if (saveGraphLayoutSettings!=nullptr) delete saveGraphLayoutSettings; } void Overview::ResetFont() @@ -308,6 +310,11 @@ void Overview::CreateAllGraphs() { G->AddLayer(sc); chartsToBeMonitored.insert(sc,G); } else if (chan->type() == schema::WAVEFORM) { + if ((code==CPAP_AHI)||(code==CPAP_Pressure) ) { + DEBUGF O("SKIPPING") NAME(code) Q(code); + //skip if channel is for AHI. + continue; + } sc= new gSummaryChart(code, chan->machtype()); G->AddLayer(sc); chartsToBeMonitored.insert(sc,G); @@ -933,3 +940,14 @@ void Overview::on_toggleVisibility_clicked(bool checked) GraphView->redraw(); } +void Overview::on_layout_clicked() { + if (!saveGraphLayoutSettings) { + saveGraphLayoutSettings= new SaveGraphLayoutSettings("overview",this); + } + if (saveGraphLayoutSettings) { + saveGraphLayoutSettings->menu(GraphView); + } +} + + + diff --git a/oscar/overview.h b/oscar/overview.h index a5497b4a..d9c7ccfd 100644 --- a/oscar/overview.h +++ b/oscar/overview.h @@ -23,6 +23,11 @@ #include "Graphs/gOverviewGraph.h" #endif #include "Graphs/gSummaryChart.h" +#include "saveGraphLayoutSettings.h" + +#include +#include +#include namespace Ui { class Overview; @@ -140,6 +145,8 @@ class Overview : public QWidget void on_RangeUpdate(double minx, double maxx); void setGraphText (); + void on_layout_clicked(); + private: void CreateAllGraphs(); void timedUpdateOverview(int ms=0); @@ -193,6 +200,7 @@ class Overview : public QWidget // Are start and end widgets displaying the same month. bool samePage; + SaveGraphLayoutSettings* saveGraphLayoutSettings=nullptr; }; diff --git a/oscar/overview.ui b/oscar/overview.ui index 267cd9b8..65483855 100644 --- a/oscar/overview.ui +++ b/oscar/overview.ui @@ -259,6 +259,20 @@ QToolButton:pressed { + + + + Layout + + + + :/icons/cog.png:/icons/cog.png + + + Save and Restore Graph Layout Settings + + + diff --git a/oscar/preferencesdialog.cpp b/oscar/preferencesdialog.cpp index a2f5ea19..827acc8f 100644 --- a/oscar/preferencesdialog.cpp +++ b/oscar/preferencesdialog.cpp @@ -130,9 +130,10 @@ PreferencesDialog::PreferencesDialog(QWidget *parent, Profile *_profile) : ui->showLeakRedline->setChecked(profile->cpap->showLeakRedline()); ui->leakRedlineSpinbox->setValue(profile->cpap->leakRedline()); - ui->oxiDesaturationThreshold->setValue(schema::channel[OXI_SPO2].lowerThreshold()); - ui->flagPulseAbove->setValue(schema::channel[OXI_Pulse].upperThreshold()); - ui->flagPulseBelow->setValue(schema::channel[OXI_Pulse].lowerThreshold()); + //change initialization from hard coded in schema to profile data. + ui->oxiDesaturationThreshold->setValue(profile->oxi->oxiDesaturationThreshold()); + ui->flagPulseAbove->setValue(profile->oxi->flagPulseAbove()); + ui->flagPulseBelow->setValue(profile->oxi->flagPulseBelow()); ui->spo2Drop->setValue(profile->oxi->spO2DropPercentage()); ui->spo2DropTime->setValue(profile->oxi->spO2DropDuration()); @@ -819,11 +820,6 @@ bool PreferencesDialog::Save() } } - schema::channel[OXI_SPO2].setLowerThreshold(ui->oxiDesaturationThreshold->value()); - schema::channel[OXI_Pulse].setLowerThreshold(ui->flagPulseBelow->value()); - schema::channel[OXI_Pulse].setUpperThreshold(ui->flagPulseAbove->value()); - - AppSetting->setUserEventPieChart(ui->showUserFlagsInPie->isChecked()); profile->session->setLockSummarySessions(ui->LockSummarySessionSplitting->isChecked()); profile->session->setWarnOnUntestedMachine(ui->warnOnUntestedMachine->isChecked()); @@ -898,6 +894,10 @@ bool PreferencesDialog::Save() profile->oxi->setPulseChangeDuration(ui->pulseChangeTime->value()); profile->oxi->setOxiDiscardThreshold(ui->oxiDiscardThreshold->value()); + profile->oxi->setOxiDesaturationThreshold(ui->oxiDesaturationThreshold->value()); + profile->oxi->setFlagPulseAbove(ui->flagPulseAbove->value()); + profile->oxi->setFlagPulseBelow(ui->flagPulseBelow->value()); + profile->cpap->setAHIWindow(ui->ahiGraphWindowSize->value()); profile->cpap->setAHIReset(ui->ahiGraphZeroReset->isChecked()); @@ -981,6 +981,7 @@ bool PreferencesDialog::Save() p_pref->Save(); profile->Save(); + profile->refrehOxiChannelsPref(); if (recompress_events) { mainwin->recompressEvents(); diff --git a/oscar/saveGraphLayoutSettings.cpp b/oscar/saveGraphLayoutSettings.cpp new file mode 100644 index 00000000..009515e4 --- /dev/null +++ b/oscar/saveGraphLayoutSettings.cpp @@ -0,0 +1,916 @@ +/* user graph settings Implementation + * + * Copyright (c) 2019-2022 The OSCAR Team + * Copyright (c) 2011-2018 Mark Watkins + * + * 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. */ + +#define TEST_MACROS_ENABLED +#include + +#include +#include +#include +#include +#include +#include "SleepLib/profiles.h" +#include "saveGraphLayoutSettings.h" + +#define USE_FRAMELESS_WINDOW +#define USE_PROFILE_SPECIFIC_FOLDERoff // off implies saved layouts worked for all profiles. + +SaveGraphLayoutSettings::SaveGraphLayoutSettings(QString title,QWidget* parent) : parent(parent),title(title) +{ + fontSizeIncrease = 1; + helpFontSizeIncrease = 1; + createSaveFolder(); + if (dir==nullptr) return; + dir->setFilter(QDir::Files | QDir::Readable | QDir::Writable | QDir::NoSymLinks); + + QString descFileName = dirName+title.toLower()+".descriptions.txt"; + descriptionMap = new DescriptionMap (dir,descFileName); + + createMenu() ; + + menuDialog->connect(menuAddFullBtn, SIGNAL(clicked()), this, SLOT (addFull_feature() )); + menuDialog->connect(menuAddBtn, SIGNAL(clicked()), this, SLOT (add_feature() )); + menuDialog->connect(menuRestoreBtn, SIGNAL(clicked()), this, SLOT (restore_feature() )); + menuDialog->connect(menuUpdateBtn, SIGNAL(clicked()), this, SLOT (update_feature() )); + menuDialog->connect(menuRenameBtn, SIGNAL(clicked()), this, SLOT (rename_feature() )); + menuDialog->connect(menuDeleteBtn, SIGNAL(clicked()), this, SLOT (delete_feature() )); + menuDialog->connect(menuHelpBtn, SIGNAL(clicked()), this, SLOT (help_feature() )); + menuDialog->connect(menuExitBtn, SIGNAL(clicked()), this, SLOT (exit() )); + menuDialog->connect(menuList, SIGNAL(itemChanged(QListWidgetItem*)), this, SLOT(itemChanged(QListWidgetItem*) )); + menuDialog->connect(menuList, SIGNAL(itemSelectionChanged()), this, SLOT(itemSelectionChanged() )); + + + + singleLineRe = new QRegularExpression( QString("^\\s*([^\\r\\n]{0,%1})").arg(1+maxDescriptionLen) ); + fileNumRe = new QRegularExpression( QString("%1(\\d+)(.shg)?$").arg(fileBaseName) ); + parseFilenameRe = new QRegularExpression(QString("^(%1[.](%2(\\d*)))[.]shg$").arg(title).arg(fileBaseName)); +} + +SaveGraphLayoutSettings::~SaveGraphLayoutSettings() +{ + + if (dir==nullptr) {return;} + menuDialog->disconnect(menuAddFullBtn, SIGNAL(clicked()), this, SLOT (addFull_feature() )); + menuDialog->disconnect(menuAddBtn, SIGNAL(clicked()), this, SLOT (add_feature() )); + menuDialog->disconnect(menuRestoreBtn, SIGNAL(clicked()), this, SLOT (restore_feature() )); + menuDialog->disconnect(menuUpdateBtn, SIGNAL(clicked()), this, SLOT (update_feature() )); + menuDialog->disconnect(menuRenameBtn, SIGNAL(clicked()), this, SLOT (rename_feature() )); + menuDialog->disconnect(menuDeleteBtn, SIGNAL(clicked()), this, SLOT (delete_feature() )); + menuDialog->disconnect(menuExitBtn, SIGNAL(clicked()), this, SLOT (exit() )); + menuDialog->disconnect(menuHelpBtn, SIGNAL(clicked()), this, SLOT (help_feature() )); + menuDialog->disconnect(menuList, SIGNAL(itemChanged(QListWidgetItem*)), this, SLOT(itemChanged(QListWidgetItem*) )); + menuDialog->disconnect(menuList, SIGNAL(itemSelectionChanged()), this, SLOT(itemSelectionChanged() )); + helpDestructor(); + + + delete descriptionMap; + delete singleLineRe; + delete fileNumRe; + delete parseFilenameRe; +} + +void SaveGraphLayoutSettings::createSaveFolder() { + // Insure that the save folder exists + // Get the directory name for the save files + //QString layoutFileFolder = "savedGraphLayoutSettings/"; + QString layoutFileFolder = "layoutSettings/"; + + #if 0 + // home directory for the current profile. + QString baseName = p_profile->Get("{DataFolder}/"); + + #else + // home directory for all profiles. + // allows settings to be shared accross profiles. + QString baseName = p_pref->Get("{home}/"); + + #endif + + dirName = baseName+layoutFileFolder; + + // Check if the save folder exists + QDir* tmpDir = new QDir(dirName); + if (!tmpDir->exists()) { + QDir* baseDir=new QDir(baseName); + if (!baseDir->exists()) { + // Base folder does not exist - terminate + return ; + } + // saved Setting folder does not exist. make it + if (!baseDir->mkdir(dirName)) { + // Did not create the folder. + return ; + } + tmpDir = new QDir(dirName); + // double check if save folder exists or not. + if (!tmpDir->exists()) { + return ; + } + } + dir=tmpDir; +} + +QPushButton* SaveGraphLayoutSettings::menuBtn(QString name, QIcon* icon, QString style,QSizePolicy::Policy hPolicy,QString tooltip) { + return newBtnRtn(menuLayoutButtons, name, icon, style, hPolicy,tooltip) ; +} +QPushButton* SaveGraphLayoutSettings::helpBtn(QString name, QIcon* icon, QString style,QSizePolicy::Policy hPolicy,QString tooltip) { + return newBtnRtn(helpLayoutButtons, name, icon, style, hPolicy,tooltip) ; +} +QPushButton* SaveGraphLayoutSettings::newBtnRtn(QHBoxLayout* layout,QString name, QIcon* icon, QString style,QSizePolicy::Policy hPolicy,QString tooltip) { + QPushButton* button = new QPushButton(name, menuDialog); + button->setIcon(*icon); + button->setStyleSheet(style); + button->setSizePolicy(hPolicy,QSizePolicy::Fixed); + button->setToolTip(tooltip); + button->setToolTipDuration(AppSetting->tooltipTimeout()); + button->setFont(*mediumfont); + layout->addWidget(button); + return button; +} + +void SaveGraphLayoutSettings::createStyleSheets() { + styleOn= calculateButtonStyle(true,false); + styleOff= calculateButtonStyle(false,false); + styleExit= calculateButtonStyle(true,true); + styleMessageBox= + "QMessageBox { " + "background-color:yellow;" + //"color:black;" + "color:red;" + "border: 2px solid black;" + "text-align: right;" + "font-size: 16px;" + "}" + "QTextEdit {" + "background-color:lightblue;" + //"color:black;" + "color:red;" + "border: 9px solid black;" + "text-align: center;" + "vertical-align:top;" + "}" + ; + styleDialog= + "QDialog { " + "border: 3px solid black;" + "}"; +} + + +void SaveGraphLayoutSettings::createMenu() { + + menuListFont =*defaultfont; + menuListFont.setPointSize(fontSizeIncrease+menuListFont.pointSize()); + + + createStyleSheets(); + + #ifdef USE_FRAMELESS_WINDOW + menuDialog= new QDialog(parent, Qt::FramelessWindowHint); + menuDialog->setSizeGripEnabled(true); // allows lower right hand corner to resize dialog + #else + menuDialog= new QDialog(parent, Qt::WindowTitleHint | Qt::WindowCloseButtonHint|Qt::WindowSystemMenuHint); + menuDialog->setWindowTitle(tr("Manage Save Layout Settings")) ; + #endif + menuDialog->setStyleSheet(styleDialog); + + menuLayout = new QVBoxLayout(menuDialog); + menuLayoutButtons = new QHBoxLayout(); + + menuAddFullBtn = menuBtn(tr("Add" ),m_icon_add , styleOff ,QSizePolicy::Minimum, tr("Add Feature inhibited. The maximum number of Items has been exceeded.") ); + menuAddBtn = menuBtn(tr("Add" ),m_icon_add , styleOn ,QSizePolicy::Minimum, tr("creates new copy of current settings.")); + menuRestoreBtn = menuBtn(tr("Restore"),m_icon_restore , styleOn ,QSizePolicy::Minimum, tr("Restores saved settings from selection.")); + menuRenameBtn = menuBtn(tr("Rename" ),m_icon_rename , styleOn ,QSizePolicy::Minimum, tr("Renames the selection. Must edit existing name then press enter.")); + menuUpdateBtn = menuBtn(tr("Update" ),m_icon_update , styleOn ,QSizePolicy::Minimum, tr("Updates the selection with current settings.")); + menuDeleteBtn = menuBtn(tr("Delete" ),m_icon_delete , styleOn ,QSizePolicy::Minimum, tr("Deletes the selection.")); + menuHelpBtn = menuBtn("" ,m_icon_help , styleOn ,QSizePolicy::Fixed , tr("Expanded Help menu.")); + menuExitBtn = menuBtn("" ,m_icon_exit , styleExit,QSizePolicy::Fixed , tr("Exits the Layout menu.")); + + #ifndef USE_FRAMELESS_WINDOW + menuExitBtn->hide(); + #endif + + menuList = new QListWidget(menuDialog); + menuList->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn); + menuList->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOn); + menuList->setFont(menuListFont); + + + menuLayout->addLayout(menuLayoutButtons); + menuLayout->addWidget(menuList, 1); + + //menuList->setSizePolicy(QSizePolicy::Preferred,QSizePolicy::Preferred); + menuList->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); + + +}; + +void SaveGraphLayoutSettings::createHelp() { + if(helpDialog) return; + + helpDialog= new QDialog(parent, Qt::FramelessWindowHint); + helpDialog->setStyleSheet(styleDialog); + menuDialog->setSizeGripEnabled(true); // allows lower right hand corner to resize dialog + + QFont helpInfoFont = menuListFont; + helpInfoFont.setPointSize(helpFontSizeIncrease+helpInfoFont.pointSize()); + + QFont helpInfoLabelFont = helpInfoFont; + helpInfoLabelFont.setPointSize(fontSizeIncrease+ helpInfoFont.pointSize()); + + QLabel* helpInfoHeaderLabel = new QLabel("helpInfoHeaderLabel",parent); + helpInfoHeaderLabel->setText(QString( tr("

Help Menu - Manage Layout Settings

"))); + helpInfoHeaderLabel->setFont(helpInfoLabelFont); + + QLabel* helpInfoLabel = new QLabel("helpInfo",parent); + helpInfoLabel->setFont(helpInfoFont); + helpInfoLabel->setText(helpInfo()) ; + + helpLayoutButtons = new QHBoxLayout(); + helpLayoutButtons->addWidget(helpInfoHeaderLabel); + helpInfoExitBtn= helpBtn("" ,m_icon_return , styleOn ,QSizePolicy::Fixed , tr("Exits the help menu.")); + helpExitBtn = helpBtn("" ,m_icon_exit , styleExit,QSizePolicy::Fixed , tr("Exits the dialog menu.")); + + QVBoxLayout* helpLayout = new QVBoxLayout(helpDialog); + helpLayout->addLayout(helpLayoutButtons); + helpLayout->addWidget(helpInfoLabel, 1); + + helpDialog->connect(helpInfoExitBtn,SIGNAL(clicked()), this, SLOT (help_off_feature() )); + helpDialog->connect(helpExitBtn, SIGNAL(clicked()), this, SLOT (help_exit_feature() )); + +} + +void SaveGraphLayoutSettings::helpDestructor() { + if(!helpDialog) return; + helpDialog->disconnect(helpInfoExitBtn,SIGNAL(clicked()), this, SLOT (help_off_feature() )); + helpDialog->disconnect(helpExitBtn, SIGNAL(clicked()), this, SLOT (help_exit_feature() )); +} + +QString SaveGraphLayoutSettings::helpInfo() { +return QString( tr("\ +

\ + This feature manages the saving and restoring of Layout Settings.\ +
\ + Layout Settings control the layout of a graph or chart.\ +
\ + Different Layouts Settings can be saved and later restored.\ +
\ +

\ + \ + \ + \ + \ + \ +
\ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ +
ButtonDescription
AddCreates a copy of the current Layout Settings.
\ + The default description is the current date.
\ + The description may be changed.
\ + The Add button will be greyed out when maximum number is reached.
Other Buttons Greyed out when there are no selections
RestoreLoads the Layout Settings from the selection. Automatically exits.
Rename Modify the description of the selection. Same as a double click.
Update Saves the current Layout Settings to the selection.
\ + Prompts for confirmation.
DeleteDeletes the selecton.
\ + Prompts for confirmation.
Control
Exit (Red circle with a white \"X\".) Returns to OSCAR menu.
ReturnNext to Exit icon. Only in Help Menu. Returns to Layout menu.
Escape KeyExit the Help or Layout menu.
\ +

Layout Settings

\ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ + \ +
* Name* Pinning* Plots Enabled * Height
* Order* Event Flags* Dotted Lines* Height Options
\ +

General Information

\ +
    \ +
  • Maximum description size = 80 characters.
  • \ +
  • Maximum Saved Layout Settings = 30.
  • \ +
  • Saved Layout Settings can be accessed by all profiles. \ +
  • Layout Settings only control the layout of a graph or chart.
    \ + They do not contain any other data.
    \ + They do not control if a graph is displayed or not.
  • \ +
  • Layout Settings for daily and overview are managed independantly.
  • \ +
\ +")); +} + +const QString SaveGraphLayoutSettings::calculateStyleMessageBox(QFont* font , QString& s1, QString& s2) { + QFontMetrics fm = QFontMetrics(*font); + int width = fm.boundingRect(s1).size().width(); + int width2 = fm.boundingRect(s2).size().width(); + width = qMax(width,width2) + iconWidthMessageBox; + + QString style = QString("%1 QLabel{%2 min-width: %3px;}" ). + arg(styleMessageBox). + arg("text-align: center;" "color:black;"). + arg(width); + + return style; +} + +bool SaveGraphLayoutSettings::verifyItem(QListWidgetItem* item,QString text,QIcon* icon) { + //if (verifyhelp() ) return false; + if (item) return true; + initminMenuListSize(); + confirmAction( text ,"" , icon); + return false; +} + +#if 1 +bool SaveGraphLayoutSettings::confirmAction(QString top ,QString bottom,QIcon* icon,QMessageBox::StandardButtons flags , QMessageBox::StandardButton adefault, QMessageBox::StandardButton success) { + //QString topText=QString("
%1
").arg(top); + //QString bottomText=QString("
%1
").arg(bottom); + QString topText=QString("

%1

").arg(top); + QString bottomText=QString("

%1

").arg(bottom); + + QMessageBox msgBox(menuDialog); + msgBox.setText(topText); + msgBox.setInformativeText(bottomText); + if (icon!=nullptr) { + msgBox.setIconPixmap(icon->pixmap(QSize(iconWidthMessageBox,iconWidthMessageBox))); + } + // may be good for help menu. a pull down box with box of data. msgBox.setDetailedText("some detailed string"); + msgBox.setStandardButtons(flags); + msgBox.setDefaultButton(adefault); + msgBox.setWindowFlag(Qt::FramelessWindowHint,true); + + msgBox.setStyleSheet(calculateStyleMessageBox(&menuListFont,top,bottom)); + + // displaywidgets((QWidget*)&msgBox); + bool ret= msgBox.exec()==success; + return ret; +} +#else +bool SaveGraphLayoutSettings::confirmAction(QString name ,QString question,QIcon* icon,QMessageBox::StandardButtons flags , QMessageBox::StandardButton adefault, QMessageBox::StandardButton success) { +//bool SaveGraphLayoutSettings::confirmAction(QString name,QString question,QIcon* icon) + Q_UNUSED(flags); + Q_UNUSED(adefault); + Q_UNUSED(success); + QMessageBox msgBox(menuDialog); + msgBox.setText(question); + if (icon!=nullptr) { + msgBox.setIconPixmap(icon->pixmap(QSize(50,50))); + } + + msgBox.setStandardButtons(QMessageBox::Cancel | QMessageBox::Yes ); + msgBox.setDefaultButton(QMessageBox::Cancel); + msgBox.setStyleSheet(styleMessageBox); + msgBox.setWindowFlag(Qt::FramelessWindowHint,true); + + return (msgBox.exec()==QMessageBox::Yes); + Q_UNUSED(name); + +} +#endif + +QString SaveGraphLayoutSettings::calculateButtonStyle(bool on,bool exitBtn){ + QString btnStyle=QString("QPushButton{%1%2}").arg(on + ?"color: black;background-color:white;" + :"color: darkgray;background-color:#e8e8e8 ;" + ).arg(!exitBtn ? + "" + : + "background: transparent;" + "border-radius: 8px;" + "border: 2px solid transparent;" + "max-width:1em;" + "border:none;" + ); + + QString toolTipStyle=" QToolTip { " + "border: 1px solid black;" + "border-width: 1px;" + "padding: 4px;" + "font: 14px ; color:black; background-color:yellow;" + "}"; + btnStyle.append(toolTipStyle); + return btnStyle; +} + +void SaveGraphLayoutSettings::looksOn(QPushButton* button,bool on){ + //button->setEnabled(on); + button->setStyleSheet(on?styleOn:styleOff); +} + +void SaveGraphLayoutSettings::manageButtonApperance() { + + bool enable = menuList->currentItem(); + looksOn(menuUpdateBtn,enable); + looksOn(menuRestoreBtn,enable); + looksOn(menuDeleteBtn,enable); + looksOn(menuRenameBtn,enable); + + if (nextNumToUse<0) { // check if at Maximum limit + // must always hide first + menuAddBtn->hide(); + menuAddFullBtn->show(); + } else { + // must always hide first + menuAddFullBtn->hide(); + menuAddBtn->show(); + } +} + + +void SaveGraphLayoutSettings::add_feature() { + if(!graphView) return; + QString fileName = QString("%1%2").arg(fileBaseName).arg(nextNumToUse,fileNumMaxLength,10,QLatin1Char('0')); + writeSettings(fileName); + // create a default description - use formatted datetime. + QString desc=QDateTime::currentDateTime().toString(); + descriptionMap->add(fileName,desc); + descriptionMap->save(); + QListWidgetItem* item = updateFileList( fileName); + if (item!=nullptr) { + menuList->setCurrentItem(item,QItemSelectionModel::ClearAndSelect); + menuList->editItem(item); + } + menuList->sortItems(); + resizeMenu(); +} + +void SaveGraphLayoutSettings::addFull_feature() { + verifyItem( nullptr,tr("Maximum number of Items exceeded.") , m_icon_add); +} + +void SaveGraphLayoutSettings::update_feature() { + if(!graphView) return; + QListWidgetItem * item=menuList->currentItem(); + if (!verifyItem(item, tr("No Item Selected") , m_icon_update)) return ; + if(!confirmAction( item->text(), tr("Ok to Update?") , m_icon_update) ) return; + QString fileName = item->data(fileNameRole).toString(); + writeSettings(fileName); +}; + +void SaveGraphLayoutSettings::restore_feature() { + if(!graphView) return; + QListWidgetItem * item=menuList->currentItem(); + if (!verifyItem(item, tr("No Item Selected") , m_icon_restore)) return ; + QString fileName = item->data(fileNameRole).toString(); + loadSettings(fileName); + exit(); +}; + +void SaveGraphLayoutSettings::rename_feature() { + if(!graphView) return; + QListWidgetItem * item=menuList->currentItem(); + if (!verifyItem(item, tr("No Item Selected") , m_icon_rename)) return ; + menuList->editItem(item); + // SaveGraphLayoutSettings::itemChanged(QListWidgetItem *item) is called when edit changes the entry. + // itemChanged will update the description map +} + +void SaveGraphLayoutSettings::help_exit_feature() { + helpDialog->close(); + exit(); +} + +void SaveGraphLayoutSettings::help_off_feature() { + helpDialog->close(); +} + +void SaveGraphLayoutSettings::help_feature() { + initminMenuListSize(); + createHelp(); + if(!helpDialog) return; + helpDialog->raise(); + helpDialog->exec(); + manageButtonApperance(); +} + +void SaveGraphLayoutSettings::delete_feature() { + if(!graphView) return; + QListWidgetItem * item=menuList->currentItem(); + if (!verifyItem(item, tr("No Item Selected") , m_icon_delete)) return ; + if(!confirmAction(item->text(), tr("Ok To Delete?") ,m_icon_delete) ) return; + + QString fileName = item->data(fileNameRole).toString(); + descriptionMap->remove(fileName); + descriptionMap->save(); + deleteSettings(fileName); + delete item; + if (nextNumToUse<0) { + nextNumToUse=fileNum(fileName); + } + manageButtonApperance(); + resizeMenu(); +} + +void SaveGraphLayoutSettings::itemChanged(QListWidgetItem *item) +{ + QString fileName=item->data(fileNameRole).toString(); + QString desc= item->text(); + + // use only the first line in a multiline string. Can be set using cut and paste + QRegularExpressionMatch match = singleLineRe->match(desc); + if (match.hasMatch()) { + // captured match is the first line and has been truncated + desc=match.captured(1).trimmed(); // reoves spaces at end. + } else { + // no match. + // an invalid name was entered. too much white space or empty + desc=""; + } + if (desc.length()>maxDescriptionLen) { + desc.append("..."); + } + if (desc.length() <=0) { + // returns name back to previous saved name + desc=descriptionMap->get(fileName); + } else { + descriptionMap->add(fileName,desc); + descriptionMap->save(); + } + item->setText(desc); + menuList->sortItems(); + menuList->setCurrentItem(item); + resizeMenu(); + +} + +void SaveGraphLayoutSettings::itemSelectionChanged() +{ + initminMenuListSize(); + manageButtonApperance(); +} + +void SaveGraphLayoutSettings::initminMenuListSize() { + if (minMenuDialogSize.width()==0) { + menuDialogSize = menuDialog->size(); + minMenuDialogSize = menuDialogSize; + + menuListSize = menuList->size(); + minMenuListSize = menuListSize; + + dialogListDiff = menuDialogSize - menuListSize; + + dialogListDiff.setWidth (horizontalWidthAdjustment + dialogListDiff.width()); + + resizeMenu(); + } +}; + +void SaveGraphLayoutSettings::writeSettings(QString filename) { + graphView->SaveSettings(title+"."+filename,dirName); +}; + +void SaveGraphLayoutSettings::loadSettings(QString filename) { + graphView->LoadSettings(title+"."+filename,dirName); +}; + +void SaveGraphLayoutSettings::deleteSettings(QString filename) { + QString fileName=graphView->settingsFilename (title+"."+filename,dirName) ; + dir->remove(fileName); +}; + +int SaveGraphLayoutSettings::fileNum(QString fileName) { + QRegularExpressionMatch match = fileNumRe->match(fileName); + int value=-1; + if (match.hasMatch()) { + value=match.captured(1).toInt(); + } + return value; +} + + +QSize SaveGraphLayoutSettings::maxSize(const QSize AA , const QSize BB ) { + return QSize ( qMax(AA.width(),BB.width()) , qMax(AA.height(),BB.height() ) ); +} + +bool SaveGraphLayoutSettings::sizeEqual(const QSize AA , const QSize BB ) { + return (AA.width()==BB.width()) && ( AA.height()==BB.height()) ; +} + + + +void SaveGraphLayoutSettings::resizeMenu() { + if (minMenuDialogSize.width()==0) return; + + QSize newSize = calculateMenuDialogSize(); + newSize.setWidth ( newSize.width()); + + menuDialogSize = menuDialog->size(); + + if ( sizeEqual(newSize , menuDialogSize)) { + // no work to do + return; + }; + + if ( menuDialogSize.width() < newSize.width() ) { + menuDialog->setMinimumWidth(newSize.width()); + menuDialog->setMaximumWidth(QWIDGETSIZE_MAX); + } else if ( menuDialogSize.width() > newSize.width() ) { + menuDialog->setMaximumWidth(newSize.width()); + menuDialog->setMinimumWidth(newSize.width()); + } + if ( menuDialogSize.height() < newSize.height() ) { + menuDialog->setMinimumHeight(newSize.height()); + menuDialog->setMaximumHeight(QWIDGETSIZE_MAX); + } else if ( menuDialogSize.height() > newSize.height() ) { + menuDialog->setMaximumHeight(newSize.height()); + menuDialog->setMinimumHeight(newSize.height()); + } + menuDialogSize = newSize; +} + +QSize SaveGraphLayoutSettings::calculateMenuDialogSize() { + if (menuDialogSize.width()==0) return QSize(0,0); + QListWidgetItem* item; + widestItem=nullptr; + QFontMetrics fm = QFontMetrics(menuListFont); + + // account for scrollbars. + QSize returnValue = QSize( 0 , fm.height() ); // add an extra space at the bottom. width didn't work + + // Account for dialog Size + returnValue += dialogListDiff; + QSize size; + + for (int index = 0; index < menuList->count(); ++index) { + item = menuList->item(index); + if (!item) continue; + size = fm.boundingRect(item->text()).size(); + if (returnValue.width() < size.width()) { + returnValue.setWidth( qMax( returnValue.width(),size.width())); + widestItem=item; + } + returnValue.setHeight( returnValue.height()+size.height()); + } + returnValue.setWidth( horizontalWidthAdjustment + returnValue.width() ) ; + + returnValue = maxSize(returnValue, minMenuDialogSize); + return returnValue; +} + +QListWidgetItem* SaveGraphLayoutSettings::updateFileList(QString find) { + QListWidgetItem* ret=nullptr; + manageButtonApperance(); + dir->refresh(); + QFileInfoList filelist = dir->entryInfoList( QDir::Files | QDir::Readable | QDir::Writable | QDir::NoSymLinks,QDir::Name); + + // Restrict number of files. easy to find availble unused entry for add function. + + int row=0; + int count=0; + menuList->clear(); + nextNumToUse=-1; + descriptionMap->load(); + for (int i = 0; i < filelist.size(); ++i) { + QFileInfo fileInfo = filelist.at(i); + QString fileName = fileInfo.fileName(); + QRegularExpressionMatch match = parseFilenameRe->match(fileName); + if (match.hasMatch()) { + if (match.lastCapturedIndex()==3) { + QString fileName=match.captured(2); + if (nextNumToUse<0) { + // check if an entry is availavle to use + int fileNum=match.captured(3).toInt(); + // find an available file name(number); + if (fileNum!=count) { + nextNumToUse=count; + } + } + count++; + + QListWidgetItem *item = new QListWidgetItem(descriptionMap->get(fileName)); + item->setData(fileNameRole,fileName); + item->setFlags(item->flags() | Qt::ItemIsEditable); + menuList->insertItem(row,item); + row++; + if (find!=nullptr && fileName==find) { + ret=item; + } + } + } + } + if (nextNumToUse<0) { // check if there is an existing empty slot + // if not then the next available slot is at the end. CHeck if at max files. + if (countsortItems(); + return ret; +} + + +void SaveGraphLayoutSettings::exit() { + menuDialog->close(); +} + +void SaveGraphLayoutSettings::menu(gGraphView* graphView) { + if (dir==nullptr) { + //const char* err=qPrintable(QString("Cannot find directory %1").arg(dirName)); + //qWarning(err); + return; + } + this->graphView=graphView; + updateFileList(); + manageButtonApperance(); + menuDialog->raise(); + menuDialog->exec(); + exit(); +} + +//==================================================================================================== +//==================================================================================================== +// Descriptions map class with file storage + +DescriptionMap::DescriptionMap(QDir* dir, QString _filename) +{ + filename = dir->absoluteFilePath(_filename); + parseDescriptionsRe = new QRegularExpression(QString("^\\s*(\\w+)%1(.*)$").arg(delimiter) ); +}; + +DescriptionMap::~DescriptionMap() { + delete parseDescriptionsRe; +}; + +void DescriptionMap::add(QString key,QString desc) { + descriptions.insert(key,desc); +}; + +void DescriptionMap::remove(QString key) { + descriptions.remove(key); +} +QString DescriptionMap::get(QString key) { + QString ret =descriptions.value(key,key); + return ret; +} + +void DescriptionMap::save() { + QFile file(filename); + file.open(QFile::WriteOnly); + QTextStream out(&file); + QMapIteratorit(descriptions); + while (it.hasNext()) { + it.next(); + QString line=QString("%1%2%3\n").arg(it.key()).arg(delimiter).arg(it.value()); + out <match(line); + if (match.hasMatch()) { + QString fileName = match.captured(1); + QString desc = match.captured(2); + add(fileName,desc); + } else { + DEBUGF QQ("MATCH ERROR",line); + } + } +} + + +//==================================================================================================== +//==================================================================================================== + + +#if 0 +Are you <b>absolutely sure</b> you want to proceed? + +QMessageBox msgBox; msgBox.setText(tr("Confirm?")); +QAbstractButton* pButtonYes = msgBox.addButton(tr("Yeah!"), QMessageBox::YesRole); +pButtonNo=msgBox.addButton(tr("Nope"), QMessageBox::NoRole); +btn.setIcon(const QIcon &icon); + +msgBox.exec(); + +if (msgBox.clickedButton()==pButtonYes) { + + +QIcon groupIcon( style()->standardIcon( QStyle::SP_DirClosedIcon ) ) +https://www.pythonguis.com/faq/built-in-qicons-pyqt/ + +QMessageBox msgBox; +msgBox.setText("The document has been modified."); +msgBox.setInformativeText("Do you want to save your changes?"); +msgBox.setStandardButtons(QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel); +msgBox.setDefaultButton(QMessageBox::Save); +int ret = msgBox.exec(); +switch (ret) { + case QMessageBox::Save: + // Save was clicked + break; + case QMessageBox::Discard: + // Don't Save was clicked + break; + case QMessageBox::Cancel: + // Cancel was clicked + break; + default: + // should never be reached + break; +} + + + +// Reminders For testing + + Different languages unicodes to test. optained from translation files + + 도움주신분들 + 已成功删除 + 删除 + הצג את מחיצת הנתונים + + הצג את מחיצת הנתונים 已成功删除 عذرا ، لا يمكن تحديد موقع ملف. + 已成功删除 عذرا ، لا يمكن تحديد موقع ملف. 删除 + Toon gegevensmap + عذرا ، لا يمكن تحديد موقع ملف. + + + menuDialog->connect(menuList, SIGNAL(itemActivated(QListWidgetItem*)), this, SLOT(itemActivated(QListWidgetItem*) )); + menuDialog->connect(menuList, SIGNAL(itemDoubleClicked(QListWidgetItem*)), this, SLOT(itemDoubleClicked(QListWidgetItem*) )); + menuDialog->connect(menuList, SIGNAL(itemClicked(QListWidgetItem*)), this, SLOT(itemClicked(QListWidgetItem*) )); + menuDialog->connect(menuList, SIGNAL(itemEntered(QListWidgetItem*)), this, SLOT(itemEntered(QListWidgetItem*) )); + menuDialog->connect(menuList, SIGNAL(itemPressed(QListWidgetItem*)), this, SLOT(itemEntered(QListWidgetItem*) )); + + + + menuDialog->disconnect(menuList, SIGNAL(itemActivated(QListWidgetItem*)), this, SLOT(itemActivated(QListWidgetItem*) )); + menuDialog->disconnect(menuList, SIGNAL(itemDoubleClicked(QListWidgetItem*)), this, SLOT(itemDoubleClicked(QListWidgetItem*) )); + menuDialog->disconnect(menuList, SIGNAL(itemClicked(QListWidgetItem*)), this, SLOT(itemClicked(QListWidgetItem*) )); + menuDialog->disconnect(menuList, SIGNAL(itemEntered(QListWidgetItem*)), this, SLOT(itemEntered(QListWidgetItem*) )); + menuDialog->disconnect(menuList, SIGNAL(itemPressed(QListWidgetItem*)), this, SLOT(itemEntered(QListWidgetItem*) )); + + +void SaveGraphLayoutSettings::itemActivated(QListWidgetItem *item) +{ + Q_UNUSED( item ); + DEBUGF Q( item->text() ); +} + +void SaveGraphLayoutSettings::itemDoubleClicked(QListWidgetItem *item) +{ + Q_UNUSED( item ); + DEBUGF Q( item->text() ); +} + +void SaveGraphLayoutSettings::itemClicked(QListWidgetItem *item) +{ + Q_UNUSED( item ); + DEBUGF Q( item->text() ); +} + +void SaveGraphLayoutSettings::itemEntered(QListWidgetItem *item) +{ + Q_UNUSED( item ); + DEBUGF Q( item->text() ); +} + +void SaveGraphLayoutSettings::itemPressed(QListWidgetItem *item) +{ + Q_UNUSED( item ); + DEBUGF Q( item->text() ); +} + +//private_slots: + void itemActivated(QListWidgetItem *item); + void itemDoubleClicked(QListWidgetItem *item); + void itemClicked(QListWidgetItem *item); + void itemEntered(QListWidgetItem *item); + void itemPressed(QListWidgetItem *item); + + + + + +#endif + diff --git a/oscar/saveGraphLayoutSettings.h b/oscar/saveGraphLayoutSettings.h new file mode 100644 index 00000000..d3e5e3be --- /dev/null +++ b/oscar/saveGraphLayoutSettings.h @@ -0,0 +1,178 @@ +/* Overview GUI Headers + * + * Copyright (c) 2019-2022 The OSCAR Team + * Copyright (C) 2011-2018 Mark Watkins + * + * 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. */ + +#ifndef SAVEGRAPHLAYOUTSETTINGS_H +#define SAVEGRAPHLAYOUTSETTINGS_H + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "Graphs/gGraphView.h" + +class DescriptionMap +{ +public: + DescriptionMap(QDir* dir, QString filename) ; + virtual ~DescriptionMap(); + void add(QString key,QString desc); + void remove(QString key); + QString get(QString key); + void load(); + void save(); +private: + QString filename; + QMap descriptions; + const QRegularExpression* parseDescriptionsRe; + const QChar delimiter = QChar(':'); +}; + +class SaveGraphLayoutSettings : public QWidget +{ + Q_OBJECT +public: + SaveGraphLayoutSettings(QString title, QWidget* parent) ; + ~SaveGraphLayoutSettings(); + void menu(gGraphView* graphView); +protected: + QIcon* m_icon_return = new QIcon(":/icons/return.png"); + QIcon* m_icon_help = new QIcon(":/icons/question_mark.png"); + QIcon* m_icon_exit = new QIcon(":/icons/exit.png"); + QIcon* m_icon_delete = new QIcon(":/icons/trash_can.png"); + QIcon* m_icon_update = new QIcon(":/icons/update.png"); + QIcon* m_icon_restore = new QIcon(":/icons/restore.png"); + QIcon* m_icon_rename = new QIcon(":/icons/rename.png"); + QIcon* m_icon_add = new QIcon(":/icons/plus.png"); + +private: + const static int fileNumMaxLength = 3; + const static int maxFiles = 30; // Max supported design limited is 1000 10**fileNumMaxLength(3). + const static int iconWidthMessageBox = 50; + const static int maxDescriptionLen = 80; + const QString fileBaseName = QString("layout"); + const int fileNameRole = Qt::UserRole; + int fontSizeIncrease = 0; + int horizontalWidthAdjustment=60; // this seem to make menu size changes work. Testing says it is 60 but what causes it is unknown. + + QSize minMenuListSize = QSize(0,0); + QSize minMenuDialogSize = QSize(0,0); + + QSize dialogListDiff = QSize(0,0); + QSize menuDialogSize = QSize(0,0); + QSize menuListSize = QSize(0,0); + void initminMenuListSize(); + QSize calculateMenuDialogSize(); + QSize maxSize(const QSize AA , const QSize BB ) ; + bool sizeEqual(const QSize AA , const QSize BB ) ; + + const QRegularExpression* singleLineRe; + const QRegularExpression* fileNumRe; + const QRegularExpression* parseFilenameRe; + + + QWidget* parent; + const QString title; + gGraphView* graphView = nullptr; + QFont menuListFont; + + QDialog* menuDialog; + QListWidget* menuList; + + QPushButton* menuAddFullBtn; // Must be first item for workaround. + QPushButton* menuAddBtn; + QPushButton* menuDeleteBtn; + QPushButton* menuRestoreBtn; + QPushButton* menuUpdateBtn; + QPushButton* menuRenameBtn; + QPushButton* menuExitBtn; + QPushButton* menuHelpBtn; + + + QVBoxLayout* menuLayout; + QHBoxLayout* menuLayoutButtons; + + void createHelp(); + void helpDestructor(); + QString helpInfo(); + QDialog* helpDialog=nullptr; + QPushButton* helpInfoExitBtn=nullptr; + QPushButton* helpExitBtn=nullptr; + int helpFontSizeIncrease = 0; + QHBoxLayout* helpLayoutButtons = nullptr; + + + + QDir* dir = nullptr; + QString dirName; + int nextNumToUse; + QListWidgetItem* updateFileList(QString find = QString()); + QListWidgetItem* widestItem=nullptr; + QString styleOn; + QString styleOff; + QString styleExit; + QString styleMessageBox; + QString styleDialog; + + QString calculateButtonStyle(bool on,bool border); + void looksOn(QPushButton* button,bool on); + DescriptionMap* descriptionMap; + bool confirmAction(QString name,QString question,QIcon* icon, + QMessageBox::StandardButtons flags = (QMessageBox::Cancel|QMessageBox::Yes) , + QMessageBox::StandardButton adefault = QMessageBox::Cancel, + QMessageBox::StandardButton success = QMessageBox::Yes + ); + bool verifyItem(QListWidgetItem* item,QString name,QIcon* icon) ; + + const QString calculateStyleMessageBox( QFont* font, QString& s1, QString& s2); + + void displaywidgets(QWidget* widget); + QSize calculateParagraphSize(QString& text,QFont& font, QString& ); + + void createMenu(); + void createStyleSheets(); + void createSaveFolder(); + QPushButton* newBtnRtn(QHBoxLayout*, QString name, QIcon* icon, QString style,QSizePolicy::Policy hPolicy,QString tooltip); + QPushButton* menuBtn( QString name, QIcon* icon, QString style,QSizePolicy::Policy hPolicy,QString tooltip); + QPushButton* helpBtn( QString name, QIcon* icon, QString style,QSizePolicy::Policy hPolicy,QString tooltip); + + void manageButtonApperance(); + void resizeMenu(); + + int fileNum(QString fileName); + void writeSettings(QString filename); + void loadSettings(QString filename); + void deleteSettings(QString filename); + +public slots: +private slots: + void add_feature(); + void addFull_feature(); + void restore_feature(); + void rename_feature(); + void update_feature(); + void help_feature(); + void help_off_feature(); + void help_exit_feature(); + void delete_feature(); + void exit(); + + void itemChanged(QListWidgetItem *item); + void itemSelectionChanged(); +}; + + +#endif // SAVEGRAPHLAYOUTSETTINGS_H + diff --git a/oscar/test_macros.h b/oscar/test_macros.h index 14e6439d..b6d4f00e 100644 --- a/oscar/test_macros.h +++ b/oscar/test_macros.h @@ -17,10 +17,10 @@ When only these macos are used then debugging is disabled. SO for production code rename TEST_MACROS_ENABLED to TEST_MACROS_ENABLEDoff ########################################### -The the following to source cpp files +The the following to source cpp files to turn on the debug macros for use. -#define TEST_MACROS_ENABLED +#define TEST_MACROS_ENABLED #include To turn off the the test macros. @@ -38,11 +38,13 @@ To turn off the the test macros. #include #include -#define DEBUGQ qDebug().noquote() -#define DEBUGL DEBUGQ < - - PreferencesDialog - - - Qt::ApplicationModal - - - - 0 - 0 - 942 - 651 - - - - - 0 - 0 - - - - - 145 - 0 - - - - Preferences - - - - :/icons/preferences.png:/icons/preferences.png - - - true - - - true - - - - - - - 0 - - - 0 - - - 0 - - - 0 - - - - - - 0 - 0 - - - - 0 - - - - &Import - - - - 4 - - - 4 - - - 4 - - - 4 - - - 4 - - - - - - 0 - 0 - - - - Session Splitting Settings - - - - 4 - - - 4 - - - 4 - - - 4 - - - 4 - - - - - - 0 - 0 - - - - [ResMed warning message] - - - true - - - true - - - - - - - - 0 - 0 - - - - - 0 - - - 0 - - - 0 - - - 0 - - - - - - - Combine Close Sessions - - - Qt::AlignLeading|Qt::AlignLeft|Qt::AlignVCenter - - - - - - - Minutes - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - - - - 0 - 0 - - - - Multiple sessions closer together than this value will be kept on the same day. - - - - 0 - - - 240 - - - 10 - - - 30 - - - 0 - - - Qt::Horizontal - - - false - - - false - - - QSlider::TicksAbove - - - 10 - - - - - - - QFrame::Box - - - 5 - - - QLCDNumber::Flat - - - - - - - - - - - Ignore Short Sessions - - - Qt::AlignLeading|Qt::AlignLeft|Qt::AlignVCenter - - - - - - - Minutes - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - 4 - - - - - <!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:'Cantarell'; font-size:11pt; font-weight:400; font-style:normal;"> -<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Sessions shorter in duration than this will not be displayed<span style=" font-style:italic;">.</span></p> -<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-style:italic;"></p></body></html> - - - 90 - - - 5 - - - Qt::Horizontal - - - QSlider::TicksAbove - - - 5 - - - - - - - QFrame::Box - - - QLCDNumber::Flat - - - - - - - - - 4 - - - - - Day Split Time - - - Qt::AlignLeading|Qt::AlignLeft|Qt::AlignVCenter - - - - - - - Sessions starting before this time will go to the previous calendar day. - - - QAbstractSpinBox::UpDownArrows - - - - 12 - 0 - 0 - 2000 - 1 - 1 - - - - - - - - Qt::Horizontal - - - QSizePolicy::Minimum - - - - 10 - 20 - - - - - - - - <html><head/><body><p><span style=" font-weight:600;">This setting should be used with caution...</span> Switching it off comes with consequences involving accuracy of summary only days, as certain calculations only work properly provided summary only sessions that came from individual day records are kept together. </p><p><span style=" font-weight:600;">ResMed users:</span> Just because it seems natural to you and I that the 12 noon session restart should be in the previous day, does not mean ResMed's data agrees with us. The STF.edf summary index format has serious weaknesses that make doing this not a good idea.</p><p>This option exists to pacify those who don't care and want to see this &quot;fixed&quot; no matter the costs, but know it comes with a cost. If you keep your SD card in every night, and import at least once a week, you won't see problems with this very often.</p></body></html> - - - Don't Split Summary Days (Warning: read the tooltip!) - - - - - - - Qt::Horizontal - - - - 40 - 20 - - - - - - - - - - - - - - - - - 0 - 0 - - - - Session Storage Options - - - - 4 - - - 9 - - - 4 - - - 0 - - - 0 - - - 4 - - - - - - true - - - - Changing SD Backup compression options doesn't automatically recompress backup data. - - - true - - - - - - - Compress ResMed (EDF) backups to save disk space. -Backed up EDF files are stored in the .gz format, -which is common on Mac & Linux platforms.. - -OSCAR can import from this compressed backup directory natively.. -To use it with ResScan will require the .gz files to be uncompressed first.. - - - Compress SD Card Backups (slower first import, but makes backups smaller) - - - - - - - - true - - - - The following options affect the amount of disk space OSCAR uses, and have an effect on how long import takes. - - - true - - - - - - - This makes OSCAR's data take around half as much space. -But it makes import and day changing take longer.. -If you've got a new computer with a small solid state disk, this is a good option. - - - Compress Session Data (makes OSCAR data smaller, but day changing slower.) - - - - - - - This maintains a backup of SD-card data for ResMed machines, - -ResMed S9 series machines delete high resolution data older than 7 days, -and graph data older than 30 days.. - -OSCAR can keep a copy of this data if you ever need to reinstall. -(Highly recomended, unless your short on disk space or don't care about the graph data) - - - Create SD Card Backups during Import (Turn this off at your own peril!) - - - - - - - 0 - - - 0 - - - 0 - - - - - Do not import sessions older than: - - - - - - - - 165 - 0 - - - - Sessions older than this date will not be imported - - - true - - - true - - - - 2099 - 12 - 31 - - - - - 1970 - 1 - 2 - - - - QDateTimeEdit::DaySection - - - dd MMMM yyyy - - - true - - - - - - - Qt::Horizontal - - - - 40 - 20 - - - - - - - - - - - - - - 0 - 0 - - - - Memory and Startup Options - - - - - - Bypass the login screen and load the most recent User Profile - - - Auto-Launch CPAP Importer after opening profile - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - <html><head/><body><p>This setting keeps waveform and event data in memory after use to speed up revisiting days.</p><p>This is not really a necessary option, as your operating system caches previously used files too.</p><p>Recommendation is to leave it switched off, unless your computer has a ton of memory.</p></body></html> - - - Keep Waveform/Event data in memory - - - - - - - <html><head/><body><p>Makes starting OSCAR a bit slower, by pre-loading all the summary data in advance, which speeds up overview browsing and a few other calculations later on. </p><p>If you have a large amount of data, it might be worth keeping this switched off, but if you typically like to view <span style=" font-style:italic;">everything</span> in overview, all the summary data still has to be loaded anyway. </p><p>Note this setting doesn't affect waveform and event data, which is always demand loaded as needed.</p></body></html> - - - Pre-Load all summary data at startup - - - - - - - Automatically load last used profile on start-up - - - - - - - <html><head/><body><p>Cuts down on any unimportant confirmation dialogs during import.</p></body></html> - - - Import without asking for confirmation - - - - - - - <html><head/><body><p>Provide an alert when importing data from any machine model that has not yet been tested by OSCAR developers.</p></body></html> - - - Warn when importing data from an untested machine - - - - - - - <html><head/><body><p>Provide an alert when importing data that is somehow different from anything previously seen by OSCAR developers.</p></body></html> - - - Warn when previously unseen data is encountered - - - - - - - - - - - &CPAP - - - - 4 - - - 4 - - - 4 - - - 4 - - - 4 - - - - - 20 - - - QLayout::SetMinimumSize - - - 5 - - - 5 - - - 5 - - - 5 - - - - - - 0 - 0 - - - - CPAP Clock Drift - - - - 4 - - - 4 - - - 4 - - - 4 - - - 4 - - - - - - 0 - 0 - - - - <html><head/><body><p>Note: This is not intended for timezone corrections! Make sure your operating system clock and timezone is set correctly.</p></body></html> - - - true - - - - - - - -9999 - - - 9999 - - - - - - - Minutes - - - - - - - -59 - - - 59 - - - 0 - - - - - - - -99 - - - - - - - Hours - - - - - - - Seconds - - - - - - - - - - - 0 - 0 - - - - This calculation requires Total Leaks data to be provided by the CPAP machine. (Eg, PRS1, but not ResMed, which has these already) - -The Unintentional Leak calculations used here are linear, they don't model the mask vent curve. - -If you use a few different masks, pick average values instead. It should still be close enough. - - - Calculate Unintentional Leaks When Not Present - - - true - - - - - - Your masks vent rate at 20 cmH2O pressure - - - 300 - - - 550 - - - 10 - - - 470 - - - Qt::Horizontal - - - QSlider::TicksAbove - - - 20 - - - - - - - - 0 - 0 - - - - - 80 - 0 - - - - QFrame::Box - - - 48 l/m - - - true - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - Your masks vent rate at 4 cmH2O pressure - - - 120 - - - 240 - - - 1 - - - 10 - - - 201 - - - Qt::Horizontal - - - false - - - QSlider::TicksAbove - - - - - - - 4 cmH2O - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - 20 cmH2O - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - 80 - 0 - - - - QFrame::Box - - - 20 l/m - - - true - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - Note: A linear calculation method is used. Changing these values requires a recalculation. - - - true - - - - - - - - - - - 0 - 0 - - - - Enable/disable experimental event flagging enhancements. -It allows detecting borderline events, and some the machine missed. -This option must be enabled before import, otherwise a purge is required. - - - Custom CPAP User Event Flagging - - - false - - - true - - - - 4 - - - 9 - - - 4 - - - 4 - - - 4 - - - - - s - - - 10.000000000000000 - - - - - - - % - - - 10.000000000000000 - - - - - - - This experimental option attempts to use OSCAR's event flagging system to improve machine detected event positioning. - - - Resync Machine Detected Events (Experimental) - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - #2 - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - 0 - 0 - - - - Flow Restriction - - - - - - - - 0 - 0 - - - - - true - - - - <!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:'Sans'; font-size:10pt; font-weight:400; font-style:italic;"> -<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Custom flagging is an experimental method of detecting events missed by the machine. They are <span style=" text-decoration: underline;">not</span> included in AHI.</p></body></html> - - - Qt::AlignLeading|Qt::AlignLeft|Qt::AlignTop - - - true - - - - - - - - 0 - 0 - - - - Percentage of restriction in airflow from the median value. -A value of 20% works well for detecting apneas. - - - % - - - 10.000000000000000 - - - - - - - Show in Event Breakdown Piechart - - - - - - - Duration of airflow restriction - - - s - - - 1.000000000000000 - - - 10.000000000000000 - - - - - - - #1 - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - Allow duplicates near machine events. - - - - - - - - 0 - 0 - - - - Event Duration - - - - - - - - - - - - QLayout::SetMinimumSize - - - 5 - - - 5 - - - - - - 0 - 0 - - - - General CPAP and Related Settings - - - - - - Show flags for machine detected events that haven't been identified yet. - - - Enable Unknown Events Channels - - - - - - - AHI/Hour Graph Time Window - - - - - - - Compliance defined as - - - - - - - - 0 - 0 - - - - Adjusts the amount of data considered for each point in the AHI/Hour graph. -Defaults to 60 minutes.. Highly recommend it's left at this value. - - - minutes - - - 5 - - - 999 - - - 60 - - - - - - - - 50 - false - - - - Whether to show the leak redline in the leak graph - - - Flag leaks over threshold - - - - - - - Preferred major event index - - - - - - - Reset the counter to zero at beginning of each (time) window. - - - Zero Reset - - - - - - - - 0 - 0 - - - - User definable threshold considered large leak - - - L/min - - - 1 - - - - - - - - 0 - 0 - - - - Regard days with under this usage as "incompliant". 4 hours is usually considered compliant. - - - hours - - - 1 - - - 8.000000000000000 - - - 4.000000000000000 - - - - - - - - 0 - 0 - - - - - AHI - - - - - RDI - - - - - - - - - - - - 0 - 0 - - - - Changes to the following settings needs a restart, but not a recalc. - - - Preferred Calculation Methods - - - - - - For consistancy, ResMed users should use 95% here, -as this is the only value available on summary-only days. - - - % - - - 1 - - - 99 - - - - - - - - 0 - 0 - - - - Middle Calculations - - - - - - - Maximum Calcs - - - - - - - Upper Percentile - - - - - - - Culminative Indices - - - - - - - Median is recommended for ResMed users. - - - - Median - - - - - Weighted Average - - - - - Normal Average - - - - - - - - - 0 - 0 - - - - - 140 - 0 - - - - <html><head/><body><p>True maximum is the maximum of the data set.</p><p>99th percentile filters out the rarest outliers.</p></body></html> - - - - True Maximum - - - - - 99% Percentile - - - - - - - - - Combined Count divided by Total Hours - - - - - Time Weighted average of Indice - - - - - Standard average of indice - - - - - Median - - - - - - - - - 0 - 0 - - - - <html><head/><body><p><span style=" font-weight:600;">Note: </span>Due to summary design limitations, ResMed machines do not support changing these settings.</p></body></html> - - - Qt::AlignLeading|Qt::AlignLeft|Qt::AlignTop - - - true - - - - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - - - - &Oximetry - - - - 4 - - - 2 - - - 2 - - - 2 - - - 2 - - - - - Oximetry Settings - - - false - - - - 4 - - - 8 - - - 4 - - - 4 - - - 5 - - - - - - 0 - 0 - - - - Other oximetry options - - - - - - % - - - - - - - bpm - - - - - - - Small chunks of oximetry data under this amount will be discarded. - - - s - - - 300 - - - - - - - Discard segments under - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - bpm - - - - - - - Flag SPO2 Desaturations Below - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - Flag Pulse Rate Below - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - Flag Pulse Rate Above - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - - - 0 - 0 - - - - Flag rapid changes in oximetry stats - - - - - - - 0 - 0 - - - - SPO2 - - - - - - - Minimum duration of drop in oxygen saturation - - - s - - - 0 - - - - - - - Sudden change in Pulse Rate of at least this amount - - - bpm - - - 0 - - - 1.000000000000000 - - - - - - - Minimum duration of pulse change event. - - - s - - - 0 - - - - - - - - 0 - 0 - - - - Pulse - - - - - - - Percentage drop in oxygen saturation - - - % - - - 0 - - - 1.000000000000000 - - - - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - - - - - 0 - 0 - - - - - 300 - 0 - - - - <!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=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Sans'; font-size:10pt; font-weight:600;">Syncing Oximetry and CPAP Data</span></p> -<p align="justify" 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-family:'Sans'; font-size:10pt;"><br /></p> -<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Sans'; font-size:10pt;">CMS50 data imported from SpO2Review (from .spoR files) or the serial import method does </span><span style=" font-family:'Sans'; font-size:10pt; font-weight:600; text-decoration: underline;">not</span><span style=" font-family:'Sans'; font-size:10pt;"> have the correct timestamp needed to sync.</span></p> -<p align="justify" 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-family:'Sans'; font-size:10pt;"><br /></p> -<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Sans'; font-size:10pt;">Live view mode (using a serial cable) is one way to acheive an accurate sync on CMS50 oximeters, but does not counter for CPAP clock drift.</span></p> -<p align="justify" 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-family:'Sans'; font-size:10pt;"><br /></p> -<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Sans'; font-size:10pt;">If you start your Oximeters recording mode at </span><span style=" font-family:'Sans'; font-size:10pt; font-style:italic;">exactly </span><span style=" font-family:'Sans'; font-size:10pt;">the same time you start your CPAP machine, you can now also achieve sync. </span></p> -<p align="justify" 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-family:'Sans'; font-size:10pt;"><br /></p> -<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Sans'; font-size:10pt;">The serial import process takes the starting time from last nights first CPAP session. (Remember to import your CPAP data first!)</span></p></body></html> - - - - - - - - Events - - - - 4 - - - 0 - - - 0 - - - 0 - - - 0 - - - - - - - Search - - - - - - - - - - - - - 0 - 0 - - - - - - - - - - - 0 - 0 - - - - Reset &Defaults - - - - - - - - 0 - 0 - - - - <html><head/><body><p><span style=" font-weight:600;">Warning: </span>Just because you can, does not mean it's good practice.</p></body></html> - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - - Waveforms - - - - 4 - - - 0 - - - 0 - - - 0 - - - 0 - - - - - - - Search - - - - - - - - - - - - - 0 - 0 - - - - - - - - - - - 0 - 0 - - - - Reset &Defaults - - - - - - - - 0 - 0 - - - - <html><head/><body><p><span style=" font-weight:600;">Warning: </span>Just because you can, does not mean it's good practice.</p></body></html> - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - - &General - - - - - - - - - 0 - 0 - - - - General Settings - - - - 4 - - - 4 - - - 4 - - - 4 - - - 4 - - - - - Allow use of multiple CPU cores where available to improve performance. -Mainly affects the importer. - - - Enable Multithreading - - - - - - - Show Remove Card reminder notification on OSCAR shutdown - - - - - - - Always save screenshots in the OSCAR Data folder - - - - - - - - - - Qt::Vertical - - - - - - - - - Qt::Horizontal - - - - - - - Graphics Engine (Requires Restart) - - - - - - Try changing this from the default setting (Desktop OpenGL) if you experience rendering problems with OSCAR's graphs. - - - true - - - - - - - - - - - - - - 0 - 0 - - - - Check For Updates - - - false - - - - - - - 0 - 0 - - - - You are using a test version of OSCAR. Test versions check for updates automatically at least once every seven days. You may set the interval to less than seven days. - - - true - - - - - - - - 50 - false - false - false - - - - Automatically check for updates - - - - - - - - - - 0 - 0 - - - - Check for new version every - - - - - - - How often OSCAR should check for updates. - - - 90 - - - - - - - - 0 - 0 - - - - days. - - - - - - - Qt::Horizontal - - - - 40 - 20 - - - - - - - - - - - - - 0 - 0 - - - - Last Checked For Updates: - - - - - - - - 0 - 0 - - - - TextLabel - - - - - - - Qt::Horizontal - - - - 40 - 20 - - - - - - - - - - If you are interested in helping test new features and bugfixes early, click here. - - - I want to try experimental and test builds. (Advanced users only please.) - - - - - - - - 0 - 0 - - - - If you would like to help test early versions of OSCAR, please see the Wiki page about testing OSCAR. We welcome everyone who would like to test OSCAR, help develop OSCAR, and help with translations to existing or new languages. https://www.sleepfiles.com/OSCAR - - - true - - - true - - - - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - - &Appearance - - - - 4 - - - 2 - - - 2 - - - 2 - - - 2 - - - - - - - - 0 - 0 - - - - Graph Settings - - - - - - On Opening - - - Qt::AlignLeading|Qt::AlignLeft|Qt::AlignVCenter - - - - - - - <html><head/><body><p>Which tab to open on loading a profile. (Note: It will default to Profile if OSCAR is set to not open a profile on startup)</p></body></html> - - - Profile - - - - Profile - - - - - Welcome - - - - - Daily - - - - - Overview - - - - - Statistics - - - - - - - - Switch Tabs - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - No change - - - - - Welcome - - - - - Daily - - - - - Overview - - - - - Statistics - - - - - - - - After Import - - - Qt::AlignLeading|Qt::AlignLeft|Qt::AlignVCenter - - - - - - - Graph Height - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - 0 - 0 - - - - Overlay Flags - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - Line Thickness - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - 0 - 0 - - - - The visual method of displaying waveform overlay flags. - - - - - Standard Bars - - - - - Top Markers - - - - - - - - - - The pixel thickness of line plots - - - 2 - - - 8 - - - 1 - - - Qt::Horizontal - - - QSlider::TicksBelow - - - - - - - - - Tooltip Timeout - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - Scroll Dampening - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - How long you want the tooltips to stay visible. - - - 1000 - - - 30000 - - - 100 - - - 500 - - - 5000 - - - Qt::Horizontal - - - QSlider::TicksBelow - - - 1000 - - - - - - - - 0 - 0 - - - - - 60 - 0 - - - - QFrame::Box - - - TextLabel - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - Graph Tooltips - - - - - - - - Bar Tops - - - - - Line Chart - - - - - - - - - 0 - 0 - - - - Default display height of graphs in pixels - - - 50 - - - 600 - - - 10 - - - 180 - - - - - - - - - <html><head/><body><p>This makes scrolling when zoomed in easier on sensitive bidirectional TouchPads</p><p>50ms is recommended value.</p></body></html> - - - 25 - - - 1 - - - 5 - - - Qt::Horizontal - - - false - - - QSlider::TicksBelow - - - 1 - - - - - - - - 0 - 0 - - - - - 60 - 0 - - - - QFrame::Box - - - TextLabel - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - Overview Linecharts - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - - - - 0 - 0 - - - - Other Visual Settings - - - - - - Anti-Aliasing applies smoothing to graph plots.. -Certain plots look more attractive with this on. -This also affects printed reports. - -Try it and see if you like it. - - - Use Anti-Aliasing - - - - - - - Makes certain plots look more "square waved". - - - Square Wave Plots - - - - - - - Pixmap caching is an graphics acceleration technique. May cause problems with font drawing in graph display area on your platform. - - - Use Pixmap Caching - - - - - - - <html><head/><body><p>These features have recently been pruned. They will come back later. </p></body></html> - - - Animations && Fancy Stuff - - - - - - - Daily view navigation buttons will skip over days without data records - - - Skip over Empty Days - - - - - - - Whether to allow changing yAxis scales by double clicking on yAxis labels - - - Allow YAxis Scaling - - - - - - - Whether to include machine serial number on machine settings changes report - - - Include Serial Number - - - - - - - Print reports in black and white, which can be more legible on non-color printers - - - Print reports in black and white (monochrome) - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - - - - - - Qt::Horizontal - - - - - - - Fonts (Application wide settings) - - - false - - - - 0 - - - 4 - - - 0 - - - 4 - - - - - - - - 0 - 0 - - - - - 75 - true - - - - Font - - - - - - - - 0 - 0 - - - - - 75 - true - - - - Size - - - - - - - - 0 - 0 - - - - - 75 - true - - - - Bold - - - Qt::AlignLeading|Qt::AlignLeft|Qt::AlignVCenter - - - - - - - - 0 - 0 - - - - - 75 - true - - - - Italic - - - - - - - - 0 - 0 - - - - Application - - - - - - - - - - - 80 - 16777215 - - - - 6 - - - 30 - - - 10 - - - - - - - - 0 - 0 - - - - Qt::LeftToRight - - - - - - - - - - - 0 - 0 - - - - - - - - - - - - 0 - 0 - - - - Graph Text - - - - - - - - - - - 80 - 16777215 - - - - 6 - - - 40 - - - 11 - - - - - - - - 0 - 0 - - - - - - - - - - - - 0 - 0 - - - - - - - - - - - - 0 - 0 - - - - Graph Titles - - - - - - - - - - - 80 - 16777215 - - - - 6 - - - 40 - - - 14 - - - - - - - - 0 - 0 - - - - - - - - - - - - 0 - 0 - - - - - - - - - - - - 0 - 0 - - - - Big Text - - - - - - - - - - - 80 - 16777215 - - - - 6 - - - 72 - - - 18 - - - - - - - - 0 - 0 - - - - - - - - - - - - 0 - 0 - - - - - - - - - - - - 0 - 0 - - - - - 75 - true - - - - Details - - - - - - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - - - - - - - - &Cancel - - - - - - - &Ok - - - - - - - Qt::Horizontal - - - - 750 - 20 - - - - - - - - - - - - cancelButton - clicked() - PreferencesDialog - reject() - - - 757 - 605 - - - 286 - 274 - - - - -