From a01395e267b2e6b45af7f103345c4dc1480c41ba Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Wed, 28 May 2014 19:35:21 +1000 Subject: [PATCH] More oximetry work, plus added MD300W1 oximeter importer --- .../SleepLib/loader_plugins/cms50_loader.cpp | 73 +++-- .../SleepLib/loader_plugins/cms50_loader.h | 1 - .../loader_plugins/md300w1_loader.cpp | 266 ++++++++++++++++++ .../SleepLib/loader_plugins/md300w1_loader.h | 91 ++++++ sleepyhead/SleepLib/serialoximeter.cpp | 10 + sleepyhead/SleepLib/serialoximeter.h | 11 +- sleepyhead/main.cpp | 2 + sleepyhead/oximeterimport.cpp | 135 +++++++-- sleepyhead/oximeterimport.h | 10 + sleepyhead/oximeterimport.ui | 176 ++++++++++-- sleepyhead/sleepyhead.pro | 6 +- sleepyhead/statistics.cpp | 40 ++- 12 files changed, 721 insertions(+), 100 deletions(-) create mode 100644 sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp create mode 100644 sleepyhead/SleepLib/loader_plugins/md300w1_loader.h diff --git a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp index d704e7a4..e4fd08f7 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp @@ -48,6 +48,8 @@ CMS50Loader::CMS50Loader() m_vendorID = 0x10c4; m_productID = 0xea60; + oxirec = nullptr; + startTimer.setParent(this); resetTimer.setParent(this); @@ -84,6 +86,10 @@ int CMS50Loader::Open(QString path, Profile *profile) m_time.start(); + if (oxirec) { + trashRecords(); + } + // Cheating using path for two serial oximetry modes if (path.compare("import") == 0) { @@ -93,7 +99,12 @@ int CMS50Loader::Open(QString path, Profile *profile) startImportTimeout(); return 1; } else if (path.compare("live") == 0) { - m_startTime = oxitime = QDateTime::currentDateTime(); + + m_startTime = QDateTime::currentDateTime(); + + oxirec = new QVector; + oxisessions[m_startTime] = oxirec; + setStatus(LIVE); return 1; } @@ -164,10 +175,12 @@ int CMS50Loader::doImportMode() while (idx < available) { unsigned char c=(unsigned char)buffer.at(idx); if (!started_import) { + // TODO: Check there might be length data after the 3 headers trios.. if (c != 0xf2) { // If not started, continue scanning for a valie header. idx++; continue; } else { + received_bytes=0; hour=(unsigned char)buffer.at(idx + 1) & 0x7f; @@ -188,8 +201,8 @@ int CMS50Loader::doImportMode() killTimers(); qDebug() << "Getting ready for import"; - oxirec.clear(); - oxirec.reserve(10000); + oxirec = new QVector; + oxirec->reserve(30000); QDate oda=QDate::currentDate(); QTime oti=QTime(hour,minute); // Only CMS50E/F's have a realtime clock. CMS50D+ will set this to midnight @@ -202,12 +215,10 @@ int CMS50Loader::doImportMode() oda = oda.addDays(-1); } - m_startTime = oxitime = QDateTime(oda,oti); + m_startTime = QDateTime(oda,oti); - // Convert it to UTC - oxitime = oxitime.toTimeSpec(Qt::UTC); - - qDebug() << "Session start (according to CMS50)" << oxitime<< hex << buffer.at(idx + 1) << buffer.at(idx + 2) << ":" << dec << hour << minute ; + oxisessions[m_startTime] = oxirec; + qDebug() << "Session start (according to CMS50)" << m_startTime << hex << buffer.at(idx + 1) << buffer.at(idx + 2) << ":" << dec << hour << minute ; cb_reset = 1; @@ -232,10 +243,10 @@ int CMS50Loader::doImportMode() return idx; } - quint8 pulse=(unsigned char)buffer.at(idx + 1) & 0x7f | ((c & 1) << 7); + quint8 pulse=(unsigned char)((buffer.at(idx + 1) & 0x7f) | ((c & 1) << 7)); quint8 spo2=(unsigned char)buffer.at(idx + 2) & 0xff; - oxirec.append(OxiRecord(pulse,spo2)); + oxirec->append(OxiRecord(pulse,spo2)); received_bytes+=3; // TODO: Store the data to the session @@ -273,6 +284,8 @@ int CMS50Loader::doImportMode() int CMS50Loader::doLiveMode() { + Q_ASSERT(oxirec != nullptr); + int available = buffer.size(); int idx = 0; @@ -287,7 +300,7 @@ int CMS50Loader::doLiveMode() int pulse=((unsigned char)buffer.at(idx + 3) & 0x7f) | ((pbeat & 0x40) << 1); int spo2=(unsigned char)buffer.at(idx + 4) & 0x7f; - oxirec.append(OxiRecord(pulse, spo2)); + oxirec->append(OxiRecord(pulse, spo2)); plethy.append(pwave); idx += 5; @@ -407,11 +420,13 @@ void CMS50Loader::resetImportTimeout() bool CMS50Loader::readSpoRFile(QString path) { QFile file(path); - if (!file.exists() || !file.isReadable()) { + if (!file.exists()) { return false; } - file.open(QFile::ReadOnly); + if (!file.open(QFile::ReadOnly)) { + return false; + } QByteArray data; @@ -421,6 +436,9 @@ bool CMS50Loader::readSpoRFile(QString path) // position data stream starts at int pos = ((unsigned char)data.at(1) << 8) | (unsigned char)data.at(0); + // next is 0x0002 + // followed by 16bit duration in seconds + // Read date and time (it's a 16bit charset) char dchr[20]; int j = 0; @@ -430,9 +448,11 @@ bool CMS50Loader::readSpoRFile(QString path) dchr[j] = 0; QString dstr(dchr); - oxitime = QDateTime::fromString(dstr, "MM/dd/yy HH:mm:ss"); + m_startTime = QDateTime::fromString(dstr, "MM/dd/yy HH:mm:ss"); + if (m_startTime.date().year() < 2000) { m_startTime = m_startTime.addYears(100); } - if (oxitime.date().year() < 2000) { oxitime = oxitime.addYears(100); } + oxirec = new QVector; + oxisessions[m_startTime] = oxirec; unsigned char o2, pr; @@ -440,7 +460,7 @@ bool CMS50Loader::readSpoRFile(QString path) for (int i = pos; i < size - 2;) { o2 = (unsigned char)(data.at(i + 1)); pr = (unsigned char)(data.at(i + 0)); - oxirec.append(OxiRecord(pr, o2)); + oxirec->append(OxiRecord(pr, o2)); i += 2; } @@ -483,21 +503,14 @@ Machine *CMS50Loader::CreateMachine(Profile *profile) void CMS50Loader::process() { - int size=oxirec.size(); - if (size<10) - return; + // Just clean up any extra crap before oximeterimport parses the oxirecords.. + return; +// if (!oxirec) +// return; +// int size=oxirec->size(); +// if (size<10) +// return; - -// EventList *PULSE=new EventList(EVL_Event); -// EventList *SPO2=new EventList(EVL_Event); - - -// quint64 ti = oxitime.toMSecsSinceEpoch(); - -// for (int i=0; i < size; ++i) { -// //PULSE->AddWaveform -// } -// qDebug() << "Processing" << oxirec.size() << "oximetry records"; } diff --git a/sleepyhead/SleepLib/loader_plugins/cms50_loader.h b/sleepyhead/SleepLib/loader_plugins/cms50_loader.h index 5d1c8fa0..0a377a7c 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50_loader.h +++ b/sleepyhead/SleepLib/loader_plugins/cms50_loader.h @@ -77,7 +77,6 @@ protected: bool finished_import; bool started_reading; bool cms50dplus; - QDateTime oxitime; int cb_reset,imp_callbacks; diff --git a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp new file mode 100644 index 00000000..74d9de1e --- /dev/null +++ b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp @@ -0,0 +1,266 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * SleepLib ChoiceMMed MD300W1 Oximeter Loader Implementation + * + * Copyright (c) 2011-2014 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 Linux + * distribution for more details. */ + +//******************************************************************************************** +/// IMPORTANT!!! +//******************************************************************************************** +// Please INCREMENT the md300w1_data_version in md300w1_loader.h when making changes to this +// loader that change loader behaviour or modify channels. +//******************************************************************************************** + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +using namespace std; + +#include "md300w1_loader.h" +#include "SleepLib/machine.h" +#include "SleepLib/session.h" + +extern QProgressBar *qprogress; + +MD300W1Loader::MD300W1Loader() +{ + m_type = MT_OXIMETER; + m_abort = false; + m_streaming = false; + m_importing = false; + imp_callbacks = 0; + + // have no idea.. assuming it's another CP2102 USB UART, which won't help detection :/ + m_vendorID = 0; + m_productID = 0; + + startTimer.setParent(this); + resetTimer.setParent(this); + +} + +MD300W1Loader::~MD300W1Loader() +{ +} + +bool MD300W1Loader::Detect(const QString &path) +{ + Q_UNUSED(path); + return false; +} + +int MD300W1Loader::Open(QString path, Profile *profile) +{ + + // Only one active Oximeter module at a time, set in preferences + Q_UNUSED(profile) + + m_itemCnt = 0; + m_itemTotal = 0; + + m_abort = false; + m_importing = false; + + started_import = false; + started_reading = false; + finished_import = false; + + imp_callbacks = 0; + cb_reset = 0; + + m_time.start(); + + // Cheating using path for two serial oximetry modes + + if (path.compare("import") == 0) { + setStatus(IMPORTING); + + startTimer.stop(); + startImportTimeout(); + return 1; + } else if (path.compare("live") == 0) { + m_startTime = oxitime = QDateTime::currentDateTime(); + setStatus(LIVE); + return 1; + } + QString ext = path.section(".",1); + if (ext.compare("dat", Qt::CaseInsensitive)==0) { + // try to read and process SpoR file.. + return readDATFile(path) ? 1 : 0; + } + + return 0; +} + + +void MD300W1Loader::processBytes(QByteArray bytes) +{ + return; +} + +int MD300W1Loader::doImportMode() +{ + return 0; +} + +int MD300W1Loader::doLiveMode() +{ + return 0; +} + +// Switch MD300W1 device to live streaming mode +void MD300W1Loader::resetDevice() +{ +} + +// Switch MD300W1 device to record transmission mode +void MD300W1Loader::requestData() +{ +} + +void MD300W1Loader::killTimers() +{ + startTimer.stop(); + resetTimer.stop(); +} + +void MD300W1Loader::startImportTimeout() +{ + return; +} + +void MD300W1Loader::resetImportTimeout() +{ + return; +} + + +// MedView .dat file (ChoiceMMed MD300B, MD300KI, MD300I, MD300W1, MD300C318, MD2000A) +// Format: +// Bytes 0 (1 2) +// id n +// n*11 0 1 2 3 4 5 6 7 8 9 10 +// 0 0 id yr mm dd hh mm ss o2 pulse +// report title etc. + +bool MD300W1Loader::readDATFile(QString path) +{ + QFile file(path); + if (!file.exists()) { + return false; + } + + if (!file.open(QFile::ReadOnly)) { + return false; + } + + QByteArray data; + + data = file.readAll(); + long size = data.size(); + + // Number of records + int n = ((unsigned char)data.at(2) << 8) | (unsigned char)data.at(1); + + unsigned char o2, pr; + + qint32 lasttime=0, ts=0; + int gap; + for (int pos = 0; pos < n; ++pos) { + int i = 3 + (pos * 11); + QString datestr = QString().sprintf("%02d/%02d/%02d %02d:%02d:%02d",(unsigned char)data.at(i+4),(unsigned char)data.at(i+5),(unsigned char)data.at(i+3),(unsigned char)data.at(i+6),(unsigned char)data.at(i+7),(unsigned char)data.at(i+8)); + QDateTime datetime = QDateTime::fromString(datestr,"MM/dd/yy HH:mm:ss"); + if (datetime.date().year() < 2000) datetime = datetime.addYears(100); + ts = datetime.toTime_t(); + gap = ts - lasttime; + if (gap > 1) { + if (gap < 360) { + // Less than 5 minutes? Merge session + gap--; + // fill with zeroes + for (int j = 0; j < gap; j++) { + oxirec->append(OxiRecord(0,0)); + } + } else { + // Create a new session + oxirec = new QVector; + oxisessions[datetime] = oxirec; + } + } + + pr=(unsigned char)(data.at(i+10)); + o2=(unsigned char)(data.at(i+9)); + + oxirec->append(OxiRecord(pr, o2)); + + lasttime = ts; + } + + // processing gets done later + return true; +} + +Machine *MD300W1Loader::CreateMachine(Profile *profile) +{ + if (!profile) { + return nullptr; + } + + // NOTE: This only allows for one MD300W1 machine per profile.. + // Upgrading their oximeter will use this same record.. + + QList ml = profile->GetMachines(MT_OXIMETER); + + for (QList::iterator i = ml.begin(); i != ml.end(); i++) { + if ((*i)->GetClass() == md300w1_class_name) { + return (*i); + break; + } + } + + qDebug() << "Create MD300W1 Machine Record"; + + Machine *m = new Oximeter(profile, 0); + m->SetClass(md300w1_class_name); + m->properties[STR_PROP_Brand] = "ChoiceMMed"; + m->properties[STR_PROP_Model] = "MD300W1"; + m->properties[STR_PROP_DataVersion] = QString::number(md300w1_data_version); + + profile->AddMachine(m); + QString path = "{" + STR_GEN_DataFolder + "}/" + m->GetClass() + "_" + m->hexid() + "/"; + m->properties[STR_PROP_Path] = path; + + return m; +} + +void MD300W1Loader::process() +{ +} + + +static bool MD300W1_initialized = false; + +void MD300W1Loader::Register() +{ + if (MD300W1_initialized) { return; } + + qDebug() << "Registering MD300W1Loader"; + RegisterLoader(new MD300W1Loader()); + MD300W1_initialized = true; +} + diff --git a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.h b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.h new file mode 100644 index 00000000..fa8ceb10 --- /dev/null +++ b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.h @@ -0,0 +1,91 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * SleepLib ChoiceMMed MD300W1 Oximeter Loader Header + * + * Copyright (c) 2011-2014 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 Linux + * distribution for more details. */ + +#ifndef MD300W1LOADER_H +#define MD300W1LOADER_H + +#include "SleepLib/serialoximeter.h" + +const QString md300w1_class_name = "MD300W1"; +const int md300w1_data_version = 1; + + +/*! \class MD300W1Loader + \brief Importer for ChoiceMMed MD300W1 data format.. + */ +class MD300W1Loader : public SerialOximeter +{ +Q_OBJECT + public: + + + MD300W1Loader(); + virtual ~MD300W1Loader(); + + virtual bool Detect(const QString &path); + virtual int Open(QString path, Profile *profile); + + static void Register(); + + virtual int Version() { return md300w1_data_version; } + virtual const QString &ClassName() { return md300w1_class_name; } + + Machine *CreateMachine(Profile *profile); + + virtual void process(); + + virtual bool isStartTimeValid() { return true; } + +protected slots: + virtual void resetImportTimeout(); + virtual void startImportTimeout(); + +protected: + + bool readDATFile(QString path); + virtual void processBytes(QByteArray bytes); + + int doImportMode(); + int doLiveMode(); + + virtual void killTimers(); + + // Switch MD300W1 device to live streaming mode + virtual void resetDevice(); + + // Switch MD300W1 device to record transmission mode + void requestData(); + + private: + EventList *PULSE; + EventList *SPO2; + + QTime m_time; + + QByteArray buffer; + + bool started_import; + bool finished_import; + bool started_reading; + QDateTime oxitime; + + int cb_reset,imp_callbacks; + + int received_bytes; + + int m_itemCnt; + int m_itemTotal; + + +}; + + +#endif // MD300W1LOADER_H diff --git a/sleepyhead/SleepLib/serialoximeter.cpp b/sleepyhead/SleepLib/serialoximeter.cpp index 620cf653..87fd4e87 100644 --- a/sleepyhead/SleepLib/serialoximeter.cpp +++ b/sleepyhead/SleepLib/serialoximeter.cpp @@ -130,3 +130,13 @@ void SerialOximeter::stopRecording() m_status = NEUTRAL; emit importComplete(this); } + +void SerialOximeter::trashRecords() +{ + QMap *>::iterator it; + for (it = oxisessions.begin(); it != oxisessions.end(); ++it) { + delete it.value(); + } + oxisessions.clear(); + oxirec = nullptr; +} diff --git a/sleepyhead/SleepLib/serialoximeter.h b/sleepyhead/SleepLib/serialoximeter.h index c90b02e3..00fa702b 100644 --- a/sleepyhead/SleepLib/serialoximeter.h +++ b/sleepyhead/SleepLib/serialoximeter.h @@ -62,11 +62,20 @@ public: virtual Machine *CreateMachine(Profile *profile)=0; - QVector oxirec; + // available sessions + QMap *> oxisessions; + + // current session + QVector * oxirec; QDateTime startTime() { return m_startTime; } + void setStartTime(QDateTime datetime) { m_startTime = datetime; } virtual bool isStartTimeValid() { return true; } + virtual qint64 importResolution() { return 1000; } + virtual qint64 liveResolution() { return 20; } + + void trashRecords(); signals: void noDeviceFound(); diff --git a/sleepyhead/main.cpp b/sleepyhead/main.cpp index a7b800d1..bae31333 100644 --- a/sleepyhead/main.cpp +++ b/sleepyhead/main.cpp @@ -35,6 +35,7 @@ // Gah! I must add the real darn plugin system one day. #include "SleepLib/loader_plugins/prs1_loader.h" #include "SleepLib/loader_plugins/cms50_loader.h" +#include "SleepLib/loader_plugins/md300w1_loader.h" #include "SleepLib/loader_plugins/zeo_loader.h" #include "SleepLib/loader_plugins/somnopose_loader.h" #include "SleepLib/loader_plugins/resmed_loader.h" @@ -249,6 +250,7 @@ retry_directory: IntellipapLoader::Register(); FPIconLoader::Register(); CMS50Loader::Register(); + MD300W1Loader::Register(); //ZEOLoader::Register(); // Use outside of directory importer.. p_pref = new Preferences("Preferences"); diff --git a/sleepyhead/oximeterimport.cpp b/sleepyhead/oximeterimport.cpp index ec3bd53f..ae352d39 100644 --- a/sleepyhead/oximeterimport.cpp +++ b/sleepyhead/oximeterimport.cpp @@ -32,7 +32,9 @@ OximeterImport::OximeterImport(QWidget *parent) : ui->stopButton->setVisible(false); ui->saveButton->setVisible(false); ui->syncButton->setVisible(false); + ui->chooseSessionButton->setVisible(false); + importMode = IM_UNDEFINED; QVBoxLayout * lvlayout = new QVBoxLayout; lvlayout->setMargin(0); @@ -228,7 +230,7 @@ void OximeterImport::on_directImportButton_clicked() ui->connectLabel->setText("

