From ffca449ac216974ecaf334204f359c49d11c0158 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Mon, 26 May 2014 17:37:28 +1000 Subject: [PATCH] Connected LCD components in Live Oximetery, added graph hide ability --- sleepyhead/Graphs/gXAxis.cpp | 2 +- .../SleepLib/loader_plugins/cms50_loader.cpp | 43 ++-- .../SleepLib/loader_plugins/cms50_loader.h | 11 - sleepyhead/SleepLib/machine_loader.cpp | 2 +- sleepyhead/SleepLib/machine_loader.h | 36 ++- sleepyhead/SleepLib/serialoximeter.cpp | 7 +- sleepyhead/SleepLib/serialoximeter.h | 33 ++- sleepyhead/oximeterimport.cpp | 147 ++++++++---- sleepyhead/oximeterimport.h | 11 +- sleepyhead/oximeterimport.ui | 223 +++++++++++++++--- 10 files changed, 376 insertions(+), 139 deletions(-) diff --git a/sleepyhead/Graphs/gXAxis.cpp b/sleepyhead/Graphs/gXAxis.cpp index 918a27be..cf95bdd9 100644 --- a/sleepyhead/Graphs/gXAxis.cpp +++ b/sleepyhead/Graphs/gXAxis.cpp @@ -24,7 +24,7 @@ const quint64 divisors[] = { 15552000000ULL, 7776000000ULL, 5184000000ULL, 2419200000ULL, 1814400000ULL, 1209600000L, 604800000L, 259200000L, 172800000L, 86400000, 2880000, 14400000, 7200000, 3600000, 2700000, 1800000, 1200000, 900000, 600000, 300000, 120000, 60000, 45000, 30000, - 20000, 15000, 10000, 5000, 2000, 1000, 100, 50, 10, 1 + 20000, 15000, 10000, 5000, 2000, 1000, 500, 100, 50, 10, 1 }; const int divcnt = sizeof(divisors) / sizeof(quint64); diff --git a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp index 78195567..0df96030 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp @@ -93,7 +93,7 @@ int CMS50Loader::Open(QString path, Profile *profile) startImportTimeout(); return 1; } else if (path.compare("live") == 0) { - oxitime = QDateTime::currentDateTime(); + m_startTime = oxitime = QDateTime::currentDateTime(); setStatus(LIVE); return 1; } @@ -111,7 +111,7 @@ void CMS50Loader::processBytes(QByteArray bytes) { // Sync to start of message type we are interested in quint8 c; - quint8 msgcode = (m_status == IMPORTING) ? 0xf0 : 0x80; + quint8 msgcode = 0x80; int idx=0; int bytesread = bytes.size(); @@ -157,17 +157,9 @@ void CMS50Loader::processBytes(QByteArray bytes) int CMS50Loader::doImportMode() { - if (finished_import) { - // CMS50E/F continue streaming after import, CMS50D+ stops dead - // there is a timer running at this stage that will kill the 50D - killTimers(); - closeDevice(); - m_importing = false; - imp_callbacks = 0; - return 0; - } - int hour,minute; int available = buffer.size(); + Q_ASSERT(!finished_import); + int hour,minute; int idx = 0; while (idx < available) { unsigned char c=(unsigned char)buffer.at(idx); @@ -256,16 +248,16 @@ int CMS50Loader::doImportMode() } else if (!started_reading) { // have not got a valid trio yet, skip... idx += 1; } else { - // trio's are over.. finish up. + //Completed + finished_import = true; killTimers(); closeDevice(); - - started_import = false; - started_reading = false; - finished_import = true; m_importing = false; - break; - //Completed + m_status = NEUTRAL; + emit importComplete(this); + return available; + imp_callbacks = cb_reset = 0; + return available; } } } @@ -298,13 +290,6 @@ int CMS50Loader::doLiveMode() oxirec.append(OxiRecord(pulse, spo2)); plethy.append(pwave); - int elapsed = m_time.elapsed(); - - // update the graph plots - if (elapsed > 1000) { - m_time.start(); - emit updateDisplay(this); - } idx += 5; } emit updatePlethy(plethy); @@ -386,7 +371,7 @@ void CMS50Loader::resetImportTimeout() if (imp_callbacks != cb_reset) { // Still receiving data.. reset timer - qDebug() << "Still receiving data in resetImportTimeout()"; + qDebug() << "Still receiving data in resetImportTimeout()" << imp_callbacks << cb_reset; if (resetTimer.isActive()) resetTimer.stop(); @@ -403,9 +388,9 @@ void CMS50Loader::resetImportTimeout() resetDevice(); // Send Reset to CMS50D+ //started_import = false; finished_import = true; - m_streaming=false; + //m_streaming=false; - closeDevice(); + //closeDevice(); //emit transferComplete(); //doImportComplete(); return; diff --git a/sleepyhead/SleepLib/loader_plugins/cms50_loader.h b/sleepyhead/SleepLib/loader_plugins/cms50_loader.h index 238bd1d7..d589fe58 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50_loader.h +++ b/sleepyhead/SleepLib/loader_plugins/cms50_loader.h @@ -17,15 +17,6 @@ const QString cms50_class_name = "CMS50"; const int cms50_data_version = 4; -struct OxiRecord -{ - quint8 pulse; - quint8 spo2; - OxiRecord():pulse(0), spo2(0) {} - OxiRecord(quint8 p, quint8 s): pulse(p), spo2(s) {} - OxiRecord(const OxiRecord & copy) { pulse = copy.pulse; spo2= copy.spo2; } -}; - /*! \class CMS50Loader \brief Bulk Importer for CMS50 SPO2Review format.. Deprecated, as the Oximetry module does a better job @@ -64,8 +55,6 @@ protected: int doImportMode(); int doLiveMode(); - QVector oxirec; - virtual void killTimers(); // Switch CMS50D+ device to live streaming mode diff --git a/sleepyhead/SleepLib/machine_loader.cpp b/sleepyhead/SleepLib/machine_loader.cpp index ceb162f2..07250463 100644 --- a/sleepyhead/SleepLib/machine_loader.cpp +++ b/sleepyhead/SleepLib/machine_loader.cpp @@ -53,7 +53,7 @@ void DestroyLoaders() MachineLoader::MachineLoader() :QObject(nullptr) { - m_importing = m_abort = m_streaming = false; + m_abort = false; m_type = MT_UNKNOWN; m_status = NEUTRAL; } diff --git a/sleepyhead/SleepLib/machine_loader.h b/sleepyhead/SleepLib/machine_loader.h index f7968f1d..9a94beb2 100644 --- a/sleepyhead/SleepLib/machine_loader.h +++ b/sleepyhead/SleepLib/machine_loader.h @@ -49,22 +49,20 @@ class MachineLoader: public QObject virtual const QString &ClassName() = 0; inline MachineType type() { return m_type; } - virtual bool openDevice() { return false; } - virtual void closeDevice() {} - virtual bool scanDevice(QString keyword="", quint16 vendor_id=0, quint16 product_id=0) { - Q_UNUSED(keyword) - Q_UNUSED(vendor_id) - Q_UNUSED(product_id) - return false; - } +// virtual bool openDevice() { return false; } +// virtual void closeDevice() {} +// virtual bool scanDevice(QString keyword="", quint16 vendor_id=0, quint16 product_id=0) { +// Q_UNUSED(keyword) +// Q_UNUSED(vendor_id) +// Q_UNUSED(product_id) +// return false; +// } void queTask(ImportTask * task); //! \brief Process Task list using all available threads. void runTasks(); - inline bool isStreaming() { return m_streaming; } - inline bool isImporting() { return m_importing; } inline bool isAborted() { return m_abort; } void abort() { m_abort = true; } @@ -75,16 +73,16 @@ class MachineLoader: public QObject signals: void updateProgress(int cnt, int total); - void updateDisplay(MachineLoader *); +// void updateDisplay(MachineLoader *); -protected slots: - virtual void dataAvailable() {} - virtual void resetImportTimeout() {} - virtual void startImportTimeout() {} +//protected slots: +// virtual void dataAvailable() {} +// virtual void resetImportTimeout() {} +// virtual void startImportTimeout() {} -protected: - virtual void killTimers(){} - virtual void resetDevice(){} +//protected: +// virtual void killTimers(){} +// virtual void resetDevice(){} protected: //! \brief Contains a list of Machine records known by this loader @@ -97,8 +95,6 @@ protected: int m_currenttask; int m_totaltasks; - bool m_streaming; - bool m_importing; bool m_abort; DeviceStatus m_status; diff --git a/sleepyhead/SleepLib/serialoximeter.cpp b/sleepyhead/SleepLib/serialoximeter.cpp index 2ee22ce6..620cf653 100644 --- a/sleepyhead/SleepLib/serialoximeter.cpp +++ b/sleepyhead/SleepLib/serialoximeter.cpp @@ -124,4 +124,9 @@ void SerialOximeter::dataAvailable() processBytes(bytes); } - +void SerialOximeter::stopRecording() +{ + closeDevice(); + m_status = NEUTRAL; + emit importComplete(this); +} diff --git a/sleepyhead/SleepLib/serialoximeter.h b/sleepyhead/SleepLib/serialoximeter.h index 33210f44..21360c46 100644 --- a/sleepyhead/SleepLib/serialoximeter.h +++ b/sleepyhead/SleepLib/serialoximeter.h @@ -19,11 +19,25 @@ const int START_TIMEOUT = 30000; + +struct OxiRecord +{ + quint8 pulse; + quint8 spo2; + OxiRecord():pulse(0), spo2(0) {} + OxiRecord(quint8 p, quint8 s): pulse(p), spo2(s) {} + OxiRecord(const OxiRecord & copy) { pulse = copy.pulse; spo2= copy.spo2; } +}; + + class SerialOximeter : public MachineLoader { Q_OBJECT public: - SerialOximeter() : MachineLoader() {} + SerialOximeter() : MachineLoader() { + m_importing = m_streaming = false; + m_productID = m_vendorID = 0; + } virtual ~SerialOximeter() {} virtual bool Detect(const QString &path)=0; @@ -40,21 +54,32 @@ public: virtual bool openDevice(); virtual void closeDevice(); + inline bool isStreaming() { return m_streaming; } + inline bool isImporting() { return m_importing; } + + virtual void process() {} virtual Machine *CreateMachine(Profile *profile)=0; + QVector oxirec; + + QDateTime startTime() { return m_startTime; } signals: void noDeviceFound(); void deviceDetected(); void updatePlethy(QByteArray plethy); + void importComplete(SerialOximeter *); protected slots: virtual void dataAvailable(); virtual void resetImportTimeout() {} virtual void startImportTimeout() {} + virtual void stopRecording(); +// virtual void abortTask(); + protected: virtual void processBytes(QByteArray buffer) { Q_UNUSED(buffer) } @@ -68,8 +93,14 @@ protected: QTimer startTimer; QTimer resetTimer; + QDateTime m_startTime; + quint16 m_productID; quint16 m_vendorID; + + bool m_streaming; + bool m_importing; + }; #endif // SERIALOXIMETER_H diff --git a/sleepyhead/oximeterimport.cpp b/sleepyhead/oximeterimport.cpp index 1daea759..992688f0 100644 --- a/sleepyhead/oximeterimport.cpp +++ b/sleepyhead/oximeterimport.cpp @@ -66,6 +66,8 @@ OximeterImport::OximeterImport(QWidget *parent) : dummyday = nullptr; session = nullptr; ELplethy = nullptr; + + pulse = spo2 = -1; } OximeterImport::~OximeterImport() @@ -205,10 +207,13 @@ void OximeterImport::on_directImportButton_clicked() // Can't abort this bit or the oximeter will get confused... ui->cancelButton->setVisible(false); - while (oximodule->isImporting() && !oximodule->isAborted()) { - QThread::msleep(50); - QApplication::processEvents(); - } + + connect(oximodule, SIGNAL(importComplete(SerialOximeter*)), this, SLOT(finishedImport(SerialOximeter*))); +} + +void OximeterImport::finishedImport(SerialOximeter * oxi) +{ + disconnect(ui->stopButton, SIGNAL(clicked()), this, SLOT(finishedImport())); ui->cancelButton->setVisible(true); updateStatus(tr("Oximeter import completed.. Processing data")); oximodule->process(); @@ -221,7 +226,6 @@ void OximeterImport::on_directImportButton_clicked() ui->calendarWidget->setMaximumDate(PROFILE.LastDay()); on_calendarWidget_clicked(PROFILE.LastDay()); - } void OximeterImport::doUpdateProgress(int v, int t) @@ -296,7 +300,7 @@ void OximeterImport::on_liveImportButton_clicked() connect(oximodule, SIGNAL(updatePlethy(QByteArray)), this, SLOT(on_updatePlethy(QByteArray))); ui->liveConnectLabel->setText("Live Oximetery Mode"); - liveView->setEmptyText(tr("Recording...")); + liveView->setEmptyText(tr("Starting up...")); ui->progressBar->hide(); liveView->update(); oximodule->Open("live",p_profile); @@ -304,7 +308,7 @@ void OximeterImport::on_liveImportButton_clicked() dummyday = new Day(mach); - quint32 starttime = QDateTime::currentDateTime().toTime_t(); + quint32 starttime = oximodule->startTime().toTime_t(); ti = qint64(starttime) * 1000L; start_ti = ti; @@ -321,32 +325,28 @@ void OximeterImport::on_liveImportButton_clicked() liveView->setDay(dummyday); - QTime time; - time.start(); - while (oximodule->isStreaming() && !oximodule->isAborted()) { - QThread::msleep(50); - QApplication::processEvents(); -// if (time.elapsed() > 100) { - time.restart(); - updateLiveDisplay(); -// } - if (!isVisible()) { - disconnect(oximodule, SIGNAL(updatePlethy(QByteArray)), this, SLOT(on_updatePlethy(QByteArray))); - oximodule->closeDevice(); - delete dummyday; - session = nullptr; - dummyday = nullptr; - return; - } - } + updateTimer.setParent(this); + updateTimer.setInterval(50); + updateTimer.start(); + connect(&updateTimer, SIGNAL(timeout()), this, SLOT(updateLiveDisplay())); + connect(ui->stopButton, SIGNAL(clicked()), this, SLOT(finishedRecording())); +} + +void OximeterImport::finishedRecording() +{ + updateTimer.stop(); + oximodule->closeDevice(); + disconnect(&updateTimer, SIGNAL(timeout()), this, SLOT(updateLiveDisplay())); + disconnect(ui->stopButton, SIGNAL(clicked()), this, SLOT(finishedRecording())); + ui->stopButton->setVisible(false); ui->liveConnectLabel->setText("Live Import Stopped"); liveView->setEmptyText(tr("Live Oximetery Stopped")); updateStatus(tr("Live Oximetery import has been stopped")); - oximodule->closeDevice(); disconnect(oximodule, SIGNAL(updatePlethy(QByteArray)), this, SLOT(on_updatePlethy(QByteArray))); +// Remember to clear sessionlist before deleting dummyday. or it will destroy session. // delete dummyday; //ui->stackedWidget->setCurrentWidget(ui->syncPage); @@ -356,6 +356,7 @@ void OximeterImport::on_liveImportButton_clicked() ui->calendarWidget->setMaximumDate(PROFILE.LastDay()); plethyGraph->SetMinX(start_ti); + liveView->SetXBounds(start_ti, ti, 0, true); plethyGraph->setBlockZoom(false); @@ -487,36 +488,92 @@ void OximeterImport::updateLiveDisplay() if (!session) { return; } - qint64 sti = ti - 20000; - plethyChart->setMinY(ELplethy->Min()); - plethyChart->setMaxY(ELplethy->Max()); - plethyGraph->SetMinY(ELplethy->Min()); - plethyGraph->SetMaxY(ELplethy->Max()); - plethyGraph->SetMinX(sti); - plethyGraph->SetMaxX(ti); - plethyGraph->setBlockZoom(true); - ELplethy->setLast(ti); - session->really_set_last(ti); + + if (ui->showLiveGraphs->isChecked()) { + qint64 sti = ti - 20000; + plethyChart->setMinY(ELplethy->Min()); + plethyChart->setMaxY(ELplethy->Max()); + plethyGraph->SetMinY(ELplethy->Min()); + plethyGraph->SetMaxY(ELplethy->Max()); + plethyGraph->SetMinX(sti); + plethyGraph->SetMaxX(ti); + plethyGraph->setBlockZoom(true); + ELplethy->setLast(ti); + session->really_set_last(ti); - //liveView->SetXBounds(sti, ti, 0, true); - session->setMin(OXI_Plethy, ELplethy->Min()); - session->setMax(OXI_Plethy, ELplethy->Max()); - session->setLast(OXI_Plethy, ti); - session->setCount(OXI_Plethy, session->count(OXI_Plethy)); + //liveView->SetXBounds(sti, ti, 0, true); + session->setMin(OXI_Plethy, ELplethy->Min()); + session->setMax(OXI_Plethy, ELplethy->Max()); + session->setLast(OXI_Plethy, ti); + session->setCount(OXI_Plethy, ELplethy->count()); - for (int i = 0; i < liveView->size(); i++) { - (*liveView)[i]->SetXBounds(sti, ti); + for (int i = 0; i < liveView->size(); i++) { + (*liveView)[i]->SetXBounds(sti, ti); + } + + liveView->updateScale(); + liveView->redraw(); } - liveView->updateScale(); - liveView->timedRedraw(25); + int size = oximodule->oxirec.size(); + if (size > 0) { + int i = oximodule->startTime().secsTo(QDateTime::currentDateTime()); + int seconds = i % 60; + int minutes = (i / 60) % 60; + int hours = i / 3600; + + size--; + + bool datagood = oximodule->oxirec[size].pulse > 0; + + QString STR_recording = tr("Recording..."); + QString STR_nofinger = tr("Finger not detected"); + + if (datagood & (pulse <= 0)) { + updateStatus(STR_recording); + liveView->setEmptyText(STR_recording); + if (!ui->showLiveGraphs->isChecked()) { + liveView->redraw(); + } + } else if (!datagood & (pulse != 0)) { + updateStatus(STR_nofinger); + liveView->setEmptyText(STR_nofinger); + if (!ui->showLiveGraphs->isChecked()) { + liveView->redraw(); + } + } + pulse = oximodule->oxirec[size].pulse; + spo2 = oximodule->oxirec[size].spo2; + if (pulse > 0) { + ui->pulseDisplay->display(QString().sprintf("%3i", pulse)); + } else { + ui->pulseDisplay->display("---"); + } + if (spo2 > 0) { + ui->spo2Display->display(QString().sprintf("%2i", spo2)); + } else { + ui->spo2Display->display("--"); + } + + ui->lcdDuration->display(QString().sprintf("%02i:%02i:%02i",hours, minutes, seconds)); + + } } void OximeterImport::on_cancelButton_clicked() { + if (oximodule && oximodule->isStreaming()) { + oximodule->closeDevice(); + } reject(); } + +void OximeterImport::on_showLiveGraphs_clicked(bool checked) +{ + plethyGraph->setVisible(checked); + liveView->redraw(); +} diff --git a/sleepyhead/oximeterimport.h b/sleepyhead/oximeterimport.h index ebd823eb..7c8f30c7 100644 --- a/sleepyhead/oximeterimport.h +++ b/sleepyhead/oximeterimport.h @@ -53,13 +53,17 @@ private slots: void on_cancelButton_clicked(); + void on_showLiveGraphs_clicked(bool checked); + protected slots: void on_updatePlethy(QByteArray plethy); + void finishedRecording(); + void finishedImport(SerialOximeter*); + void updateLiveDisplay(); protected: SerialOximeter * detectOximeter(); void updateStatus(QString msg); - void updateLiveDisplay(); private: Ui::OximeterImport *ui; @@ -72,6 +76,11 @@ private: SessionBar * sessbar; EventList * ELplethy; qint64 start_ti, ti; + QTimer updateTimer; + + int pulse; + int spo2; + }; #endif // OXIMETERIMPORT_H diff --git a/sleepyhead/oximeterimport.ui b/sleepyhead/oximeterimport.ui index 76761420..ff2b1a22 100644 --- a/sleepyhead/oximeterimport.ui +++ b/sleepyhead/oximeterimport.ui @@ -6,7 +6,7 @@ 0 0 - 851 + 915 615 @@ -620,7 +620,7 @@ - 4 + 3 @@ -895,6 +895,16 @@ p, li { white-space: pre-wrap; } + + + + Press Start to commence recording + + + Qt::AlignCenter + + + @@ -910,28 +920,69 @@ p, li { white-space: pre-wrap; } QFrame::Raised + + 4 + + + 0 + + + 0 + + + 0 + + + 0 + - - - Press Start to commence recording - - - - - - - Qt::Horizontal - - - - 40 - 20 - - - - - - + + + + + Show Live Graphs + + + true + + + + + + + Qt::Horizontal + + + + 40 + 20 + + + + + + + + Duration + + + + + + + + 200 + 0 + + + + QFrame::NoFrame + + + 8 + + + @@ -940,12 +991,67 @@ p, li { white-space: pre-wrap; } - + + + + 0 + 0 + + + + + 60 + 50 + + + + + + + + + 0 + 106 + 255 + + + + + + + + + 0 + 106 + 255 + + + + + + + + + 127 + 127 + 127 + + + + + + + + Qt::LeftToRight + + + QFrame::NoFrame + + + 2 + + - - - - @@ -954,7 +1060,66 @@ p, li { white-space: pre-wrap; } - + + + + 0 + 0 + + + + + 80 + 50 + + + + + + + + + 255 + 0 + 0 + + + + + + + + + 255 + 0 + 0 + + + + + + + + + 159 + 0 + 85 + + + + + + + + QFrame::NoFrame + + + false + + + 3 + +