From 215fc41f9a961cbb00a875933d56b5556dd32e23 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Mon, 31 Jul 2017 16:58:36 -0400 Subject: [PATCH 01/13] Fix to import md300w1 dat files --- .../loader_plugins/md300w1_loader.cpp | 6 ++ sleepyhead/SleepLib/serialoximeter.cpp | 2 +- sleepyhead/docs/schema.xml | 18 ++--- sleepyhead/oximeterimport.cpp | 70 ++++++++++++++----- 4 files changed, 68 insertions(+), 28 deletions(-) diff --git a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp index 743446d1..49b0d73a 100644 --- a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp @@ -65,6 +65,7 @@ int MD300W1Loader::Open(QString path) { // Only one active Oximeter module at a time, set in preferences + qDebug() << "MD300W1 Loader opening " << path; m_itemCnt = 0; m_itemTotal = 0; @@ -158,10 +159,12 @@ bool MD300W1Loader::readDATFile(QString path) { QFile file(path); if (!file.exists()) { + qDebug() << "File does not exist: " << path; return false; } if (!file.open(QFile::ReadOnly)) { + qDebug() << "Can't open file R/O: " << path; return false; } @@ -201,8 +204,11 @@ bool MD300W1Loader::readDATFile(QString path) } } else { // Create a new session + qDebug() << "Create session for " << datestr; + qDebug() << "Create session for " << datetime.toString("yyyy.MM.dd HH:mm:ss"); oxirec = new QVector; oxisessions[datetime] = oxirec; + m_startTime = datetime; // works for single session files... } } diff --git a/sleepyhead/SleepLib/serialoximeter.cpp b/sleepyhead/SleepLib/serialoximeter.cpp index 8c461d17..79cf3da4 100644 --- a/sleepyhead/SleepLib/serialoximeter.cpp +++ b/sleepyhead/SleepLib/serialoximeter.cpp @@ -30,7 +30,7 @@ bool SerialOximeter::scanDevice(QString keyword, quint16 vendor_id, quint16 prod static bool dumponce = true; QStringList ports; - //qDebug() << "Scanning for USB Serial devices"; + qDebug() << "Scanning for USB Serial devices"; QList list=QSerialPortInfo::availablePorts(); // How does the mac detect this as a SPO2 device? diff --git a/sleepyhead/docs/schema.xml b/sleepyhead/docs/schema.xml index 1a824e4f..7e5b80ca 100644 --- a/sleepyhead/docs/schema.xml +++ b/sleepyhead/docs/schema.xml @@ -1,11 +1,11 @@ - + - + @@ -20,14 +20,14 @@ - + - - - - - + + + + + @@ -42,7 +42,7 @@ - + diff --git a/sleepyhead/oximeterimport.cpp b/sleepyhead/oximeterimport.cpp index 11665b91..09b82b6b 100644 --- a/sleepyhead/oximeterimport.cpp +++ b/sleepyhead/oximeterimport.cpp @@ -36,24 +36,24 @@ OximeterImport::OximeterImport(QWidget *parent) : ui->setupUi(this); setWindowTitle(tr("Oximeter Import Wizard")); ui->stackedWidget->setCurrentIndex(0); - oximodule = nullptr; - liveView = new gGraphView(this); - liveView->setVisible(false); - liveView->setShowAuthorMessage(false); ui->retryButton->setVisible(false); ui->stopButton->setVisible(false); ui->saveButton->setVisible(false); ui->syncButton->setVisible(false); ui->chooseSessionButton->setVisible(false); + oximodule = nullptr; importMode = IM_UNDEFINED; + liveView = new gGraphView(this); + liveView->setVisible(false); + liveView->setShowAuthorMessage(false); QVBoxLayout * lvlayout = new QVBoxLayout; lvlayout->setMargin(0); - ui->liveViewFrame->setLayout(lvlayout); lvlayout->addWidget(liveView); - plethyGraph = new gGraph("Plethy", liveView, STR_TR_Plethy, STR_UNIT_Hz); + ui->liveViewFrame->setLayout(lvlayout); + plethyGraph = new gGraph("Plethy", liveView, STR_TR_Plethy, STR_UNIT_Hz); plethyGraph->AddLayer(new gYAxis(), LayerLeft, gYAxis::Margin); plethyGraph->AddLayer(new gXAxis(), LayerBottom, 0, 20); plethyGraph->AddLayer(plethyChart = new gLineChart(OXI_Plethy)); @@ -155,6 +155,7 @@ void OximeterImport::on_nextButton_clicked() void OximeterImport::updateStatus(QString msg) { + qDebug() << "updateStatus to " << msg; ui->logBox->appendPlainText(msg); ui->directImportStatus->setText(msg); ui->liveStatusLabel->setText(msg); @@ -166,7 +167,7 @@ SerialOximeter * OximeterImport::detectOximeter() const int PORTSCAN_TIMEOUT=30000; const int delay=100; - + qDebug() << "Attempt to detect Oximeter"; ui->retryButton->setVisible(false); QList loaders; //= GetOxiLoaders(); @@ -235,6 +236,7 @@ void OximeterImport::on_directImportButton_clicked() ui->informationButton->setVisible(false); ui->stackedWidget->setCurrentWidget(ui->directImportPage); + qDebug() << "Direct Import button clicked" ; oximodule = detectOximeter(); if (!oximodule) return; @@ -311,6 +313,8 @@ void OximeterImport::on_directImportButton_clicked() void OximeterImport::doImport() { + qDebug() << "Starting doImport"; + if (oximodule->commandDriven()) { if (chosen_sessions.size() == 0) { ui->connectLabel->setText("

"+tr("Nothing to import")+"

"); @@ -371,6 +375,8 @@ void OximeterImport::finishedImport(SerialOximeter * oxi) { Q_UNUSED(oxi); + qDebug() << "finished Import "; + connect(oximodule, SIGNAL(importComplete(SerialOximeter*)), this, SLOT(finishedImport(SerialOximeter*))); ui->cancelButton->setVisible(true); disconnect(oximodule, SIGNAL(updateProgress(int,int)), this, SLOT(doUpdateProgress(int,int))); @@ -400,7 +406,8 @@ void OximeterImport::on_fileImportButton_clicked() const QString documentsFolder = QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation); #endif - + qDebug() << "File Import button clicked"; + QString filename = QFileDialog::getOpenFileName(nullptr , tr("Select a valid oximetry data file"), documentsFolder, tr("Oximetry Files (*.spo *.spor *.spo2 *.dat)")); if (filename.isEmpty()) @@ -411,6 +418,7 @@ void OximeterImport::on_fileImportButton_clicked() // Make sure filename dialog had time to close properly.. QApplication::processEvents(); + qDebug() << "Chosen filename is " << filename; QList loaders = GetOxiLoaders(); bool success = false; @@ -427,6 +435,8 @@ void OximeterImport::on_fileImportButton_clicked() QMessageBox::warning(this, STR_MessageBox_Warning, tr("No Oximetery module could parse the given file:")+QString("\n\n%1").arg(filename), QMessageBox::Ok); return; } + qDebug() << "Using loader " << oximodule->loaderName(); + ui->informationButton->setVisible(false); importMode = IM_FILE; @@ -434,14 +444,17 @@ void OximeterImport::on_fileImportButton_clicked() if (oximodule->oxisessions.size() > 1) { chooseSession(); } else { + // oximodule->setStartTime( ??? ); Nope, it was set in the loader module by the file import routime on_syncButton_clicked(); } + qDebug() << "Oximodule startTime is " << oximodule->startTime().toString("yyyy.MM.dd HH:mm:ss"); } void OximeterImport::on_liveImportButton_clicked() { ui->informationButton->setVisible(false); + qDebug() << "Live Import button clicked"; ui->stackedWidget->setCurrentWidget(ui->liveImportPage); ui->liveImportPage->layout()->addWidget(ui->progressBar); QApplication::processEvents(); @@ -507,6 +520,8 @@ void OximeterImport::on_liveImportButton_clicked() void OximeterImport::finishedRecording() { + qDebug() << "Finished Recording"; + updateTimer.stop(); oximodule->closeDevice(); disconnect(&updateTimer, SIGNAL(timeout()), this, SLOT(updateLiveDisplay())); @@ -529,6 +544,7 @@ void OximeterImport::finishedRecording() void OximeterImport::on_retryButton_clicked() { + qDebug() << "Retry button clicked"; if (ui->stackedWidget->currentWidget() == ui->directImportPage) { on_directImportButton_clicked(); } else if (ui->stackedWidget->currentWidget() == ui->liveImportPage) { @@ -538,6 +554,7 @@ void OximeterImport::on_retryButton_clicked() void OximeterImport::on_stopButton_clicked() { + qDebug() << "Stop button clicked"; if (oximodule) { oximodule->abort(); } @@ -545,6 +562,7 @@ void OximeterImport::on_stopButton_clicked() void OximeterImport::on_calendarWidget_clicked(const QDate &date) { + qDebug() << "Calendar widget clicked " << date.toString("yyyy.MM.dd"); if (ui->radioSyncCPAP->isChecked()) { Day * day = p_profile->GetGoodDay(date, MT_CPAP); @@ -581,6 +599,7 @@ void OximeterImport::on_calendarWidget_selectionChanged() { on_calendarWidget_clicked(ui->calendarWidget->selectedDate()); } + void OximeterImport::onSessionSelected(Session * session) { QDateTime time=QDateTime::fromMSecsSinceEpoch(session->first(), Qt::UTC); @@ -589,6 +608,7 @@ void OximeterImport::onSessionSelected(Session * session) void OximeterImport::on_sessionBackButton_clicked() { + qDebug() << "Session Back button clicked"; int idx = (sessbar->selected()-1); if (idx >= 0) { sessbar->setSelected(idx); @@ -600,6 +620,7 @@ void OximeterImport::on_sessionBackButton_clicked() void OximeterImport::on_sessionForwardButton_clicked() { + qDebug() << "Session Forward button clicked"; int idx = (sessbar->selected()+1); if (idx < sessbar->count()) { sessbar->setSelected(idx); @@ -619,10 +640,15 @@ void OximeterImport::on_radioSyncCPAP_clicked() void OximeterImport::on_radioSyncOximeter_clicked() { + qDebug() << "Use OximeterTime button clicked"; ui->syncCPAPGroup->setVisible(false); - if (oximodule && oximodule->isStartTimeValid()) { - ui->calendarWidget->setSelectedDate(oximodule->startTime().date()); - ui->dateTimeEdit->setDateTime(oximodule->startTime()); + if ( oximodule ) { + if (oximodule->isStartTimeValid()) { + qDebug() << "Oximeter time is valid " << oximodule->startTime().toString(); + ui->calendarWidget->setSelectedDate(oximodule->startTime().date()); + ui->dateTimeEdit->setDateTime(oximodule->startTime()); + } else + qDebug() << "Oximeter time is not valid"; } } @@ -721,6 +747,7 @@ void OximeterImport::updateLiveDisplay() void OximeterImport::on_cancelButton_clicked() { + qDebug() << "Cancel button clicked"; if (oximodule && oximodule->isStreaming()) { oximodule->closeDevice(); oximodule->trashRecords(); @@ -754,7 +781,9 @@ void OximeterImport::on_informationButton_clicked() void OximeterImport::on_syncButton_clicked() { + qDebug() << "Sync button clicked"; Q_ASSERT(oximodule != nullptr); + qDebug() << "Oximodule Start Time is " << oximodule->startTime().toString("yyyy.MM.dd HH.mm.ss") << "Duration: " << oximodule->getDuration(/* dummy */ 0 ); ui->stackedWidget->setCurrentWidget(ui->syncPage); @@ -765,6 +794,7 @@ void OximeterImport::on_syncButton_clicked() QDate last = p_profile->LastDay(); QDate oxidate = oximodule->startTime().date(); + qDebug() << "Oximodule start date is " << oximodule->startTime().date().toString("yyyy.MM.dd"); if ((oxidate >= first) && (oxidate <= last)) { @@ -795,6 +825,7 @@ void OximeterImport::on_syncButton_clicked() void OximeterImport::on_saveButton_clicked() { + qDebug() << "Oximeter Save button clicked"; if (!oximodule) return; QVector * oxirec = nullptr; @@ -1008,6 +1039,7 @@ void OximeterImport::on_saveButton_clicked() void OximeterImport::chooseSession() { + qDebug() << "Oximeter Choose Session called"; selecting_session = false; ui->stackedWidget->setCurrentWidget(ui->chooseSessionPage); @@ -1034,7 +1066,7 @@ void OximeterImport::chooseSession() ui->tableOxiSessions->setItem(row, 1, item); item->setFlags(item->flags() & ~Qt::ItemIsEditable); - item = new QTableWidgetItem(tr("CMS50 Session %1").arg(row+1, 0)); + item = new QTableWidgetItem(tr("Oximeter Session %1").arg(row+1, 0)); ui->tableOxiSessions->setItem(row, 2, item); item->setFlags(item->flags() & ~Qt::ItemIsEditable); @@ -1046,18 +1078,20 @@ void OximeterImport::chooseSession() void OximeterImport::on_chooseSessionButton_clicked() { + qDebug() << "Chosen session clicked"; ui->chooseSessionButton->setVisible(false); - QTableWidgetItem * item = ui->tableOxiSessions->item(ui->tableOxiSessions->currentRow(),0); + QTableWidgetItem * item_0 = ui->tableOxiSessions->item(ui->tableOxiSessions->currentRow(),0); + QTableWidgetItem * item_1 = ui->tableOxiSessions->item(ui->tableOxiSessions->currentRow(),1); - if (!item) return; - QDateTime datetime = item->data(Qt::UserRole+1).toDateTime(); + if (!item_0 || !item_1) return; + QDateTime datetime = item_0->data(Qt::DisplayRole).toDateTime(); oximodule->setStartTime(datetime); - oximodule->setDuration(item->data(Qt::UserRole+2).toInt()); + oximodule->setDuration(item_1->data(Qt::DisplayRole).toInt()); if (selecting_session) { ui->stackedWidget->setCurrentWidget(ui->directImportPage); - chosen_sessions.push_back(item->data(Qt::UserRole).toInt()); + chosen_sessions.push_back(ui->tableOxiSessions->currentRow()); // go back and start import doImport(); @@ -1118,7 +1152,7 @@ void OximeterImport::on_oximeterType_currentIndexChanged(int index) ui->oldCMS50specific->setVisible(true); ui->newCMS50settingsPanel->setVisible(false); break; - default: + default: // ChoiceMMed oximeters, and others? ui->directImportButton->setEnabled(false); ui->liveImportButton->setEnabled(false); ui->fileImportButton->setEnabled(true); From 30cc04f691e223fafced205081fb1af2672b69b9 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Mon, 31 Jul 2017 19:56:45 -0400 Subject: [PATCH 02/13] Clean up and fix duration display --- sleepyhead/oximeterimport.cpp | 15 ++++++++++----- 1 file changed, 10 insertions(+), 5 deletions(-) diff --git a/sleepyhead/oximeterimport.cpp b/sleepyhead/oximeterimport.cpp index 09b82b6b..8a067ddf 100644 --- a/sleepyhead/oximeterimport.cpp +++ b/sleepyhead/oximeterimport.cpp @@ -286,9 +286,9 @@ void OximeterImport::on_directImportButton_clicked() item = new QTableWidgetItem(datetime.date().toString(Qt::SystemLocaleShortDate)+" "+datetime.time().toString("HH:mm:ss")); ui->tableOxiSessions->setItem(i, 0, item); - item->setData(Qt::UserRole+1, datetime); - item->setData(Qt::UserRole, i); - item->setData(Qt::UserRole+2, duration); + // item->setData(Qt::UserRole+1, datetime); + // item->setData(Qt::UserRole, i); + // item->setData(Qt::UserRole+2, duration); item->setFlags(item->flags() & ~Qt::ItemIsEditable); item = new QTableWidgetItem(QString(). sprintf("%02i:%02i:%02i", h,m,s)); @@ -447,7 +447,7 @@ void OximeterImport::on_fileImportButton_clicked() // oximodule->setStartTime( ??? ); Nope, it was set in the loader module by the file import routime on_syncButton_clicked(); } - qDebug() << "Oximodule startTime is " << oximodule->startTime().toString("yyyy.MM.dd HH:mm:ss"); + qDebug() << "Finished file import: Oximodule startTime is " << oximodule->startTime().toString("yyyy.MM.dd HH:mm:ss"); } void OximeterImport::on_liveImportButton_clicked() @@ -1062,7 +1062,12 @@ void OximeterImport::chooseSession() ui->tableOxiSessions->setItem(row, 0, item); item->setFlags(item->flags() & ~Qt::ItemIsEditable); - item = new QTableWidgetItem(QString(). sprintf("%lli", oxirec->size() * oximodule->importResolution() / 1000L)); + long int duration = oxirec->size() * oximodule->importResolution() / 1000L; + int h = duration / 3600; + int m = (duration / 60) % 60; + int s = duration % 60; + + item = new QTableWidgetItem(QString(). sprintf("%02i:%02i:%02i", h,m,s)); ui->tableOxiSessions->setItem(row, 1, item); item->setFlags(item->flags() & ~Qt::ItemIsEditable); From 01a56a01dcc2a4a869a9f4471b117c5e292fe7bf Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Tue, 1 Aug 2017 20:10:06 -0400 Subject: [PATCH 03/13] Fix percent labels in Exported CSV Daily and Session Summary files --- sleepyhead/exportcsv.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sleepyhead/exportcsv.cpp b/sleepyhead/exportcsv.cpp index 0bae9cac..8bb528da 100644 --- a/sleepyhead/exportcsv.cpp +++ b/sleepyhead/exportcsv.cpp @@ -212,7 +212,7 @@ void ExportCSV::on_exportButton_clicked() } for (int i = 0; i < p90list.size(); i++) { - header += sep + schema::channel[p90list[i]].label() + tr(" %1%").arg(percent, 0, 'f', 0); + header += sep + schema::channel[p90list[i]].label() + tr(" %1%").arg(percent*100.0, 0, 'f', 0); } } From 2c9b69619771feb4db5ac97a6ab6e3698eb1fd64 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Tue, 1 Aug 2017 21:14:39 -0400 Subject: [PATCH 04/13] Enhance CSV output with Maximums, Flow Limitations and Session data.\nUse Preference values to select average and maximum values. --- sleepyhead/SleepLib/session.cpp | 25 ++++++++++++++++++++ sleepyhead/SleepLib/session.h | 5 ++++ sleepyhead/exportcsv.cpp | 42 +++++++++++++++++++++++++-------- 3 files changed, 62 insertions(+), 10 deletions(-) diff --git a/sleepyhead/SleepLib/session.cpp b/sleepyhead/SleepLib/session.cpp index af1da9fd..5daaef84 100644 --- a/sleepyhead/SleepLib/session.cpp +++ b/sleepyhead/SleepLib/session.cpp @@ -2197,6 +2197,31 @@ EventDataType Session::wavg(ChannelID id) return val; } +EventDataType Session::calcMiddle(ChannelID code) +{ + int c = p_profile->general->prefCalcMiddle(); + + if (c == 0) { + return percentile(code, 0.5); // Median + } else if (c == 1 ) { + return wavg(code); // Weighted Average + } else { + return avg(code); // Average + } +} + +EventDataType Session::calcMax(ChannelID code) +{ + return p_profile->general->prefCalcMax() ? percentile(code, 0.995f) : Max(code); +} + +EventDataType Session::calcPercentile(ChannelID code) +{ + double p = p_profile->general->prefCalcPercentile() / 100.0; + return percentile(code, p); +} + + EventList *Session::AddEventList(ChannelID code, EventListType et, EventDataType gain, EventDataType offset, EventDataType min, EventDataType max, EventDataType rate, bool second_field) { diff --git a/sleepyhead/SleepLib/session.h b/sleepyhead/SleepLib/session.h index ae174cdd..c11e6830 100644 --- a/sleepyhead/SleepLib/session.h +++ b/sleepyhead/SleepLib/session.h @@ -346,6 +346,11 @@ class Session //! \brief Returns the amount of time (in decimal minutes) the Channel spent below the threshold EventDataType timeBelowThreshold(ChannelID id, EventDataType threshold); + //! \brief According to preferences.. + EventDataType calcMiddle(ChannelID code); + EventDataType calcMax(ChannelID code); + EventDataType calcPercentile(ChannelID code); + //! \brief Returns true if the channel has events loaded, or a record of a count for when they are not bool channelExists(ChannelID name); diff --git a/sleepyhead/exportcsv.cpp b/sleepyhead/exportcsv.cpp index 8bb528da..b16eb552 100644 --- a/sleepyhead/exportcsv.cpp +++ b/sleepyhead/exportcsv.cpp @@ -163,7 +163,7 @@ void ExportCSV::on_exportButton_clicked() // fields.append(DumpField(NoChannel,MT_CPAP,ST_SESSIONS)); - QList countlist, avglist, p90list; + QList countlist, avglist, p90list, maxlist; countlist.append(CPAP_Hypopnea); countlist.append(CPAP_Obstructive); countlist.append(CPAP_Apnea); @@ -180,16 +180,23 @@ void ExportCSV::on_exportButton_clicked() countlist.append(CPAP_UserFlag2); countlist.append(CPAP_PressurePulse); - - avglist.append(CPAP_Pressure); avglist.append(CPAP_IPAP); avglist.append(CPAP_EPAP); + avglist.append(CPAP_FLG); // Pholynyk, 25Aug2015, add ResMed Flow Limitation p90list.append(CPAP_Pressure); p90list.append(CPAP_IPAP); p90list.append(CPAP_EPAP); - EventDataType percent = 0.90F; + p90list.append(CPAP_FLG); + + float percentile=p_profile->general->prefCalcPercentile()/100.0; // Pholynyk, 18Aug2015 + EventDataType percent = percentile; // was 0.90F + + maxlist.append(CPAP_Pressure); // Pholynyk, 18Aug2015, add maximums + maxlist.append(CPAP_IPAP); + maxlist.append(CPAP_EPAP); + maxlist.append(CPAP_FLG); // Not sure this section should be translateable.. :-/ if (ui->rb1_details->isChecked()) { @@ -208,11 +215,15 @@ void ExportCSV::on_exportButton_clicked() } for (int i = 0; i < avglist.size(); i++) { - header += sep + schema::channel[avglist[i]].label() + " " + tr(" Avg"); + header += sep + Day::calcMiddleLabel(avglist[i]); // Pholynyk, 18Aug2015 } for (int i = 0; i < p90list.size(); i++) { - header += sep + schema::channel[p90list[i]].label() + tr(" %1%").arg(percent*100.0, 0, 'f', 0); + header += sep + tr(" %1%").arg(percent*100.0, 0, 'f', 0) + schema::channel[p90list[i]].label(); + } + + for (int i = 0; i < maxlist.size(); i++) { + header += sep + Day::calcMaxLabel(maxlist[i]); // added -- Pholynyk, 18Aug2015 } } @@ -256,11 +267,18 @@ void ExportCSV::on_exportButton_clicked() } for (int i = 0; i < avglist.size(); i++) { - data += sep + QString::number(day->wavg(avglist.at(i))); + float avg = day->calcMiddle(avglist.at(i)); + data += sep + QString::number(avg); // Pholynyk, 11Aug2015 } for (int i = 0; i < p90list.size(); i++) { - data += sep + QString::number(day->p90(p90list.at(i))); + float p90 = day->percentile(p90list.at(i), percent); + data += sep + QString::number(p90); // Pholynyk, 11Aug2015 + } + + for (int i = 0; i < maxlist.size(); i++) { + float max = day->calcMax(maxlist.at(i)); + data += sep + QString::number(max); // added -- Pholynyk, 18Aug2015 } data += newline; @@ -292,11 +310,15 @@ void ExportCSV::on_exportButton_clicked() } for (int j = 0; j < avglist.size(); j++) { - data += sep + QString::number(day->wavg(avglist.at(j))); + data += sep + QString::number(sess->calcMiddle(avglist.at(j))); // Pholynyk, 11Aug2015 } for (int j = 0; j < p90list.size(); j++) { - data += sep + QString::number(day->p90(p90list.at(j))); + data += sep + QString::number(sess->percentile(p90list.at(j), percent)); // Pholynyk, 11Aug2015 + } + + for (int i = 0; i < maxlist.size(); i++) { + data += sep + QString::number(sess->calcMax(maxlist.at(i))); // Pholynyk, 11Aug2015 } data += newline; From c5247f59582a884f082b6242a92894d6d10f8879 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Tue, 1 Aug 2017 21:31:59 -0400 Subject: [PATCH 05/13] Use Prefs value for Upper Percentile in statistics page --- sleepyhead/statistics.cpp | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/sleepyhead/statistics.cpp b/sleepyhead/statistics.cpp index 89cb8f8d..34b65c7d 100644 --- a/sleepyhead/statistics.cpp +++ b/sleepyhead/statistics.cpp @@ -541,11 +541,14 @@ Statistics::Statistics(QObject *parent) : rows.push_back(StatisticsRow("PulseChange", SC_CPH, MT_OXIMETER)); // These are for formatting the headers for the first column + int percentile=trunc(p_profile->general->prefCalcPercentile()); // Pholynyk, 10Mar2016 + char perCentStr[20]; + snprintf(perCentStr, 20, "%d%% %%1", percentile); // calcnames[SC_UNDEFINED] = ""; calcnames[SC_MEDIAN] = tr("%1 Median"); calcnames[SC_AVG] = tr("Average %1"); calcnames[SC_WAVG] = tr("Average %1"); - calcnames[SC_90P] = tr("90% %1"); // this gets converted to whatever the upper percentile is set to + calcnames[SC_90P] = tr(perCentStr); // this gets converted to whatever the upper percentile is set to calcnames[SC_MIN] = tr("Min %1"); calcnames[SC_MAX] = tr("Max %1"); calcnames[SC_CPH] = tr("%1 Index"); @@ -2157,6 +2160,9 @@ QString StatisticsRow::value(QDate start, QDate end) QString value; float days = p_profile->countDays(type, start, end); + float percentile=p_profile->general->prefCalcPercentile()/100.0; // Pholynyk, 10Mar2016 + EventDataType percent = percentile; // was 0.90F + // Handle special data sources first if (calc == SC_AHI) { value = QString("%1").arg(calcAHI(start, end), 0, 'f', decimals); @@ -2187,7 +2193,7 @@ QString StatisticsRow::value(QDate start, QDate end) val = p_profile->calcPercentile(code, 0.5F, type, start, end); break; case SC_90P: - val = p_profile->calcPercentile(code, 0.9F, type, start, end); + val = p_profile->calcPercentile(code, percent, type, start, end); break; case SC_MIN: val = p_profile->calcMin(code, type, start, end); From 98e9204329ab4bafb2a9633bf1cdf88b4dc8a78a Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Tue, 1 Aug 2017 22:16:08 -0400 Subject: [PATCH 06/13] Fix percentile lables in header --- sleepyhead/exportcsv.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sleepyhead/exportcsv.cpp b/sleepyhead/exportcsv.cpp index b16eb552..49fb2fbc 100644 --- a/sleepyhead/exportcsv.cpp +++ b/sleepyhead/exportcsv.cpp @@ -219,7 +219,7 @@ void ExportCSV::on_exportButton_clicked() } for (int i = 0; i < p90list.size(); i++) { - header += sep + tr(" %1%").arg(percent*100.0, 0, 'f', 0) + schema::channel[p90list[i]].label(); + header += sep + tr("%1% ").arg(percent*100.0, 0, 'f', 0) + schema::channel[p90list[i]].label(); } for (int i = 0; i < maxlist.size(); i++) { From 5a3239711f1e0f2ac5dc993e1ef1067d3098adaf Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Fri, 4 Aug 2017 10:54:53 -0400 Subject: [PATCH 07/13] Mark this release as mine (PMO) and beta --- sleepyhead/version.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sleepyhead/version.h b/sleepyhead/version.h index 80115ff7..479a62a4 100644 --- a/sleepyhead/version.h +++ b/sleepyhead/version.h @@ -18,7 +18,7 @@ const int major_version = 1; // incompatible API changes const int minor_version = 0; // new features that don't break things const int revision_number = 0; // bugfixes, revisions -const QString ReleaseStatus = "beta"; +const QString ReleaseStatus = "PMO-beta"; const QString VersionString = QString("%1.%2.%3-%4-%5").arg(major_version).arg(minor_version).arg(revision_number).arg(ReleaseStatus).arg(build_number); From dea11f39796f2a5afc0527e8d012881fae6a5a74 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Thu, 21 Sep 2017 10:33:17 -0400 Subject: [PATCH 08/13] add SpO2 ext for case-sensitive file systems --- sleepyhead/oximeterimport.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sleepyhead/oximeterimport.cpp b/sleepyhead/oximeterimport.cpp index 8a067ddf..935e44e3 100644 --- a/sleepyhead/oximeterimport.cpp +++ b/sleepyhead/oximeterimport.cpp @@ -408,7 +408,7 @@ void OximeterImport::on_fileImportButton_clicked() qDebug() << "File Import button clicked"; - QString filename = QFileDialog::getOpenFileName(nullptr , tr("Select a valid oximetry data file"), documentsFolder, tr("Oximetry Files (*.spo *.spor *.spo2 *.dat)")); + QString filename = QFileDialog::getOpenFileName(nullptr , tr("Select a valid oximetry data file"), documentsFolder, tr("Oximetry Files (*.spo *.spor *.spo2 *.SpO2 *.dat)")); if (filename.isEmpty()) return; From 96d5516cef0479372b4852dda1440abca5a8a68f Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Thu, 21 Sep 2017 10:41:35 -0400 Subject: [PATCH 09/13] Set m_starttime to current day if no date in file (CMS50D+) --- sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp index a50e0d8c..64bb71ad 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp @@ -599,8 +599,11 @@ bool CMS50Loader::readSpoRFile(QString path) in >> year >> month >> day; in >> hour >> minute >> second; - - m_startTime = QDateTime(QDate(year, month, day), QTime(hour, minute, second)); + + if ( year == 0 ) // typically from a CMS50D+ + m_startTime = QDateTime(QDate::currentDate(), QTime(hour, minute, second)); + else + m_startTime = QDateTime(QDate(year, month, day), QTime(hour, minute, second)); // ignoring it for now pos += 0x1c + 200; @@ -612,7 +615,7 @@ bool CMS50Loader::readSpoRFile(QString path) bytes_per_record = remainder / samples; qDebug() << samples << "samples of" << bytes_per_record << "bytes each"; - // CMS50I .spo2 data have 4 digits, a 16bit, followed by spo2 then pulse + // CMS50I .spo2 data have 4 bytes: a 16bit, followed by spo2 then pulse } From 7c367b5f01792ca86fdcce83af57f7e8558de7d2 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Thu, 21 Sep 2017 10:45:48 -0400 Subject: [PATCH 10/13] add debug code to direct import routines (data duplication?) --- .../loader_plugins/cms50f37_loader.cpp | 95 +++++++++++++------ 1 file changed, 64 insertions(+), 31 deletions(-) diff --git a/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp b/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp index 4ccb44d4..6bb1f5c2 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp @@ -72,12 +72,15 @@ bool CMS50F37Loader::openDevice() b = scanDevice("rfcomm", 0, 0); // Linux } if (!b) { + qDebug() << "No oximeter found"; return false; } } serial.setPortName(port); - if (!serial.open(QSerialPort::ReadWrite)) + if (!serial.open(QSerialPort::ReadWrite)) { + qDebug() << "Failed to open oximeter"; return false; + } // forward this stuff @@ -193,7 +196,8 @@ QString CMS50F37Loader::getUser() do { QApplication::processEvents(); } while (user.isEmpty() && (time.elapsed() < TIMEOUT)); - + + qDebug() << "User = " << user; return user; } @@ -208,6 +212,8 @@ QString CMS50F37Loader::getVendor() do { QApplication::processEvents(); } while (vendor.isEmpty() && (time.elapsed() < TIMEOUT)); + + qDebug() << "Vendor is " << vendor; return vendor; } @@ -233,6 +239,8 @@ QString CMS50F37Loader::getModel() } else { duration_divisor = 2; } + + qDebug() << "Model is " << model; return model; } @@ -252,6 +260,8 @@ QString CMS50F37Loader::getDeviceID() do { QApplication::processEvents(); } while (devid.isEmpty() && (time.elapsed() < TIMEOUT)); + + qDebug() << "Device Id is " << devid; return devid; } @@ -266,6 +276,7 @@ int CMS50F37Loader::getSessionCount() QApplication::processEvents(); } while ((session_count < 0) && (time.elapsed() < TIMEOUT)); + qDebug() << "Session count is " << session_count; return session_count; } @@ -279,6 +290,7 @@ int CMS50F37Loader::getOximeterInfo() QApplication::processEvents(); } while ((device_info < 0) && (time.elapsed() < TIMEOUT)); + qDebug() << "Device Info is " << device_info; return device_info; } @@ -295,12 +307,14 @@ int CMS50F37Loader::getDuration(int session) QApplication::processEvents(); } while ((duration < 0) && (time.elapsed() < TIMEOUT)); + qDebug() << "Session duration is " << duration << "Divided by " << duration_divisor; return duration / duration_divisor; } QDateTime CMS50F37Loader::getDateTime(int session) { + QDateTime datetime; imp_date = QDate(); imp_time = QTime(); sendCommand(COMMAND_GET_SESSION_TIME, session); @@ -311,9 +325,12 @@ QDateTime CMS50F37Loader::getDateTime(int session) } while ((imp_date.isNull() || imp_time.isNull()) && (time.elapsed() < TIMEOUT)); if (imp_date.isNull() || imp_time.isNull()) - return QDateTime(); + datetime = QDateTime(); + else + datetime = QDateTime(imp_date, imp_time); - return QDateTime(imp_date, imp_time); + qDebug() << "Oximeter DateTime is " << datetime.toString("yyyy-MMM-dd HH:mm:ss"); + return datetime; } @@ -335,7 +352,9 @@ void CMS50F37Loader::processBytes(QByteArray bytes) int year, month, day; quint8 pulse; - + quint8 spo2; + quint16 pi; // perfusion index + do { quint8 res = buffer.at(idx); @@ -356,7 +375,7 @@ void CMS50F37Loader::processBytes(QByteArray bytes) if (res == resimport) break; } // add a dummy to make up for it. - qDebug() << "lost sync, padding..."; + qDebug() << "pB: lost sync, padding..."; oxirec->append(OxiRecord(0,0,0)); } continue; @@ -375,14 +394,14 @@ void CMS50F37Loader::processBytes(QByteArray bytes) QString extra = QString(buffer.mid(idx+3, 6)); model += extra.trimmed(); modelsegments++; - qDebug() << "Model:" << model; + qDebug() << "pB: Model:" << model; } break; case 0x03: // Vendor string data = buffer.at(idx+1); if (data == 0) { vendor = QString(buffer.mid(idx+2, 6)); - qDebug() << "Vendor:" << vendor; + qDebug() << "pB: Vendor:" << vendor; } break; @@ -392,14 +411,14 @@ void CMS50F37Loader::processBytes(QByteArray bytes) } devid = QString(buffer.mid(idx+2, 7)); - qDebug() << "Device ID:" << devid; + qDebug() << "pB: Device ID:" << devid; break; // COMMAND_GET_USER_INFO case 0x05: // 5,80,80,f5,f3,e5,f2,80,80 // User user = QString(buffer.mid(idx+3).trimmed()); - qDebug() << "0x05:" << user; + qDebug() << "pB: 0x05:" << user; break; case 0x6: // 6,80,80,87 @@ -418,7 +437,7 @@ void CMS50F37Loader::processBytes(QByteArray bytes) day = QString().sprintf("%02i", buffer.at(idx+7)).toInt(); imp_date = QDate(year,month,day); - qDebug() << imp_date; + qDebug() << "pB: ymd " << year << month << day << " impDate " << imp_date; break; // COMMAND_GET_SESSION_DURATION @@ -458,7 +477,7 @@ void CMS50F37Loader::processBytes(QByteArray bytes) case 0x12: // 12,80,80,80,82,a6,92,80 tmpstr = QString().sprintf("%02i:%02i:%02i",buffer.at(idx+4), buffer.at(idx+5), buffer.at(idx+6)); imp_time = QTime::fromString(tmpstr, "HH:mm:ss"); - qDebug() << imp_time; + qDebug() << "pB: tmpStr:" << tmpstr << " impTime " << imp_time; break; case 0x13: // 13,80,a0,a0,a0,a0,a0,a0,a0 @@ -469,6 +488,7 @@ void CMS50F37Loader::processBytes(QByteArray bytes) case 0x09: // cms50i data sequence case 0x0f: // f,80,de,c2,de,c2,de,c2 cms50F data... if (!started_import) { + qDebug() << "pB: Starting import"; started_import = true; started_reading = true; finished_import = false; @@ -493,7 +513,7 @@ void CMS50F37Loader::processBytes(QByteArray bytes) break; default: - qDebug() << "unknown cms50 result?" << hex << (int)res; + qDebug() << "pB: unknown cms50F result?" << hex << (int)res; break; } @@ -507,10 +527,10 @@ void CMS50F37Loader::processBytes(QByteArray bytes) buf[i] = (buf[i] & 0x7f) | ((msb & 0x01) ? 0x80 : 0); } - quint16 pi = buf[4] | buf[5] << 8; + pi = buf[4] | buf[5] << 8; pulse = buf[3]; - quint8 spo2 = buf[2] & 0x7f; - qDebug() << "Pulse=" << pulse << "SPO2=" << spo2 << "Pi=" << pi; + spo2 = buf[2] & 0x7f; + qDebug() << "pB: Pulse=" << pulse << "SPO2=" << spo2 << "PI=" << pi; oxirec->append(((spo2 == 0) || (pulse == 0)) ? OxiRecord(0,0,0) : OxiRecord(pulse, spo2, pi)); } else if (res == 0x0f) { @@ -521,13 +541,19 @@ void CMS50F37Loader::processBytes(QByteArray bytes) } pulse = buffer.at(idx+3); - oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(pulse, buffer.at(idx+2))); + spo2 = buffer.at(idx+2); + qDebug() << "pB: Pulse=" << pulse << "SPO2=" << spo2; + oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(pulse, spo2)); pulse = buffer.at(idx+5); - oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(pulse, buffer.at(idx+4))); + spo2 = buffer.at(idx+4); + qDebug() << "pB: Pulse=" << pulse << "SPO2=" << spo2; + oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(pulse, spo2)); pulse = buffer.at(idx+7); - oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(pulse, buffer.at(idx+6))); + spo2 = buffer.at(idx+6); + qDebug() << "pB: Pulse=" << pulse << "SPO2=" << spo2; + oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(pulse, spo2)); } QStringList str; @@ -537,9 +563,9 @@ void CMS50F37Loader::processBytes(QByteArray bytes) if (!started_import) { // startTimer.singleShot(2000, this, SLOT(requestData())); - qDebug() << "Read:" << len << size << str.join(","); + qDebug() << "pB: Read:" << len << size << str.join(","); } else { - qDebug() << "Import:" << len << size << str.join(","); + qDebug() << "pB: Import:" << len << size << str.join(","); } idx += len; @@ -593,11 +619,12 @@ void CMS50F37Loader::sendCommand(quint8 c) cmd[2] |= (c & 0x7f); QString out; - for (int i=0;i < 9;i++) out += QString().sprintf("%02X ",cmd[i]); + for (int i=0;i < 9;i++) + out += QString().sprintf("%02X ",cmd[i]); qDebug() << "Write:" << out; if (serial.write((char *)cmd, 9) == -1) { - qDebug() << "Couldn't write data reset bytes to CMS50"; + qDebug() << "Couldn't write data bytes to CMS50F"; } } @@ -608,11 +635,12 @@ void CMS50F37Loader::sendCommand(quint8 c, quint8 c2) cmd[4] |= (c2 & 0x7f); QString out; - for (int i=0; i < 9; ++i) out += QString().sprintf("%02X ",cmd[i]); + for (int i=0; i < 9; ++i) + out += QString().sprintf("%02X ",cmd[i]); qDebug() << "Write:" << out; if (serial.write((char *)cmd, 9) == -1) { - qDebug() << "Couldn't write data reset bytes to CMS50"; + qDebug() << "Couldn't write data bytes to CMS50F"; } } @@ -624,11 +652,12 @@ void CMS50F37Loader::eraseSession(int user, int session) cmd[4] = (session & 0x7f) | 0x80; QString out; - for (int i=0; i < 9; ++i) out += QString().sprintf("%02X ",cmd[i]); + for (int i=0; i < 9; ++i) + out += QString().sprintf("%02X ",cmd[i]); qDebug() << "Write:" << out; if (serial.write((char *)cmd, 9) == -1) { - qDebug() << "Couldn't write data reset bytes to CMS50"; + qDebug() << "Couldn't write data bytes to CMS50F"; } int z = timectr; @@ -660,11 +689,12 @@ void CMS50F37Loader::setDeviceID(QString str) cmd[1] = msb | 0x80; QString out; - for (int i=0; i < 9; ++i) out += QString().sprintf("%02X ",cmd[i]); + for (int i=0; i < 9; ++i) + out += QString().sprintf("%02X ",cmd[i]); qDebug() << "Write:" << out; if (serial.write((char *)cmd, 9) == -1) { - qDebug() << "Couldn't write data reset bytes to CMS50"; + qDebug() << "Couldn't write data bytes to CMS50F"; } // Supposed to return 0x04 command, so reset devid.. @@ -691,7 +721,7 @@ void CMS50F37Loader::syncClock() timectr = 0; if (serial.write((char *)datecmd, 9) == -1) { - qDebug() << "Couldn't write data reset bytes to CMS50"; + qDebug() << "Couldn't write data bytes to CMS50F"; } QTime time; @@ -713,7 +743,7 @@ void CMS50F37Loader::syncClock() timectr = 0; if (serial.write((char *)timecmd, 9) == -1) { - qDebug() << "Couldn't write data reset bytes to CMS50"; + qDebug() << "Couldn't write data bytes to CMS50F"; } time.start(); @@ -725,6 +755,7 @@ void CMS50F37Loader::syncClock() void CMS50F37Loader::nextCommand() { + qDebug() << "nextCommand sequence: " << sequence; if (++sequence < cms50_seqlength) { // Send the next command packet in sequence sendCommand(cms50_sequence[sequence]); @@ -742,6 +773,7 @@ void CMS50F37Loader::getSessionData(int session) void CMS50F37Loader::resetDevice() { + qDebug() << "Resetting oximeter"; sendCommand(COMMAND_CMS50_HELLO1); QThread::msleep(100); QApplication::processEvents(); @@ -752,6 +784,7 @@ void CMS50F37Loader::resetDevice() void CMS50F37Loader::requestData() { + qDebug() << "Requesting session data"; sendCommand(COMMAND_GET_SESSION_DATA, selected_session); } From 552d1aef95b6176fda66187703e36ea9abfe4f19 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Thu, 21 Sep 2017 10:48:15 -0400 Subject: [PATCH 11/13] add debug code from other repo - DON'T USE that one! --- .../SleepLib/loader_plugins/md300w1_loader.cpp | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp index 49b0d73a..41703e11 100644 --- a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp @@ -65,7 +65,8 @@ int MD300W1Loader::Open(QString path) { // Only one active Oximeter module at a time, set in preferences - qDebug() << "MD300W1 Loader opening " << path; + qDebug() << "MD300W1 Loader opening " << path; + m_itemCnt = 0; m_itemTotal = 0; @@ -158,13 +159,16 @@ void MD300W1Loader::resetImportTimeout() bool MD300W1Loader::readDATFile(QString path) { QFile file(path); + + qDebug() << "MD300W Loader attempting to read " << path; + if (!file.exists()) { - qDebug() << "File does not exist: " << path; + qDebug() << "File does not exist: " << path; return false; } if (!file.open(QFile::ReadOnly)) { - qDebug() << "Can't open file R/O: " << path; + qDebug() << "Can't open file R/O: " << path; return false; } @@ -194,7 +198,7 @@ bool MD300W1Loader::readDATFile(QString path) if (datetime.date().year() < 2000) datetime = datetime.addYears(100); ts = datetime.toTime_t(); gap = ts - lasttime; - if (gap > 1) { + if (gap > 1) { // always true for first record, b/c time started on 1 Jan 1970 if (gap < 360) { // Less than 5 minutes? Merge session gap--; @@ -203,7 +207,7 @@ bool MD300W1Loader::readDATFile(QString path) oxirec->append(OxiRecord(0,0)); } } else { - // Create a new session + // Create a new session, always for first record qDebug() << "Create session for " << datestr; qDebug() << "Create session for " << datetime.toString("yyyy.MM.dd HH:mm:ss"); oxirec = new QVector; From e5bcb16b30b21488c93878f7e545d91bf0c79d7a Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Thu, 21 Sep 2017 10:50:18 -0400 Subject: [PATCH 12/13] add debug code - trying to fix 'normal average' problem --- sleepyhead/Graphs/gSessionTimesChart.cpp | 10 ++++++---- sleepyhead/Graphs/gSessionTimesChart.h | 4 ++-- sleepyhead/SleepLib/day.cpp | 4 ++-- 3 files changed, 10 insertions(+), 8 deletions(-) diff --git a/sleepyhead/Graphs/gSessionTimesChart.cpp b/sleepyhead/Graphs/gSessionTimesChart.cpp index a44a7302..dd94574c 100644 --- a/sleepyhead/Graphs/gSessionTimesChart.cpp +++ b/sleepyhead/Graphs/gSessionTimesChart.cpp @@ -1138,7 +1138,7 @@ void gAHIChart::preCalc() ahi_wavg = 0; ahi_avg = 0; - calc_cnt = 0; + total_days = 0; total_hours = 0; min_ahi = 99999; max_ahi = -99999; @@ -1187,13 +1187,15 @@ void gAHIChart::customCalc(Day *day, QVector &list) ahi_wavg += ahi_cnt; ahi_avg += ahi_cnt; total_hours += hours; - calc_cnt++; + total_days++; + qDebug() << "Leaving gAHIChart::customCalc - ahi_avg: " << ahi_avg << " total_days: " << total_days ; } void gAHIChart::afterDraw(QPainter & /*painter */, gGraph &graph, QRect rect) { if (totaldays == nousedays) return; //int size = idx_end - idx_start; + qDebug() << "Entering gAHIChart::afterDraw - ahi_avg: " << ahi_avg << " total_days: " << total_days ; bool skip = true; float med = 0; @@ -1211,8 +1213,8 @@ void gAHIChart::afterDraw(QPainter & /*painter */, gGraph &graph, QRect rect) } break; case 2: // avg - if (calc_cnt > 0) { - med = ahi_avg / calc_cnt; + if (total_days > 0) { + med = ahi_avg / total_days; skip = false; } break; diff --git a/sleepyhead/Graphs/gSessionTimesChart.h b/sleepyhead/Graphs/gSessionTimesChart.h index 148db904..6f3c905e 100644 --- a/sleepyhead/Graphs/gSessionTimesChart.h +++ b/sleepyhead/Graphs/gSessionTimesChart.h @@ -412,7 +412,7 @@ public: // layer->total_hours = total_hours; // layer->max_ahi = max_ahi; // layer->min_ahi = min_ahi; -// layer->calc_cnt = calc_cnt; +// layer->total_days = total_days; // layer->ahi_data = ahi_data; } @@ -424,7 +424,7 @@ public: float max_ahi; float min_ahi; - int calc_cnt; + int total_days; QList ahi_data; }; diff --git a/sleepyhead/SleepLib/day.cpp b/sleepyhead/SleepLib/day.cpp index 3a3b4361..47b6e7f0 100644 --- a/sleepyhead/SleepLib/day.cpp +++ b/sleepyhead/SleepLib/day.cpp @@ -150,9 +150,9 @@ QString Day::calcMiddleLabel(ChannelID code) if (c == 0) { return QObject::tr("%1 %2").arg(STR_TR_Median).arg(schema::channel[code].label()); } else if (c == 1) { - return QObject::tr("%1 %2").arg(STR_TR_Average).arg(schema::channel[code].label()); + return QObject::tr("%1 %2").arg(STR_TR_WAvg).arg(schema::channel[code].label()); } else { - return QObject::tr("%1 %2").arg(STR_TR_Average).arg(schema::channel[code].label()); + return QObject::tr("%1 %2").arg(STR_TR_Avg).arg(schema::channel[code].label()); } } QString Day::calcMaxLabel(ChannelID code) From c3fc4bb267bfd133f6f069d02d1f1a9d3639aa00 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Sun, 12 Nov 2017 11:40:02 -0500 Subject: [PATCH 13/13] Use LAST . to find file extension --- .../SleepLib/loader_plugins/cms50_loader.cpp | 8 ++++++-- .../SleepLib/loader_plugins/cms50f37_loader.cpp | 14 +++++++++----- .../SleepLib/loader_plugins/md300w1_loader.cpp | 2 +- 3 files changed, 16 insertions(+), 8 deletions(-) diff --git a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp index 64bb71ad..0eacd747 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp @@ -121,7 +121,7 @@ int CMS50Loader::Open(QString path) setStatus(LIVE); return 1; } - QString ext = path.section(".",1); + QString ext = path.section(".", -1); // find the last '.' if ((ext.compare("spo2", Qt::CaseInsensitive)==0) || (ext.compare("spo", Qt::CaseInsensitive)==0) || (ext.compare("spor", Qt::CaseInsensitive)==0)) { // try to read and process SpoR file.. return readSpoRFile(path) ? 1 : 0; @@ -536,17 +536,21 @@ bool CMS50Loader::readSpoRFile(QString path) { QFile file(path); if (!file.exists()) { + qWarning() << "Can't find the oximeter file: " << path; return false; } if (!file.open(QFile::ReadOnly)) { + qWarning() << "Can't open the oximeter file: " << path; return false; } bool spo2header = false; QString ext = path.section('.', -1); + qDebug() << "Oximeter file extention is " << ext; if (ext.compare("spo2",Qt::CaseInsensitive) == 0) { spo2header = true; + qDebug() << "Oximeter file looks like an SpO2 type" ; } QByteArray data; @@ -591,7 +595,7 @@ bool CMS50Loader::readSpoRFile(QString path) quint32 hour, minute, second; if (data.at(pos) != 1) { - qWarning() << ".spo2 file" << path << "might be a different"; + qWarning() << "oximeter file" << path << "might be odd format"; } // Unknown cruft header... diff --git a/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp b/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp index 6bb1f5c2..84b4be32 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp @@ -72,7 +72,7 @@ bool CMS50F37Loader::openDevice() b = scanDevice("rfcomm", 0, 0); // Linux } if (!b) { - qDebug() << "No oximeter found"; + qWarning() << "No oximeter found"; return false; } } @@ -154,7 +154,7 @@ int CMS50F37Loader::Open(QString path) } else if (path.compare("live") == 0) { return 0; } - QString ext = path.section(".",1); + QString ext = path.section(".", -1); // find the LAST '.' if ((ext.compare("spo2", Qt::CaseInsensitive)==0) || (ext.compare("spo", Qt::CaseInsensitive)==0) || (ext.compare("spor", Qt::CaseInsensitive)==0)) { // try to read and process SpoR file.. return readSpoRFile(path) ? 1 : 0; @@ -815,7 +815,7 @@ void CMS50F37Loader::resetImportTimeout() qDebug() << "Oximeter device stopped transmitting.. Transfer complete"; // We were importing, but now are done if (!finished_import && (started_import && started_reading)) { - qDebug() << "Switching CMS50 back to live mode and finalizing import"; + qDebug() << "Switching CMS50F37 back to live mode and finalizing import"; // Turn back on live streaming so the end of capture can be dealt with @@ -835,7 +835,7 @@ void CMS50F37Loader::resetImportTimeout() return; } - qDebug() << "Should CMS50 resetImportTimeout reach here?"; + qDebug() << "Should CMS50F37 resetImportTimeout reach here?"; // else what??? } cb_reset = imp_callbacks; @@ -853,17 +853,21 @@ bool CMS50F37Loader::readSpoRFile(QString path) { QFile file(path); if (!file.exists()) { + qWarning() << "Can't find the oximeter file: " << path; return false; } if (!file.open(QFile::ReadOnly)) { + qWarning() << "Can't open the oximeter file: " << path; return false; } bool spo2header = false; QString ext = path.section('.', -1); + qDebug() << "Oximeter file extention is " << ext; if (ext.compare("spo2",Qt::CaseInsensitive) == 0) { spo2header = true; + qDebug() << "Oximeter file looks like an SpO2 type" ; } QByteArray data; @@ -897,7 +901,7 @@ bool CMS50F37Loader::readSpoRFile(QString path) QString dstr(dchr); m_startTime = QDateTime::fromString(dstr, "MM/dd/yy HH:mm:ss"); if (m_startTime.date().year() < 2000) { m_startTime = m_startTime.addYears(100); } - } else { + } else { // this should probaly find the most recent SH data day m_startTime = QDateTime(QDate::currentDate(), QTime(0,0,0)); } } else { // !spo2header diff --git a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp index 41703e11..3caede48 100644 --- a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp @@ -95,7 +95,7 @@ int MD300W1Loader::Open(QString path) setStatus(LIVE); return 1; } - QString ext = path.section(".",1); + QString ext = path.section(".", -1); // find the last '.' if (ext.compare("dat", Qt::CaseInsensitive)==0) { // try to read and process SpoR file.. return readDATFile(path) ? 1 : 0;