"+tr("%1 device is uploading data...").arg(oximodule->ClassName())+"

"); updateStatus(tr("Please wait until oximeter upload process completes. Do not unplug your oximeter.")); - // Wait for import streaming to finish + importMode = IM_RECORDING; // Can't abort this bit or the oximeter will get confused... ui->cancelButton->setVisible(false); @@ -238,14 +240,18 @@ void OximeterImport::on_directImportButton_clicked() void OximeterImport::finishedImport(SerialOximeter * oxi) { - disconnect(ui->stopButton, SIGNAL(clicked()), this, SLOT(finishedImport())); + Q_UNUSED(oxi); + + connect(oximodule, SIGNAL(importComplete(SerialOximeter*)), this, SLOT(finishedImport(SerialOximeter*))); ui->cancelButton->setVisible(true); - updateStatus(tr("Oximeter import completed.. Processing data")); - oximodule->process(); disconnect(oximodule, SIGNAL(updateProgress(int,int)), this, SLOT(doUpdateProgress(int,int))); + updateStatus(tr("Oximeter import completed..")); - - on_syncButton_clicked(); + if (oximodule->oxisessions.size() > 1) { + chooseSession(); + } else { + on_syncButton_clicked(); + } } void OximeterImport::doUpdateProgress(int v, int t) @@ -260,17 +266,20 @@ void OximeterImport::on_fileImportButton_clicked() { #if QT_VERSION < QT_VERSION_CHECK(5,0,0) - const QString documentsFolder = QDesktopServices::storageLocation( - QDesktopServices::DocumentsLocation); + const QString documentsFolder = QDesktopServices::storageLocation(QDesktopServices::DocumentsLocation); #else const QString documentsFolder = QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation); #endif - QString filename = QFileDialog::getOpenFileName(this, tr("Select a valid oximetry data file"), documentsFolder, "Oximetry Files (*.spo *.spor *.spo2)"); + QString filename = QFileDialog::getOpenFileName(this, tr("Select a valid oximetry data file"), documentsFolder, "Oximetry Files (*.spo *.spor *.dat)"); if (filename.isEmpty()) return; + + // Make sure filename dialog had time to close properly.. + QApplication::processEvents(); + QList loaders = GetOxiLoaders(); bool success = false; @@ -288,8 +297,14 @@ void OximeterImport::on_fileImportButton_clicked() return; } ui->informationButton->setVisible(false); + importMode = IM_FILE; - on_syncButton_clicked(); + + if (oximodule->oxisessions.size() > 1) { + chooseSession(); + } else { + on_syncButton_clicked(); + } } void OximeterImport::on_liveImportButton_clicked() @@ -351,6 +366,9 @@ void OximeterImport::on_liveImportButton_clicked() updateTimer.start(); connect(&updateTimer, SIGNAL(timeout()), this, SLOT(updateLiveDisplay())); connect(ui->stopButton, SIGNAL(clicked()), this, SLOT(finishedRecording())); + + importMode = IM_LIVE; + } void OximeterImport::finishedRecording() @@ -367,17 +385,12 @@ void OximeterImport::finishedRecording() disconnect(oximodule, SIGNAL(updatePlethy(QByteArray)), this, SLOT(on_updatePlethy(QByteArray))); -// delete dummyday; - - //ui->stackedWidget->setCurrentWidget(ui->syncPage); ui->syncButton->setVisible(true); plethyGraph->SetMinX(start_ti); liveView->SetXBounds(start_ti, ti, 0, true); plethyGraph->setBlockZoom(false); - - } void OximeterImport::on_retryButton_clicked() @@ -526,7 +539,7 @@ void OximeterImport::updateLiveDisplay() liveView->redraw(); } - int size = oximodule->oxirec.size(); + int size = oximodule->oxirec->size(); if (size > 0) { int i = oximodule->startTime().secsTo(QDateTime::currentDateTime()); @@ -537,7 +550,7 @@ void OximeterImport::updateLiveDisplay() size--; - bool datagood = oximodule->oxirec[size].pulse > 0; + bool datagood = (*(oximodule->oxirec))[size].pulse > 0; if (datagood & (pulse <= 0)) { @@ -555,8 +568,8 @@ void OximeterImport::updateLiveDisplay() liveView->redraw(); } } - pulse = oximodule->oxirec[size].pulse; - spo2 = oximodule->oxirec[size].spo2; + pulse = (*(oximodule->oxirec))[size].pulse; + spo2 = (*(oximodule->oxirec))[size].spo2; if (pulse > 0) { ui->pulseDisplay->display(QString().sprintf("%3i", pulse)); } else { @@ -578,6 +591,7 @@ void OximeterImport::on_cancelButton_clicked() { if (oximodule && oximodule->isStreaming()) { oximodule->closeDevice(); + oximodule->trashRecords(); } reject(); } @@ -604,11 +618,12 @@ void OximeterImport::on_informationButton_clicked() ui->stackedWidget->setCurrentWidget(ui->welcomePage); ui->nextButton->setVisible(true); ui->informationButton->setVisible(false); - } void OximeterImport::on_syncButton_clicked() { + Q_ASSERT(oximodule != nullptr); + ui->stackedWidget->setCurrentWidget(ui->syncPage); ui->syncButton->setVisible(false); @@ -618,34 +633,45 @@ void OximeterImport::on_syncButton_clicked() ui->calendarWidget->setMaximumDate(PROFILE.LastDay()); on_calendarWidget_clicked(PROFILE.LastDay()); - Q_ASSERT(oximodule != nullptr); ui->radioSyncOximeter->setChecked(true); on_radioSyncOximeter_clicked(); - if (ELplethy != nullptr) { + if (importMode == IM_LIVE) { // Live Recording ui->labelSyncOximeter->setText(tr("I want to use the time my computer recorded for this live oximetry session.")); } else if (!oximodule->isStartTimeValid()) { // Oximeter doesn't provide a clock ui->labelSyncOximeter->setText(tr("I need to set the time manually, because my oximeter doesn't have an internal clock.")); } - - } void OximeterImport::on_saveButton_clicked() { - int size = oximodule->oxirec.size(); + int size = oximodule->oxirec->size(); if (size < 2) { QMessageBox::warning(this, STR_MessageBox_Warning, tr("Not enough recorded oximetry data."), QMessageBox::Ok); return; } if (!oximodule) return; + + QVector * oxirec = nullptr; + + if (!oximodule->oxisessions.contains(oximodule->startTime())) { + QMessageBox::warning(this, STR_MessageBox_Error, tr("Something went wrong getting session data"), QMessageBox::Ok); + reject(); + return; + } + oxirec = oximodule->oxisessions[oximodule->startTime()]; + + + // this can move to SerialOximeter class process function... Machine * mach = oximodule->CreateMachine(p_profile); SessionID sid = ui->dateTimeEdit->dateTime().toUTC().toTime_t(); quint64 start = quint64(sid) * 1000L; + + if (!session) { session = new Session(mach, sid); session->really_set_first(start); @@ -677,8 +703,12 @@ void OximeterImport::on_saveButton_clicked() quint16 lastgoodspo2 = 0; quint64 ti = start; + + + qint64 step = (importMode == IM_LIVE) ? oximodule->liveResolution() : oximodule->importResolution(); + for (int i=1; i < size; ++i) { - OxiRecord * rec = &oximodule->oxirec[i]; + OxiRecord * rec = &(*oxirec)[i]; if (rec->pulse > 0) { if (lastpulse == 0) { @@ -723,9 +753,9 @@ void OximeterImport::on_saveButton_clicked() } lastspo2 = rec->spo2; - ti += 20; + ti += step; } - ti -= 20; + ti -= step; if (lastpulse > 0) { ELpulse->AddEvent(ti, lastpulse); session->setLast(OXI_Pulse, ti); @@ -777,5 +807,54 @@ void OximeterImport::on_saveButton_clicked() ELplethy = nullptr; session = nullptr; + oximodule->trashRecords(); accept(); } + +void OximeterImport::chooseSession() +{ + ui->stackedWidget->setCurrentWidget(ui->chooseSessionPage); + ui->syncButton->setVisible(false); + ui->chooseSessionButton->setVisible(true); + QMap *>::iterator it; + + ui->tableOxiSessions->clearContents(); + int row = 0; + QTableWidgetItem * item; + QVector * oxirec; + + ui->tableOxiSessions->setRowCount(oximodule->oxisessions.size()); + ui->tableOxiSessions->setSelectionBehavior(QAbstractItemView::SelectRows); + + for (it = oximodule->oxisessions.begin(); it != oximodule->oxisessions.end(); ++it) { + const QDateTime & key = it.key(); + oxirec = it.value(); + item = new QTableWidgetItem(key.toString(Qt::ISODate)); + ui->tableOxiSessions->setItem(row, 0, item); + item->setFlags(item->flags() & ~Qt::ItemIsEditable); + + item = new QTableWidgetItem(QString(). sprintf("%lli", oxirec->size() * oximodule->importResolution() / 1000L)); + ui->tableOxiSessions->setItem(row, 1, item); + item->setFlags(item->flags() & ~Qt::ItemIsEditable); + + item = new QTableWidgetItem(tr("CMS50 Session %1").arg(row+1, 0)); + ui->tableOxiSessions->setItem(row, 2, item); + item->setFlags(item->flags() & ~Qt::ItemIsEditable); + + row++; + } + + ui->tableOxiSessions->selectRow(0); +} + +void OximeterImport::on_chooseSessionButton_clicked() +{ + ui->chooseSessionButton->setVisible(false); + + QTableWidgetItem * item = ui->tableOxiSessions->item(ui->tableOxiSessions->currentRow(),0); + + QDateTime datetime = QDateTime::fromString(item->text(), Qt::ISODate); + oximodule->setStartTime(datetime); + + on_syncButton_clicked(); +} diff --git a/sleepyhead/oximeterimport.h b/sleepyhead/oximeterimport.h index 984868f9..d1f45cca 100644 --- a/sleepyhead/oximeterimport.h +++ b/sleepyhead/oximeterimport.h @@ -12,6 +12,10 @@ namespace Ui { class OximeterImport; } +enum OximeterImportMode { + IM_UNDEFINED = 0, IM_LIVE, IM_RECORDING, IM_FILE +}; + class OximeterImport : public QDialog { Q_OBJECT @@ -60,6 +64,10 @@ private slots: void on_saveButton_clicked(); + void chooseSession(); + + void on_chooseSessionButton_clicked(); + protected slots: void on_updatePlethy(QByteArray plethy); void finishedRecording(); @@ -82,6 +90,8 @@ private: EventList * ELplethy; qint64 start_ti, ti; QTimer updateTimer; + OximeterImportMode importMode; + int pulse; int spo2; diff --git a/sleepyhead/oximeterimport.ui b/sleepyhead/oximeterimport.ui index af9f016c..ed4288ee 100644 --- a/sleepyhead/oximeterimport.ui +++ b/sleepyhead/oximeterimport.ui @@ -792,6 +792,19 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc) + + + + Qt::Vertical + + + + 20 + 40 + + + + @@ -810,6 +823,19 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc) + + + + Qt::Vertical + + + + 20 + 40 + + + + @@ -888,6 +914,19 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc) + + + + Qt::Vertical + + + + 20 + 40 + + + + @@ -1017,6 +1056,35 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc) 6 + + + + + 75 + true + + + + Show Live Graphs + + + true + + + + + + + Qt::Horizontal + + + + 40 + 20 + + + + @@ -1053,35 +1121,6 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc) - - - - - 75 - true - - - - Show Live Graphs - - - true - - - - - - - Qt::Horizontal - - - - 40 - 20 - - - - @@ -1251,6 +1290,78 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc) + + + + + + + 0 + 0 + + + + + 21 + + + + Multiple Sessions Detected + + + Qt::AlignCenter + + + + + + + + 0 + 0 + + + + Please choose which one you want to import into SleepyHead + + + Qt::AlignCenter + + + + + + + true + + + QAbstractItemView::SingleSelection + + + QAbstractItemView::SelectRows + + + true + + + + Import Time + + + + + Duration + + + + + Details + + + + + + @@ -1663,6 +1774,13 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc) + + + + &Choose Session + + + diff --git a/sleepyhead/sleepyhead.pro b/sleepyhead/sleepyhead.pro index 43713a2b..67beeb88 100644 --- a/sleepyhead/sleepyhead.pro +++ b/sleepyhead/sleepyhead.pro @@ -144,7 +144,8 @@ SOURCES += \ translation.cpp \ statistics.cpp \ oximeterimport.cpp \ - SleepLib/serialoximeter.cpp + SleepLib/serialoximeter.cpp \ + SleepLib/loader_plugins/md300w1_loader.cpp HEADERS += \ common_gui.h \ @@ -197,7 +198,8 @@ HEADERS += \ translation.h \ statistics.h \ oximeterimport.h \ - SleepLib/serialoximeter.h + SleepLib/serialoximeter.h \ + SleepLib/loader_plugins/md300w1_loader.h FORMS += \ daily.ui \ diff --git a/sleepyhead/statistics.cpp b/sleepyhead/statistics.cpp index b697f54c..a73ad6d4 100644 --- a/sleepyhead/statistics.cpp +++ b/sleepyhead/statistics.cpp @@ -141,12 +141,18 @@ QString htmlHeader() "border-radius:10px;" "-moz-border-radius:10px;" "-webkit-border-radius:10px;" - "width: 95%" + "width: 95%;" + "page-break-after:auto;" + "-fs-table-paginate: paginate;" "}" "tr.datarow:nth-child(even) {" "background-color: #f8f8f8;" "}" - + "table { page-break-after:auto; -fs-table-paginate: paginate; }" + "tr { page-break-inside:avoid; page-break-after:auto }" + "td { page-break-inside:avoid; page-break-after:auto }" + "thead { display:table-header-group; }" + "tfoot { display:table-footer-group; }" "" @@ -610,7 +616,7 @@ QString Statistics::GenerateHTML() int days = PROFILE.countDays(row.type, first, last); skipsection = (days == 0); if (days > 0) { - html+=QString("%3\n"). + html+=QString("%3\n"). arg(heading_color).arg(periods.size()+1).arg(row.src); } continue; @@ -1036,9 +1042,13 @@ QString Statistics::GenerateHTML() /*RXsort=RX_min; RXorder=true; qSort(rxchange.begin(),rxchange.end());*/ + + html += "

