From 8617b1634cd082f6c6f75bd68f3f92c98da6b5eb Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Thu, 31 Jul 2014 06:25:06 +1000 Subject: [PATCH] Resmed Session splitting improvements and preference --- .../SleepLib/loader_plugins/cms50_loader.h | 2 +- .../SleepLib/loader_plugins/icon_loader.cpp | 8 ++-- .../loader_plugins/intellipap_loader.cpp | 2 +- .../SleepLib/loader_plugins/prs1_loader.cpp | 40 +++++++++++-------- .../SleepLib/loader_plugins/resmed_loader.cpp | 32 +++++++-------- sleepyhead/SleepLib/machine.cpp | 26 ++++++++++-- sleepyhead/SleepLib/machine_loader.cpp | 14 +++++++ sleepyhead/SleepLib/machine_loader.h | 11 +++++ sleepyhead/SleepLib/profiles.h | 6 +++ sleepyhead/daily.cpp | 20 +++++----- sleepyhead/docs/release_notes.html | 5 ++- sleepyhead/mainwindow.cpp | 28 ++++++------- sleepyhead/preferencesdialog.cpp | 31 ++++++++------ sleepyhead/preferencesdialog.ui | 10 +++++ 14 files changed, 154 insertions(+), 81 deletions(-) diff --git a/sleepyhead/SleepLib/loader_plugins/cms50_loader.h b/sleepyhead/SleepLib/loader_plugins/cms50_loader.h index c8ef7e9f..6ab5d34b 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50_loader.h +++ b/sleepyhead/SleepLib/loader_plugins/cms50_loader.h @@ -39,7 +39,7 @@ Q_OBJECT virtual const QString &loaderName() { return cms50_class_name; } virtual MachineInfo newInfo() { - return MachineInfo(MT_OXIMETER, cms50_class_name, QObject::tr("Contec"), QString(), QString(), QString(), QObject::tr("CMS50"), QDateTime::currentDateTime(), cms50_data_version); + return MachineInfo(MT_OXIMETER, cms50_class_name, QObject::tr("Contec"), QObject::tr("CMS50"), QString(), QString(), QObject::tr("CMS50"), QDateTime::currentDateTime(), cms50_data_version); } diff --git a/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp b/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp index cd1fe78a..bf0a8680 100644 --- a/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp @@ -293,6 +293,8 @@ int FPIconLoader::OpenMachine(Machine *mach, QString &path) int c = Sessions.size(); mach->Save(); + finishAddingSessions(); + return c; } @@ -595,7 +597,7 @@ bool FPIconLoader::OpenFLW(Machine *mach, QString filename) } if (newsess) { - mach->AddSession(sess); + addSession(sess); } if (p_profile->session->backupCardData()) { @@ -748,7 +750,7 @@ bool FPIconLoader::OpenSummary(Machine *mach, QString filename) sess->settings[CPAP_HumidSetting] = x2; //sess->settings[CPAP_PresReliefType]=PR_SENSAWAKE; Sessions[ts] = sess; - mach->AddSession(sess); + addSession(sess); } } while (!in.atEnd()); @@ -913,7 +915,7 @@ bool FPIconLoader::OpenDetail(Machine *mach, QString filename) // sess->really_set_last(ti-360000L); // sess->SetChanged(true); - // mach->AddSession(sess,profile); + // addSession(sess,profile); } if (p_profile->session->backupCardData()) { unsigned char *data = (unsigned char *)index.data(); diff --git a/sleepyhead/SleepLib/loader_plugins/intellipap_loader.cpp b/sleepyhead/SleepLib/loader_plugins/intellipap_loader.cpp index 4697c00e..f2fd9e30 100644 --- a/sleepyhead/SleepLib/loader_plugins/intellipap_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/intellipap_loader.cpp @@ -518,7 +518,7 @@ int IntellipapLoader::Open(QString path) quint64 last = qint64(SessionEnd[i]) * 1000L; if (sess->last() > 0) { - sess->really_set_last(last); + // sess->really_set_last(last); sess->settings[CPAP_PresReliefType] = (PRTypes)PR_SMARTFLEX; diff --git a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp index 3b593d0b..18aed646 100644 --- a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp @@ -282,6 +282,9 @@ bool PRS1Loader::ParseProperties(Machine *m, QString filename) QChar sep = '='; QString key, value; + MachineInfo info = newInfo(); + bool ok; + while (!f.atEnd()) { key = s.section(sep, 0, 0); @@ -291,23 +294,29 @@ bool PRS1Loader::ParseProperties(Machine *m, QString filename) if (value == s) { continue; } - prop[key] = value; + if (key.toLower() == "serialnumber") { + info.serial = value; + } else if (key.toLower() == "modelnumber") { + info.modelnumber = value; + } else { + if (key.toLower() == "producttype") { + int i = value.toInt(&ok, 16); + + if (ok) { + if (ModelMap.find(i) != ModelMap.end()) { + info.model = ModelMap[i]; + } + } + } + prop[key] = value; + } s = f.readLine(); } - bool ok; - QString pt = prop["ProductType"]; - int i = pt.toInt(&ok, 16); - - if (ok) { - if (ModelMap.find(i) != ModelMap.end()) { - m->setModel(ModelMap[i]); - } + if (info.serial != m->serial()) { + qDebug() << "Serial Number in PRS1 properties.txt doesn't match machine record"; } - - if (prop["SerialNumber"] != m->serial()) { - qDebug() << "Serial Number in PRS1 properties.txt doesn't match directory structure"; - } else { prop.erase(prop.find("SerialNumber")); } // already got it stored. + m->setInfo(info); for (QHash::iterator i = prop.begin(); i != prop.end(); i++) { m->properties[i.key()] = i.value(); @@ -466,6 +475,7 @@ int PRS1Loader::OpenMachine(Machine *m, QString path) int tasks = countTasks(); runTasks(p_profile->session->multithreading()); + finishAddingSessions(); return tasks; } @@ -1423,9 +1433,7 @@ void PRS1Import::run() } sg->session->SetChanged(true); - loader->sessionMutex.lock(); - mach->AddSession(sg->session); - loader->sessionMutex.unlock(); + loader->addSession(sg->session); // Update indexes, process waveform and perform flagging sg->session->UpdateSummaries(); diff --git a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp index 0f157b24..665e0d73 100644 --- a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp @@ -731,14 +731,7 @@ void ResmedImport::run() // Ignore all the rest of the sumary data, because there is enough available to calculate it with higher accuracy. if (sess->length() > 0) { - loader->saveMutex.lock(); - - if (!mach->AddSession(sess)) { - delete sess; - loader->saveMutex.unlock(); - return; - } - loader->saveMutex.unlock(); + loader->addSession(sess); } else { delete sess; return; @@ -862,8 +855,8 @@ void ResmedImportStage2::run() } } + loader->addSession(sess); loader->saveMutex.lock(); - mach->AddSession(sess); sess->Store(mach->getDataPath()); loader->saveMutex.unlock(); } @@ -922,8 +915,8 @@ MachineInfo ResmedLoader::PeekInfo(const QString & path) } else if (key == "PNA") { // Product Name value.replace("_"," "); + value.replace("S9 ", ""); info.model = value; - } else if (key == "PCD") { // Product Code info.modelnumber = value; } @@ -1340,8 +1333,16 @@ int ResmedLoader::Open(QString path) continue; } else if (key == "PNA") { // Product Name + value.replace("S9", ""); value.replace("_"," "); - info.model = value; + value.replace("(",""); + value.replace(")",""); + if (value.contains("Adapt", Qt::CaseInsensitive)) { + if (!value.contains("VPAP")) { + value.replace("Adapt", QObject::tr("VPAP Adapt")); + } + } + info.model = value.trimmed(); continue; } else if (key == "PCD") { // Product Code @@ -1511,13 +1512,6 @@ int ResmedLoader::Open(QString path) // Scan DATALOG files, sort, and import any new sessions /////////////////////////////////////////////////////////////////////////////////// -#ifdef LOCK_RESMED_SESSIONS - // Have to sacrifice these features to get access to summary data. - p_profile->session->setCombineCloseSessions(0); - p_profile->session->setDaySplitTime(QTime(12,0,0)); - p_profile->session->setIgnoreShortSessions(false); -#endif - int new_sessions = scanFiles(m, newpath); //////////////////////////////////////////////////////////////////////////////////// @@ -1578,6 +1572,8 @@ int ResmedLoader::Open(QString path) new_sessions += countTasks(); runTasks(); + finishAddingSessions(); + #ifdef DEBUG_EFFICIENCY { qint64 totalbytes = 0; diff --git a/sleepyhead/SleepLib/machine.cpp b/sleepyhead/SleepLib/machine.cpp index 01d066f9..6f51623e 100644 --- a/sleepyhead/SleepLib/machine.cpp +++ b/sleepyhead/SleepLib/machine.cpp @@ -116,11 +116,19 @@ bool Machine::AddSession(Session *s) highest_sessionid = s->session(); } - QTime split_time = p_profile->session->daySplitTime(); - int combine_sessions = p_profile->session->combineCloseSessions(); - int ignore_sessions = p_profile->session->ignoreShortSessions(); + QTime split_time; + int combine_sessions; + bool locksessions = p_profile->session->lockSummarySessions(); - // ResMed machines can't do this.. but don't really want to do a slow string compare here + if (locksessions) { + split_time = s->summaryOnly() ? QTime(12,0,0) : p_profile->session->daySplitTime(); + combine_sessions = s->summaryOnly() ? 0 : p_profile->session->combineCloseSessions(); + } else { + split_time = p_profile->session->daySplitTime(); + combine_sessions = p_profile->session->combineCloseSessions(); + } + + int ignore_sessions = p_profile->session->ignoreShortSessions(); int session_length = s->last() - s->first(); session_length /= 60000; @@ -139,6 +147,9 @@ bool Machine::AddSession(Session *s) bool combine_next_day = false; int closest_session = 0; + + // Multithreaded import screws this up. :( + if (time < split_time) { date = date.addDays(-1); } else if (combine_sessions > 0) { @@ -150,6 +161,12 @@ bool Machine::AddSession(Session *s) if (closest_session < combine_sessions) { date = date.addDays(-1); + } else { + if ((split_time < time) && (split_time.secsTo(time) < 2)) { + if (s->machine()->loaderName() == STR_MACH_ResMed) { + date = date.addDays(-1); + } + } } } else { nextday = day.find(date.addDays(1)); // Check Day Afterwards @@ -200,6 +217,7 @@ bool Machine::AddSession(Session *s) if (combine_next_day) { for (QList::iterator i = nextday.value()->begin(); i != nextday.value()->end(); i++) { // i may need to do something here + if (locksessions && (*i)->summaryOnly()) continue; // can't move summary only sessions.. unlinkSession(*i); // Add it back diff --git a/sleepyhead/SleepLib/machine_loader.cpp b/sleepyhead/SleepLib/machine_loader.cpp index 7bfa52bf..81851984 100644 --- a/sleepyhead/SleepLib/machine_loader.cpp +++ b/sleepyhead/SleepLib/machine_loader.cpp @@ -127,6 +127,20 @@ MachineLoader::~MachineLoader() } } +void MachineLoader::finishAddingSessions() +{ + QMap::iterator it; + QMap::iterator it_end = new_sessions.end(); + + // Using a map specifically so they are inserted in order. + for (it = new_sessions.begin(); it != it_end; ++it) { + Session * sess = it.value(); + Machine * mach = sess->machine(); + mach->AddSession(sess); + } + new_sessions.clear(); +} + bool compressFile(QString inpath, QString outpath) { if (outpath.isEmpty()) { diff --git a/sleepyhead/SleepLib/machine_loader.h b/sleepyhead/SleepLib/machine_loader.h index e3d3a3c0..d62535d1 100644 --- a/sleepyhead/SleepLib/machine_loader.h +++ b/sleepyhead/SleepLib/machine_loader.h @@ -70,6 +70,14 @@ class MachineLoader: public QObject void queTask(ImportTask * task); + void addSession(Session * sess) + { + sessionMutex.lock(); + new_sessions[sess->session()] = sess; + sessionMutex.unlock(); + } + + //! \brief Process Task list using all available threads. void runTasks(bool threaded=true); @@ -114,9 +122,12 @@ signals: DeviceStatus m_status; + void finishAddingSessions(); + QMap new_sessions; private: QList m_tasklist; + }; struct ImportPath diff --git a/sleepyhead/SleepLib/profiles.h b/sleepyhead/SleepLib/profiles.h index f36fe825..a1bba20d 100644 --- a/sleepyhead/SleepLib/profiles.h +++ b/sleepyhead/SleepLib/profiles.h @@ -299,6 +299,7 @@ const QString STR_IS_CompressBackupData = "CompressBackupData"; const QString STR_IS_CompressSessionData = "CompressSessionData"; const QString STR_IS_IgnoreOlderSessions = "IgnoreOlderSessions"; const QString STR_IS_IgnoreOlderSessionsDate = "IgnoreOlderSessionsDate"; +const QString STR_IS_LockSummarySessions = "LockSummarySessions"; // AppearanceSettings Strings const QString STR_AS_GraphHeight = "GraphHeight"; @@ -621,6 +622,8 @@ class SessionSettings : public ProfileSettings initPref(STR_IS_CompressSessionData, false); initPref(STR_IS_IgnoreOlderSessions, false); initPref(STR_IS_IgnoreOlderSessionsDate, QDateTime(QDate::currentDate().addYears(-1), daySplitTime()) ); + initPref(STR_IS_LockSummarySessions, true); + } QTime daySplitTime() const { return getPref(STR_IS_DaySplitTime).toTime(); } @@ -633,6 +636,7 @@ class SessionSettings : public ProfileSettings bool backupCardData() const { return getPref(STR_IS_BackupCardData).toBool(); } bool ignoreOlderSessions() const { return getPref(STR_IS_IgnoreOlderSessions).toBool(); } QDateTime ignoreOlderSessionsDate() const { return getPref(STR_IS_IgnoreOlderSessionsDate).toDateTime(); } + bool lockSummarySessions() const { return getPref(STR_IS_LockSummarySessions).toBool(); } void setDaySplitTime(QTime time) { setPref(STR_IS_DaySplitTime, time); } void setCacheSessions(bool c) { setPref(STR_IS_CacheSessions, c); } @@ -644,6 +648,8 @@ class SessionSettings : public ProfileSettings void setCompressSessionData(bool enabled) { setPref(STR_IS_CompressSessionData, enabled); } void setIgnoreOlderSessions(bool enabled) { setPref(STR_IS_IgnoreOlderSessions, enabled); } void setIgnoreOlderSessionsDate(QDate date) { setPref(STR_IS_IgnoreOlderSessionsDate, QDateTime(date, daySplitTime())); } + void setLockSummarySessions(bool b) { setPref(STR_IS_LockSummarySessions, b); } + }; /*! \class AppearanceSettings diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index dd612ca9..8d2fec32 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -1004,7 +1004,7 @@ QString Daily::getOximeterInformation(Day * oxi) html=""; html+=QString("\n").arg(tr("Oximeter Information")); html+=""; - html+="\n"; + html+="\n"; html+=""; html+=QString("").arg(tr("SpO2 Desaturations")).arg(oxi->count(OXI_SPO2Drop)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_SPO2Drop)/3600.0),0,'f',2); html+=QString("").arg(tr("Pulse Change events")).arg(oxi->count(OXI_PulseChange)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_PulseChange)/3600.0),0,'f',2); @@ -1025,7 +1025,7 @@ QString Daily::getCPAPInformation(Day * cpap) html="
%1
 
"+oxi->machine->brand()+" "+oxi->machine->model()+"
"+oxi->machine->brand()+" "+oxi->machine->series()+"
 
%1: %2 (%3%)
%1: %2 (%3%)
\n"; - html+="").arg(l / 3600, 2, 10, QChar('0')).arg((l / 60) % 60, 2, 10, QChar('0')).arg(l % 60, 2, 10, QChar('0')); + html+="").arg(l / 3600, 2, 10, QChar('0')).arg((l / 60) % 60, 2, 10, QChar('0')).arg(l % 60, 2, 10, QChar('0')); float v = (cpap->hours() - (float(l) / 3600.0)); int q = v * 3600.0; - html+="").arg(q / 3600, 2, 10, QChar('0')).arg((q / 60) % 60, 2, 10, QChar('0')).arg(q % 60, 2, 10, QChar('0')); + html+="").arg(q / 3600, 2, 10, QChar('0')).arg((q / 60) % 60, 2, 10, QChar('0')).arg(q % 60, 2, 10, QChar('0')); EventDataType hc = cpap->count(CPAP_Hypopnea) - cpap->countInsideSpan(CPAP_Ramp, CPAP_Hypopnea); EventDataType oc = cpap->count(CPAP_Obstructive) - cpap->countInsideSpan(CPAP_Ramp, CPAP_Obstructive); EventDataType tc = cpap->count(CPAP_Hypopnea) + cpap->count(CPAP_Obstructive); - EventDataType ahi = (hc+oc) / (float(l)/3600.0); - html+="").arg(ahi, 0, 'f', 2); + EventDataType ahi = (hc+oc) / v; + html+="").arg(ahi, 0, 'f', 2); } } diff --git a/sleepyhead/docs/release_notes.html b/sleepyhead/docs/release_notes.html index 3d9617ef..1ed699c1 100644 --- a/sleepyhead/docs/release_notes.html +++ b/sleepyhead/docs/release_notes.html @@ -10,6 +10,8 @@
New features & bug fixes in v0.9.7
+
  • Fixed inverted CMS50 .spoR file pulse and spo2 +
  • Added CMS50i .spo2 file import support
  • New Feature: Added a Welcome page to make things a little friendlier
  • Fixes another issue that caused session dupplicates on ResMed machines
  • Improved support for Intellipap BiLevel machines
  • @@ -17,8 +19,7 @@
  • New Feature: Preference option to Realign machine detected codes and fix dodgy PRS1 durations using user event flagging.
  • New Feature: Added second set of User Flags detection and preferences.
  • New Feature: Hit Escape key to go back through previous graph selection history
  • -
  • New Feature: Holding Alt shows a vertical line, and the current time position when hovering over LineCharts
  • -
  • New Feature: Hold Alt while selecting a graph area to allow you to take another attempt
  • +
  • New Feature: Hold Alt while selecting a graph area with the mouse, pauses till you release the key, to allow you to take another attempt.
  • Fixed changing languages clobbering graph settings
  • Import now remembers last place you imported from
  • Lock files are now used to help protect the same profiles being open multiple times
  • diff --git a/sleepyhead/mainwindow.cpp b/sleepyhead/mainwindow.cpp index 09cf664a..3d007de0 100644 --- a/sleepyhead/mainwindow.cpp +++ b/sleepyhead/mainwindow.cpp @@ -169,21 +169,21 @@ MainWindow::MainWindow(QWidget *parent) : #endif #endif -#ifdef LOCK_RESMED_SESSIONS - QList machines = p_profile->GetMachines(MT_CPAP); - for (QList::iterator it = machines.begin(); it != machines.end(); ++it) { - QString mclass=(*it)->loaderName(); - if (mclass == STR_MACH_ResMed) { - qDebug() << "ResMed machine found.. locking Session splitting capabilities"; +//#ifdef LOCK_RESMED_SESSIONS +// QList machines = p_profile->GetMachines(MT_CPAP); +// for (QList::iterator it = machines.begin(); it != machines.end(); ++it) { +// QString mclass=(*it)->loaderName(); +// if (mclass == STR_MACH_ResMed) { +// qDebug() << "ResMed machine found.. locking Session splitting capabilities"; - // Have to sacrifice these features to get access to summary data. - p_profile->session->setCombineCloseSessions(0); - p_profile->session->setDaySplitTime(QTime(12,0,0)); - p_profile->session->setIgnoreShortSessions(false); - break; - } - } -#endif +// // Have to sacrifice these features to get access to summary data. +// p_profile->session->setCombineCloseSessions(0); +// p_profile->session->setDaySplitTime(QTime(12,0,0)); +// p_profile->session->setIgnoreShortSessions(false); +// break; +// } +// } +//#endif ui->actionToggle_Line_Cursor->setChecked(p_profile->appearance->lineCursorMode()); diff --git a/sleepyhead/preferencesdialog.cpp b/sleepyhead/preferencesdialog.cpp index 4a5404ec..02519527 100644 --- a/sleepyhead/preferencesdialog.cpp +++ b/sleepyhead/preferencesdialog.cpp @@ -71,18 +71,18 @@ PreferencesDialog::PreferencesDialog(QWidget *parent, Profile *_profile) : }*/ } -#ifdef LOCK_RESMED_SESSIONS - QList machines = p_profile->GetMachines(MT_CPAP); - for (QList::iterator it = machines.begin(); it != machines.end(); ++it) { - const QString & mclass=(*it)->loaderName(); - if (mclass == STR_MACH_ResMed) { - ui->combineSlider->setEnabled(false); - ui->IgnoreSlider->setEnabled(false); - ui->timeEdit->setEnabled(false); - break; - } - } -#endif +//#ifdef LOCK_RESMED_SESSIONS +// QList machines = p_profile->GetMachines(MT_CPAP); +// for (QList::iterator it = machines.begin(); it != machines.end(); ++it) { +// const QString & mclass=(*it)->loaderName(); +// if (mclass == STR_MACH_ResMed) { +// ui->combineSlider->setEnabled(false); +// ui->IgnoreSlider->setEnabled(false); +// ui->timeEdit->setEnabled(false); +// break; +// } +// } +//#endif QLocale locale = QLocale::system(); QString shortformat = locale.dateFormat(QLocale::ShortFormat); @@ -137,6 +137,8 @@ PreferencesDialog::PreferencesDialog(QWidget *parent, Profile *_profile) : ui->IgnoreLCD->display(val); } else { ui->IgnoreLCD->display(STR_GEN_Off); } + ui->LockSummarySessionSplitting->setChecked(profile->session->lockSummarySessions()); + ui->applicationFont->setCurrentFont(QApplication::font()); //ui->applicationFont->setFont(QApplication::font()); ui->applicationFontSize->setValue(QApplication::font().pointSize()); @@ -348,6 +350,10 @@ bool PreferencesDialog::Save() needs_restart = true; } + if (profile->session->lockSummarySessions() != ui->LockSummarySessionSplitting->isChecked()) { + needs_restart = true; + } + if (profile->cpap->userEventPieChart() != ui->showUserFlagsInPie->isChecked()) { // lazy.. fix me needs_restart = true; @@ -403,6 +409,7 @@ bool PreferencesDialog::Save() } profile->cpap->setUserEventPieChart(ui->showUserFlagsInPie->isChecked()); + profile->session->setLockSummarySessions(ui->LockSummarySessionSplitting->isChecked()); profile->appearance->setAllowYAxisScaling(ui->allowYAxisScaling->isChecked()); profile->appearance->setGraphTooltips(ui->graphTooltips->isChecked()); diff --git a/sleepyhead/preferencesdialog.ui b/sleepyhead/preferencesdialog.ui index ad3bb015..fa802463 100644 --- a/sleepyhead/preferencesdialog.ui +++ b/sleepyhead/preferencesdialog.ui @@ -368,6 +368,16 @@ p, li { white-space: pre-wrap; } + + + + Summary only data is more accurate for ResMed users if this is left on. + + + Lock Summary Session Splitting + + +
    "+info.model+""; + html+="
    "+info.series+" "+info.model+""; QString tooltip=(info.brand+"\n"+info.series+" "+info.modelnumber+"\n"+info.serial); tooltip=tooltip.replace(" "," "); @@ -1166,23 +1166,23 @@ QString Daily::getStatisticsInfo(Day * cpap,Day * oxi,Day *pos) } if (cpap) { - int l = cpap->sum(CPAP_Ramp) - (15*60); + int l = cpap->sum(CPAP_Ramp); if (l > 0) { - html+="
    "+tr("Total ramp time")+ - QString("%1:%2:%3
    "+tr("Total ramp time")+ + QString("%1:%2:%3
    "+tr("Time outside of ramp")+ - QString("%1:%2:%3
    "+tr("Time outside of ramp")+ + QString("%1:%2:%3
    "+tr("AHI excluding ramp")+ - QString("%1
    "+tr("AHI excluding ramp")+ + QString("%1