"; html += "


"; html += QString(""); //cellpadding=2 cellspacing=0 border=1 - html += ""; + html += ""; + html += ""; + QString extratxt; QString tooltip; @@ -1058,9 +1068,19 @@ QString Statistics::GenerateHTML() html+="\n"; for (int i=0; i < hdrlist.size(); ++i) { - html+=QString(" \n").arg(hdrlist.at(i)); + html+=QString(" \n").arg(hdrlist.at(i)); } html+="\n"; + html += ""; + html += ""; + html += ""; + html += ""; + for (int i = 0; i < rxchange.size(); i++) { RXChange rx = rxchange.at(i); @@ -1185,9 +1205,6 @@ QString Statistics::GenerateHTML() } html += "
" + tr("Changes to Prescription Settings") + "
" + tr("Changes to Prescription Settings") + "
%1%1
"; + html += QString("") + + tr("Efficacy highlighting ignores prescription settings with less than %1 days of recorded data."). + arg(rxthresh) + QString("
"); + + html += "
"; - html += QString("") + - tr("Efficacy highlighting ignores prescription settings with less than %1 days of recorded data.").arg( - rxthresh) + QString("
"); html += "
"; } @@ -1195,7 +1212,9 @@ QString Statistics::GenerateHTML() if (mach.size() > 0) { html += "

"; - html += QString(""); // cellpadding=2 cellspacing=0 border=1 width=90%>"); + html += QString("
"); + + html += ""; html += ""; html += QString("") @@ -1204,6 +1223,9 @@ QString Statistics::GenerateHTML() .arg(STR_TR_Serial) .arg(tr("First Use")) .arg(tr("Last Use")); + + html += ""; + Machine *m; for (int i = 0; i < mach.size(); i++) {
" + tr("Machine Information") + "
%1%2%3%4%5