mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
More oximetry work, plus added MD300W1 oximeter importer
This commit is contained in:
parent
260e83e330
commit
a01395e267
@ -48,6 +48,8 @@ CMS50Loader::CMS50Loader()
|
|||||||
m_vendorID = 0x10c4;
|
m_vendorID = 0x10c4;
|
||||||
m_productID = 0xea60;
|
m_productID = 0xea60;
|
||||||
|
|
||||||
|
oxirec = nullptr;
|
||||||
|
|
||||||
startTimer.setParent(this);
|
startTimer.setParent(this);
|
||||||
resetTimer.setParent(this);
|
resetTimer.setParent(this);
|
||||||
|
|
||||||
@ -84,6 +86,10 @@ int CMS50Loader::Open(QString path, Profile *profile)
|
|||||||
|
|
||||||
m_time.start();
|
m_time.start();
|
||||||
|
|
||||||
|
if (oxirec) {
|
||||||
|
trashRecords();
|
||||||
|
}
|
||||||
|
|
||||||
// Cheating using path for two serial oximetry modes
|
// Cheating using path for two serial oximetry modes
|
||||||
|
|
||||||
if (path.compare("import") == 0) {
|
if (path.compare("import") == 0) {
|
||||||
@ -93,7 +99,12 @@ int CMS50Loader::Open(QString path, Profile *profile)
|
|||||||
startImportTimeout();
|
startImportTimeout();
|
||||||
return 1;
|
return 1;
|
||||||
} else if (path.compare("live") == 0) {
|
} else if (path.compare("live") == 0) {
|
||||||
m_startTime = oxitime = QDateTime::currentDateTime();
|
|
||||||
|
m_startTime = QDateTime::currentDateTime();
|
||||||
|
|
||||||
|
oxirec = new QVector<OxiRecord>;
|
||||||
|
oxisessions[m_startTime] = oxirec;
|
||||||
|
|
||||||
setStatus(LIVE);
|
setStatus(LIVE);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@ -164,10 +175,12 @@ int CMS50Loader::doImportMode()
|
|||||||
while (idx < available) {
|
while (idx < available) {
|
||||||
unsigned char c=(unsigned char)buffer.at(idx);
|
unsigned char c=(unsigned char)buffer.at(idx);
|
||||||
if (!started_import) {
|
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.
|
if (c != 0xf2) { // If not started, continue scanning for a valie header.
|
||||||
idx++;
|
idx++;
|
||||||
continue;
|
continue;
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
received_bytes=0;
|
received_bytes=0;
|
||||||
|
|
||||||
hour=(unsigned char)buffer.at(idx + 1) & 0x7f;
|
hour=(unsigned char)buffer.at(idx + 1) & 0x7f;
|
||||||
@ -188,8 +201,8 @@ int CMS50Loader::doImportMode()
|
|||||||
killTimers();
|
killTimers();
|
||||||
qDebug() << "Getting ready for import";
|
qDebug() << "Getting ready for import";
|
||||||
|
|
||||||
oxirec.clear();
|
oxirec = new QVector<OxiRecord>;
|
||||||
oxirec.reserve(10000);
|
oxirec->reserve(30000);
|
||||||
|
|
||||||
QDate oda=QDate::currentDate();
|
QDate oda=QDate::currentDate();
|
||||||
QTime oti=QTime(hour,minute); // Only CMS50E/F's have a realtime clock. CMS50D+ will set this to midnight
|
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);
|
oda = oda.addDays(-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
m_startTime = oxitime = QDateTime(oda,oti);
|
m_startTime = QDateTime(oda,oti);
|
||||||
|
|
||||||
// Convert it to UTC
|
oxisessions[m_startTime] = oxirec;
|
||||||
oxitime = oxitime.toTimeSpec(Qt::UTC);
|
qDebug() << "Session start (according to CMS50)" << m_startTime << hex << buffer.at(idx + 1) << buffer.at(idx + 2) << ":" << dec << hour << minute ;
|
||||||
|
|
||||||
qDebug() << "Session start (according to CMS50)" << oxitime<< hex << buffer.at(idx + 1) << buffer.at(idx + 2) << ":" << dec << hour << minute ;
|
|
||||||
|
|
||||||
cb_reset = 1;
|
cb_reset = 1;
|
||||||
|
|
||||||
@ -232,10 +243,10 @@ int CMS50Loader::doImportMode()
|
|||||||
return idx;
|
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;
|
quint8 spo2=(unsigned char)buffer.at(idx + 2) & 0xff;
|
||||||
|
|
||||||
oxirec.append(OxiRecord(pulse,spo2));
|
oxirec->append(OxiRecord(pulse,spo2));
|
||||||
received_bytes+=3;
|
received_bytes+=3;
|
||||||
|
|
||||||
// TODO: Store the data to the session
|
// TODO: Store the data to the session
|
||||||
@ -273,6 +284,8 @@ int CMS50Loader::doImportMode()
|
|||||||
|
|
||||||
int CMS50Loader::doLiveMode()
|
int CMS50Loader::doLiveMode()
|
||||||
{
|
{
|
||||||
|
Q_ASSERT(oxirec != nullptr);
|
||||||
|
|
||||||
int available = buffer.size();
|
int available = buffer.size();
|
||||||
int idx = 0;
|
int idx = 0;
|
||||||
|
|
||||||
@ -287,7 +300,7 @@ int CMS50Loader::doLiveMode()
|
|||||||
int pulse=((unsigned char)buffer.at(idx + 3) & 0x7f) | ((pbeat & 0x40) << 1);
|
int pulse=((unsigned char)buffer.at(idx + 3) & 0x7f) | ((pbeat & 0x40) << 1);
|
||||||
int spo2=(unsigned char)buffer.at(idx + 4) & 0x7f;
|
int spo2=(unsigned char)buffer.at(idx + 4) & 0x7f;
|
||||||
|
|
||||||
oxirec.append(OxiRecord(pulse, spo2));
|
oxirec->append(OxiRecord(pulse, spo2));
|
||||||
plethy.append(pwave);
|
plethy.append(pwave);
|
||||||
|
|
||||||
idx += 5;
|
idx += 5;
|
||||||
@ -407,11 +420,13 @@ void CMS50Loader::resetImportTimeout()
|
|||||||
bool CMS50Loader::readSpoRFile(QString path)
|
bool CMS50Loader::readSpoRFile(QString path)
|
||||||
{
|
{
|
||||||
QFile file(path);
|
QFile file(path);
|
||||||
if (!file.exists() || !file.isReadable()) {
|
if (!file.exists()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
file.open(QFile::ReadOnly);
|
if (!file.open(QFile::ReadOnly)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
QByteArray data;
|
QByteArray data;
|
||||||
|
|
||||||
@ -421,6 +436,9 @@ bool CMS50Loader::readSpoRFile(QString path)
|
|||||||
// position data stream starts at
|
// position data stream starts at
|
||||||
int pos = ((unsigned char)data.at(1) << 8) | (unsigned char)data.at(0);
|
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)
|
// Read date and time (it's a 16bit charset)
|
||||||
char dchr[20];
|
char dchr[20];
|
||||||
int j = 0;
|
int j = 0;
|
||||||
@ -430,9 +448,11 @@ bool CMS50Loader::readSpoRFile(QString path)
|
|||||||
|
|
||||||
dchr[j] = 0;
|
dchr[j] = 0;
|
||||||
QString dstr(dchr);
|
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<OxiRecord>;
|
||||||
|
oxisessions[m_startTime] = oxirec;
|
||||||
|
|
||||||
unsigned char o2, pr;
|
unsigned char o2, pr;
|
||||||
|
|
||||||
@ -440,7 +460,7 @@ bool CMS50Loader::readSpoRFile(QString path)
|
|||||||
for (int i = pos; i < size - 2;) {
|
for (int i = pos; i < size - 2;) {
|
||||||
o2 = (unsigned char)(data.at(i + 1));
|
o2 = (unsigned char)(data.at(i + 1));
|
||||||
pr = (unsigned char)(data.at(i + 0));
|
pr = (unsigned char)(data.at(i + 0));
|
||||||
oxirec.append(OxiRecord(pr, o2));
|
oxirec->append(OxiRecord(pr, o2));
|
||||||
i += 2;
|
i += 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -483,21 +503,14 @@ Machine *CMS50Loader::CreateMachine(Profile *profile)
|
|||||||
|
|
||||||
void CMS50Loader::process()
|
void CMS50Loader::process()
|
||||||
{
|
{
|
||||||
int size=oxirec.size();
|
// Just clean up any extra crap before oximeterimport parses the oxirecords..
|
||||||
if (size<10)
|
return;
|
||||||
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";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -77,7 +77,6 @@ protected:
|
|||||||
bool finished_import;
|
bool finished_import;
|
||||||
bool started_reading;
|
bool started_reading;
|
||||||
bool cms50dplus;
|
bool cms50dplus;
|
||||||
QDateTime oxitime;
|
|
||||||
|
|
||||||
int cb_reset,imp_callbacks;
|
int cb_reset,imp_callbacks;
|
||||||
|
|
||||||
|
266
sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp
Normal file
266
sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp
Normal file
@ -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 <jedimark@users.sourceforge.net>
|
||||||
|
*
|
||||||
|
* 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 <QProgressBar>
|
||||||
|
#include <QApplication>
|
||||||
|
#include <QDir>
|
||||||
|
#include <QString>
|
||||||
|
#include <QDateTime>
|
||||||
|
#include <QFile>
|
||||||
|
#include <QDebug>
|
||||||
|
#include <QList>
|
||||||
|
#include <QMessageBox>
|
||||||
|
#include <QLabel>
|
||||||
|
#include <QVBoxLayout>
|
||||||
|
#include <QPushButton>
|
||||||
|
|
||||||
|
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<OxiRecord>;
|
||||||
|
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<Machine *> ml = profile->GetMachines(MT_OXIMETER);
|
||||||
|
|
||||||
|
for (QList<Machine *>::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;
|
||||||
|
}
|
||||||
|
|
91
sleepyhead/SleepLib/loader_plugins/md300w1_loader.h
Normal file
91
sleepyhead/SleepLib/loader_plugins/md300w1_loader.h
Normal file
@ -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 <jedimark@users.sourceforge.net>
|
||||||
|
*
|
||||||
|
* 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
|
@ -130,3 +130,13 @@ void SerialOximeter::stopRecording()
|
|||||||
m_status = NEUTRAL;
|
m_status = NEUTRAL;
|
||||||
emit importComplete(this);
|
emit importComplete(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SerialOximeter::trashRecords()
|
||||||
|
{
|
||||||
|
QMap<QDateTime, QVector<OxiRecord> *>::iterator it;
|
||||||
|
for (it = oxisessions.begin(); it != oxisessions.end(); ++it) {
|
||||||
|
delete it.value();
|
||||||
|
}
|
||||||
|
oxisessions.clear();
|
||||||
|
oxirec = nullptr;
|
||||||
|
}
|
||||||
|
@ -62,11 +62,20 @@ public:
|
|||||||
|
|
||||||
virtual Machine *CreateMachine(Profile *profile)=0;
|
virtual Machine *CreateMachine(Profile *profile)=0;
|
||||||
|
|
||||||
QVector<OxiRecord> oxirec;
|
// available sessions
|
||||||
|
QMap<QDateTime, QVector<OxiRecord> *> oxisessions;
|
||||||
|
|
||||||
|
// current session
|
||||||
|
QVector<OxiRecord> * oxirec;
|
||||||
|
|
||||||
QDateTime startTime() { return m_startTime; }
|
QDateTime startTime() { return m_startTime; }
|
||||||
|
void setStartTime(QDateTime datetime) { m_startTime = datetime; }
|
||||||
virtual bool isStartTimeValid() { return true; }
|
virtual bool isStartTimeValid() { return true; }
|
||||||
|
|
||||||
|
virtual qint64 importResolution() { return 1000; }
|
||||||
|
virtual qint64 liveResolution() { return 20; }
|
||||||
|
|
||||||
|
void trashRecords();
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void noDeviceFound();
|
void noDeviceFound();
|
||||||
|
@ -35,6 +35,7 @@
|
|||||||
// Gah! I must add the real darn plugin system one day.
|
// Gah! I must add the real darn plugin system one day.
|
||||||
#include "SleepLib/loader_plugins/prs1_loader.h"
|
#include "SleepLib/loader_plugins/prs1_loader.h"
|
||||||
#include "SleepLib/loader_plugins/cms50_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/zeo_loader.h"
|
||||||
#include "SleepLib/loader_plugins/somnopose_loader.h"
|
#include "SleepLib/loader_plugins/somnopose_loader.h"
|
||||||
#include "SleepLib/loader_plugins/resmed_loader.h"
|
#include "SleepLib/loader_plugins/resmed_loader.h"
|
||||||
@ -249,6 +250,7 @@ retry_directory:
|
|||||||
IntellipapLoader::Register();
|
IntellipapLoader::Register();
|
||||||
FPIconLoader::Register();
|
FPIconLoader::Register();
|
||||||
CMS50Loader::Register();
|
CMS50Loader::Register();
|
||||||
|
MD300W1Loader::Register();
|
||||||
//ZEOLoader::Register(); // Use outside of directory importer..
|
//ZEOLoader::Register(); // Use outside of directory importer..
|
||||||
|
|
||||||
p_pref = new Preferences("Preferences");
|
p_pref = new Preferences("Preferences");
|
||||||
|
@ -32,7 +32,9 @@ OximeterImport::OximeterImport(QWidget *parent) :
|
|||||||
ui->stopButton->setVisible(false);
|
ui->stopButton->setVisible(false);
|
||||||
ui->saveButton->setVisible(false);
|
ui->saveButton->setVisible(false);
|
||||||
ui->syncButton->setVisible(false);
|
ui->syncButton->setVisible(false);
|
||||||
|
ui->chooseSessionButton->setVisible(false);
|
||||||
|
|
||||||
|
importMode = IM_UNDEFINED;
|
||||||
|
|
||||||
QVBoxLayout * lvlayout = new QVBoxLayout;
|
QVBoxLayout * lvlayout = new QVBoxLayout;
|
||||||
lvlayout->setMargin(0);
|
lvlayout->setMargin(0);
|
||||||
@ -228,7 +230,7 @@ void OximeterImport::on_directImportButton_clicked()
|
|||||||
ui->connectLabel->setText("<h2>"+tr("%1 device is uploading data...").arg(oximodule->ClassName())+"</h2>");
|
ui->connectLabel->setText("<h2>"+tr("%1 device is uploading data...").arg(oximodule->ClassName())+"</h2>");
|
||||||
updateStatus(tr("Please wait until oximeter upload process completes. Do not unplug your oximeter."));
|
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...
|
// Can't abort this bit or the oximeter will get confused...
|
||||||
ui->cancelButton->setVisible(false);
|
ui->cancelButton->setVisible(false);
|
||||||
@ -238,14 +240,18 @@ void OximeterImport::on_directImportButton_clicked()
|
|||||||
|
|
||||||
void OximeterImport::finishedImport(SerialOximeter * oxi)
|
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);
|
ui->cancelButton->setVisible(true);
|
||||||
updateStatus(tr("Oximeter import completed.. Processing data"));
|
|
||||||
oximodule->process();
|
|
||||||
disconnect(oximodule, SIGNAL(updateProgress(int,int)), this, SLOT(doUpdateProgress(int,int)));
|
disconnect(oximodule, SIGNAL(updateProgress(int,int)), this, SLOT(doUpdateProgress(int,int)));
|
||||||
|
updateStatus(tr("Oximeter import completed.."));
|
||||||
|
|
||||||
|
if (oximodule->oxisessions.size() > 1) {
|
||||||
on_syncButton_clicked();
|
chooseSession();
|
||||||
|
} else {
|
||||||
|
on_syncButton_clicked();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void OximeterImport::doUpdateProgress(int v, int t)
|
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)
|
#if QT_VERSION < QT_VERSION_CHECK(5,0,0)
|
||||||
const QString documentsFolder = QDesktopServices::storageLocation(
|
const QString documentsFolder = QDesktopServices::storageLocation(QDesktopServices::DocumentsLocation);
|
||||||
QDesktopServices::DocumentsLocation);
|
|
||||||
#else
|
#else
|
||||||
const QString documentsFolder = QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation);
|
const QString documentsFolder = QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation);
|
||||||
#endif
|
#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())
|
if (filename.isEmpty())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
// Make sure filename dialog had time to close properly..
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
QList<SerialOximeter *> loaders = GetOxiLoaders();
|
QList<SerialOximeter *> loaders = GetOxiLoaders();
|
||||||
|
|
||||||
bool success = false;
|
bool success = false;
|
||||||
@ -288,8 +297,14 @@ void OximeterImport::on_fileImportButton_clicked()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ui->informationButton->setVisible(false);
|
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()
|
void OximeterImport::on_liveImportButton_clicked()
|
||||||
@ -351,6 +366,9 @@ void OximeterImport::on_liveImportButton_clicked()
|
|||||||
updateTimer.start();
|
updateTimer.start();
|
||||||
connect(&updateTimer, SIGNAL(timeout()), this, SLOT(updateLiveDisplay()));
|
connect(&updateTimer, SIGNAL(timeout()), this, SLOT(updateLiveDisplay()));
|
||||||
connect(ui->stopButton, SIGNAL(clicked()), this, SLOT(finishedRecording()));
|
connect(ui->stopButton, SIGNAL(clicked()), this, SLOT(finishedRecording()));
|
||||||
|
|
||||||
|
importMode = IM_LIVE;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OximeterImport::finishedRecording()
|
void OximeterImport::finishedRecording()
|
||||||
@ -367,17 +385,12 @@ void OximeterImport::finishedRecording()
|
|||||||
|
|
||||||
disconnect(oximodule, SIGNAL(updatePlethy(QByteArray)), this, SLOT(on_updatePlethy(QByteArray)));
|
disconnect(oximodule, SIGNAL(updatePlethy(QByteArray)), this, SLOT(on_updatePlethy(QByteArray)));
|
||||||
|
|
||||||
// delete dummyday;
|
|
||||||
|
|
||||||
//ui->stackedWidget->setCurrentWidget(ui->syncPage);
|
|
||||||
ui->syncButton->setVisible(true);
|
ui->syncButton->setVisible(true);
|
||||||
|
|
||||||
plethyGraph->SetMinX(start_ti);
|
plethyGraph->SetMinX(start_ti);
|
||||||
liveView->SetXBounds(start_ti, ti, 0, true);
|
liveView->SetXBounds(start_ti, ti, 0, true);
|
||||||
|
|
||||||
plethyGraph->setBlockZoom(false);
|
plethyGraph->setBlockZoom(false);
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OximeterImport::on_retryButton_clicked()
|
void OximeterImport::on_retryButton_clicked()
|
||||||
@ -526,7 +539,7 @@ void OximeterImport::updateLiveDisplay()
|
|||||||
liveView->redraw();
|
liveView->redraw();
|
||||||
}
|
}
|
||||||
|
|
||||||
int size = oximodule->oxirec.size();
|
int size = oximodule->oxirec->size();
|
||||||
|
|
||||||
if (size > 0) {
|
if (size > 0) {
|
||||||
int i = oximodule->startTime().secsTo(QDateTime::currentDateTime());
|
int i = oximodule->startTime().secsTo(QDateTime::currentDateTime());
|
||||||
@ -537,7 +550,7 @@ void OximeterImport::updateLiveDisplay()
|
|||||||
|
|
||||||
size--;
|
size--;
|
||||||
|
|
||||||
bool datagood = oximodule->oxirec[size].pulse > 0;
|
bool datagood = (*(oximodule->oxirec))[size].pulse > 0;
|
||||||
|
|
||||||
|
|
||||||
if (datagood & (pulse <= 0)) {
|
if (datagood & (pulse <= 0)) {
|
||||||
@ -555,8 +568,8 @@ void OximeterImport::updateLiveDisplay()
|
|||||||
liveView->redraw();
|
liveView->redraw();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pulse = oximodule->oxirec[size].pulse;
|
pulse = (*(oximodule->oxirec))[size].pulse;
|
||||||
spo2 = oximodule->oxirec[size].spo2;
|
spo2 = (*(oximodule->oxirec))[size].spo2;
|
||||||
if (pulse > 0) {
|
if (pulse > 0) {
|
||||||
ui->pulseDisplay->display(QString().sprintf("%3i", pulse));
|
ui->pulseDisplay->display(QString().sprintf("%3i", pulse));
|
||||||
} else {
|
} else {
|
||||||
@ -578,6 +591,7 @@ void OximeterImport::on_cancelButton_clicked()
|
|||||||
{
|
{
|
||||||
if (oximodule && oximodule->isStreaming()) {
|
if (oximodule && oximodule->isStreaming()) {
|
||||||
oximodule->closeDevice();
|
oximodule->closeDevice();
|
||||||
|
oximodule->trashRecords();
|
||||||
}
|
}
|
||||||
reject();
|
reject();
|
||||||
}
|
}
|
||||||
@ -604,11 +618,12 @@ void OximeterImport::on_informationButton_clicked()
|
|||||||
ui->stackedWidget->setCurrentWidget(ui->welcomePage);
|
ui->stackedWidget->setCurrentWidget(ui->welcomePage);
|
||||||
ui->nextButton->setVisible(true);
|
ui->nextButton->setVisible(true);
|
||||||
ui->informationButton->setVisible(false);
|
ui->informationButton->setVisible(false);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OximeterImport::on_syncButton_clicked()
|
void OximeterImport::on_syncButton_clicked()
|
||||||
{
|
{
|
||||||
|
Q_ASSERT(oximodule != nullptr);
|
||||||
|
|
||||||
ui->stackedWidget->setCurrentWidget(ui->syncPage);
|
ui->stackedWidget->setCurrentWidget(ui->syncPage);
|
||||||
|
|
||||||
ui->syncButton->setVisible(false);
|
ui->syncButton->setVisible(false);
|
||||||
@ -618,34 +633,45 @@ void OximeterImport::on_syncButton_clicked()
|
|||||||
ui->calendarWidget->setMaximumDate(PROFILE.LastDay());
|
ui->calendarWidget->setMaximumDate(PROFILE.LastDay());
|
||||||
|
|
||||||
on_calendarWidget_clicked(PROFILE.LastDay());
|
on_calendarWidget_clicked(PROFILE.LastDay());
|
||||||
Q_ASSERT(oximodule != nullptr);
|
|
||||||
|
|
||||||
ui->radioSyncOximeter->setChecked(true);
|
ui->radioSyncOximeter->setChecked(true);
|
||||||
on_radioSyncOximeter_clicked();
|
on_radioSyncOximeter_clicked();
|
||||||
|
|
||||||
if (ELplethy != nullptr) {
|
if (importMode == IM_LIVE) {
|
||||||
// Live Recording
|
// Live Recording
|
||||||
ui->labelSyncOximeter->setText(tr("I want to use the time my computer recorded for this live oximetry session."));
|
ui->labelSyncOximeter->setText(tr("I want to use the time my computer recorded for this live oximetry session."));
|
||||||
} else if (!oximodule->isStartTimeValid()) {
|
} else if (!oximodule->isStartTimeValid()) {
|
||||||
// Oximeter doesn't provide a clock
|
// 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."));
|
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()
|
void OximeterImport::on_saveButton_clicked()
|
||||||
{
|
{
|
||||||
int size = oximodule->oxirec.size();
|
int size = oximodule->oxirec->size();
|
||||||
if (size < 2) {
|
if (size < 2) {
|
||||||
QMessageBox::warning(this, STR_MessageBox_Warning, tr("Not enough recorded oximetry data."), QMessageBox::Ok);
|
QMessageBox::warning(this, STR_MessageBox_Warning, tr("Not enough recorded oximetry data."), QMessageBox::Ok);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!oximodule) return;
|
if (!oximodule) return;
|
||||||
|
|
||||||
|
|
||||||
|
QVector<OxiRecord> * 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);
|
Machine * mach = oximodule->CreateMachine(p_profile);
|
||||||
SessionID sid = ui->dateTimeEdit->dateTime().toUTC().toTime_t();
|
SessionID sid = ui->dateTimeEdit->dateTime().toUTC().toTime_t();
|
||||||
quint64 start = quint64(sid) * 1000L;
|
quint64 start = quint64(sid) * 1000L;
|
||||||
|
|
||||||
|
|
||||||
if (!session) {
|
if (!session) {
|
||||||
session = new Session(mach, sid);
|
session = new Session(mach, sid);
|
||||||
session->really_set_first(start);
|
session->really_set_first(start);
|
||||||
@ -677,8 +703,12 @@ void OximeterImport::on_saveButton_clicked()
|
|||||||
quint16 lastgoodspo2 = 0;
|
quint16 lastgoodspo2 = 0;
|
||||||
|
|
||||||
quint64 ti = start;
|
quint64 ti = start;
|
||||||
|
|
||||||
|
|
||||||
|
qint64 step = (importMode == IM_LIVE) ? oximodule->liveResolution() : oximodule->importResolution();
|
||||||
|
|
||||||
for (int i=1; i < size; ++i) {
|
for (int i=1; i < size; ++i) {
|
||||||
OxiRecord * rec = &oximodule->oxirec[i];
|
OxiRecord * rec = &(*oxirec)[i];
|
||||||
|
|
||||||
if (rec->pulse > 0) {
|
if (rec->pulse > 0) {
|
||||||
if (lastpulse == 0) {
|
if (lastpulse == 0) {
|
||||||
@ -723,9 +753,9 @@ void OximeterImport::on_saveButton_clicked()
|
|||||||
}
|
}
|
||||||
lastspo2 = rec->spo2;
|
lastspo2 = rec->spo2;
|
||||||
|
|
||||||
ti += 20;
|
ti += step;
|
||||||
}
|
}
|
||||||
ti -= 20;
|
ti -= step;
|
||||||
if (lastpulse > 0) {
|
if (lastpulse > 0) {
|
||||||
ELpulse->AddEvent(ti, lastpulse);
|
ELpulse->AddEvent(ti, lastpulse);
|
||||||
session->setLast(OXI_Pulse, ti);
|
session->setLast(OXI_Pulse, ti);
|
||||||
@ -777,5 +807,54 @@ void OximeterImport::on_saveButton_clicked()
|
|||||||
ELplethy = nullptr;
|
ELplethy = nullptr;
|
||||||
session = nullptr;
|
session = nullptr;
|
||||||
|
|
||||||
|
oximodule->trashRecords();
|
||||||
accept();
|
accept();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void OximeterImport::chooseSession()
|
||||||
|
{
|
||||||
|
ui->stackedWidget->setCurrentWidget(ui->chooseSessionPage);
|
||||||
|
ui->syncButton->setVisible(false);
|
||||||
|
ui->chooseSessionButton->setVisible(true);
|
||||||
|
QMap<QDateTime, QVector<OxiRecord> *>::iterator it;
|
||||||
|
|
||||||
|
ui->tableOxiSessions->clearContents();
|
||||||
|
int row = 0;
|
||||||
|
QTableWidgetItem * item;
|
||||||
|
QVector<OxiRecord> * 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();
|
||||||
|
}
|
||||||
|
@ -12,6 +12,10 @@ namespace Ui {
|
|||||||
class OximeterImport;
|
class OximeterImport;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum OximeterImportMode {
|
||||||
|
IM_UNDEFINED = 0, IM_LIVE, IM_RECORDING, IM_FILE
|
||||||
|
};
|
||||||
|
|
||||||
class OximeterImport : public QDialog
|
class OximeterImport : public QDialog
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@ -60,6 +64,10 @@ private slots:
|
|||||||
|
|
||||||
void on_saveButton_clicked();
|
void on_saveButton_clicked();
|
||||||
|
|
||||||
|
void chooseSession();
|
||||||
|
|
||||||
|
void on_chooseSessionButton_clicked();
|
||||||
|
|
||||||
protected slots:
|
protected slots:
|
||||||
void on_updatePlethy(QByteArray plethy);
|
void on_updatePlethy(QByteArray plethy);
|
||||||
void finishedRecording();
|
void finishedRecording();
|
||||||
@ -82,6 +90,8 @@ private:
|
|||||||
EventList * ELplethy;
|
EventList * ELplethy;
|
||||||
qint64 start_ti, ti;
|
qint64 start_ti, ti;
|
||||||
QTimer updateTimer;
|
QTimer updateTimer;
|
||||||
|
OximeterImportMode importMode;
|
||||||
|
|
||||||
|
|
||||||
int pulse;
|
int pulse;
|
||||||
int spo2;
|
int spo2;
|
||||||
|
@ -792,6 +792,19 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc)
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<spacer name="verticalSpacer_5">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>20</width>
|
||||||
|
<height>40</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QLabel" name="label">
|
<widget class="QLabel" name="label">
|
||||||
<property name="font">
|
<property name="font">
|
||||||
@ -810,6 +823,19 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc)
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<spacer name="verticalSpacer_4">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>20</width>
|
||||||
|
<height>40</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="directImportButton">
|
<widget class="QPushButton" name="directImportButton">
|
||||||
<property name="toolTip">
|
<property name="toolTip">
|
||||||
@ -888,6 +914,19 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc)
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<spacer name="verticalSpacer_2">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>20</width>
|
||||||
|
<height>40</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QLabel" name="label_6">
|
<widget class="QLabel" name="label_6">
|
||||||
<property name="font">
|
<property name="font">
|
||||||
@ -1017,6 +1056,35 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc)
|
|||||||
<property name="bottomMargin">
|
<property name="bottomMargin">
|
||||||
<number>6</number>
|
<number>6</number>
|
||||||
</property>
|
</property>
|
||||||
|
<item>
|
||||||
|
<widget class="QCheckBox" name="showLiveGraphs">
|
||||||
|
<property name="font">
|
||||||
|
<font>
|
||||||
|
<weight>75</weight>
|
||||||
|
<bold>true</bold>
|
||||||
|
</font>
|
||||||
|
</property>
|
||||||
|
<property name="text">
|
||||||
|
<string>Show Live Graphs</string>
|
||||||
|
</property>
|
||||||
|
<property name="checked">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<spacer name="horizontalSpacer_7">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Horizontal</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>40</width>
|
||||||
|
<height>20</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout_8">
|
<layout class="QVBoxLayout" name="verticalLayout_8">
|
||||||
<item>
|
<item>
|
||||||
@ -1053,35 +1121,6 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc)
|
|||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
|
||||||
<widget class="QCheckBox" name="showLiveGraphs">
|
|
||||||
<property name="font">
|
|
||||||
<font>
|
|
||||||
<weight>75</weight>
|
|
||||||
<bold>true</bold>
|
|
||||||
</font>
|
|
||||||
</property>
|
|
||||||
<property name="text">
|
|
||||||
<string>Show Live Graphs</string>
|
|
||||||
</property>
|
|
||||||
<property name="checked">
|
|
||||||
<bool>true</bool>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
|
||||||
<spacer name="horizontalSpacer_7">
|
|
||||||
<property name="orientation">
|
|
||||||
<enum>Qt::Horizontal</enum>
|
|
||||||
</property>
|
|
||||||
<property name="sizeHint" stdset="0">
|
|
||||||
<size>
|
|
||||||
<width>40</width>
|
|
||||||
<height>20</height>
|
|
||||||
</size>
|
|
||||||
</property>
|
|
||||||
</spacer>
|
|
||||||
</item>
|
|
||||||
<item>
|
<item>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout_9">
|
<layout class="QVBoxLayout" name="verticalLayout_9">
|
||||||
<item>
|
<item>
|
||||||
@ -1251,6 +1290,78 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc)
|
|||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
|
<widget class="QWidget" name="chooseSessionPage">
|
||||||
|
<layout class="QVBoxLayout" name="verticalLayout_3">
|
||||||
|
<item>
|
||||||
|
<widget class="QLabel" name="liveConnectLabel_2">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Preferred">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="font">
|
||||||
|
<font>
|
||||||
|
<pointsize>21</pointsize>
|
||||||
|
</font>
|
||||||
|
</property>
|
||||||
|
<property name="text">
|
||||||
|
<string>Multiple Sessions Detected</string>
|
||||||
|
</property>
|
||||||
|
<property name="alignment">
|
||||||
|
<set>Qt::AlignCenter</set>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QLabel" name="label_12">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Preferred">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="text">
|
||||||
|
<string>Please choose which one you want to import into SleepyHead</string>
|
||||||
|
</property>
|
||||||
|
<property name="alignment">
|
||||||
|
<set>Qt::AlignCenter</set>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QTableWidget" name="tableOxiSessions">
|
||||||
|
<property name="alternatingRowColors">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
<property name="selectionMode">
|
||||||
|
<enum>QAbstractItemView::SingleSelection</enum>
|
||||||
|
</property>
|
||||||
|
<property name="selectionBehavior">
|
||||||
|
<enum>QAbstractItemView::SelectRows</enum>
|
||||||
|
</property>
|
||||||
|
<attribute name="horizontalHeaderStretchLastSection">
|
||||||
|
<bool>true</bool>
|
||||||
|
</attribute>
|
||||||
|
<column>
|
||||||
|
<property name="text">
|
||||||
|
<string>Import Time</string>
|
||||||
|
</property>
|
||||||
|
</column>
|
||||||
|
<column>
|
||||||
|
<property name="text">
|
||||||
|
<string>Duration</string>
|
||||||
|
</property>
|
||||||
|
</column>
|
||||||
|
<column>
|
||||||
|
<property name="text">
|
||||||
|
<string>Details</string>
|
||||||
|
</property>
|
||||||
|
</column>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
<widget class="QWidget" name="syncPage">
|
<widget class="QWidget" name="syncPage">
|
||||||
<layout class="QVBoxLayout" name="verticalLayout_10">
|
<layout class="QVBoxLayout" name="verticalLayout_10">
|
||||||
<item>
|
<item>
|
||||||
@ -1663,6 +1774,13 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc)
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="chooseSessionButton">
|
||||||
|
<property name="text">
|
||||||
|
<string>&Choose Session</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="stopButton">
|
<widget class="QPushButton" name="stopButton">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
|
@ -144,7 +144,8 @@ SOURCES += \
|
|||||||
translation.cpp \
|
translation.cpp \
|
||||||
statistics.cpp \
|
statistics.cpp \
|
||||||
oximeterimport.cpp \
|
oximeterimport.cpp \
|
||||||
SleepLib/serialoximeter.cpp
|
SleepLib/serialoximeter.cpp \
|
||||||
|
SleepLib/loader_plugins/md300w1_loader.cpp
|
||||||
|
|
||||||
HEADERS += \
|
HEADERS += \
|
||||||
common_gui.h \
|
common_gui.h \
|
||||||
@ -197,7 +198,8 @@ HEADERS += \
|
|||||||
translation.h \
|
translation.h \
|
||||||
statistics.h \
|
statistics.h \
|
||||||
oximeterimport.h \
|
oximeterimport.h \
|
||||||
SleepLib/serialoximeter.h
|
SleepLib/serialoximeter.h \
|
||||||
|
SleepLib/loader_plugins/md300w1_loader.h
|
||||||
|
|
||||||
FORMS += \
|
FORMS += \
|
||||||
daily.ui \
|
daily.ui \
|
||||||
|
@ -141,12 +141,18 @@ QString htmlHeader()
|
|||||||
"border-radius:10px;"
|
"border-radius:10px;"
|
||||||
"-moz-border-radius:10px;"
|
"-moz-border-radius:10px;"
|
||||||
"-webkit-border-radius:10px;"
|
"-webkit-border-radius:10px;"
|
||||||
"width: 95%"
|
"width: 95%;"
|
||||||
|
"page-break-after:auto;"
|
||||||
|
"-fs-table-paginate: paginate;"
|
||||||
"}"
|
"}"
|
||||||
"tr.datarow:nth-child(even) {"
|
"tr.datarow:nth-child(even) {"
|
||||||
"background-color: #f8f8f8;"
|
"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; }"
|
||||||
|
|
||||||
|
|
||||||
"</style>"
|
"</style>"
|
||||||
@ -610,7 +616,7 @@ QString Statistics::GenerateHTML()
|
|||||||
int days = PROFILE.countDays(row.type, first, last);
|
int days = PROFILE.countDays(row.type, first, last);
|
||||||
skipsection = (days == 0);
|
skipsection = (days == 0);
|
||||||
if (days > 0) {
|
if (days > 0) {
|
||||||
html+=QString("<tr bgcolor='%1'><td colspan=%2 align=center><font size=+3>%3</font></td></tr>\n").
|
html+=QString("<tr bgcolor='%1'><th colspan=%2 align=center><font size=+3>%3</font></th></tr>\n").
|
||||||
arg(heading_color).arg(periods.size()+1).arg(row.src);
|
arg(heading_color).arg(periods.size()+1).arg(row.src);
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
@ -1036,9 +1042,13 @@ QString Statistics::GenerateHTML()
|
|||||||
/*RXsort=RX_min;
|
/*RXsort=RX_min;
|
||||||
RXorder=true;
|
RXorder=true;
|
||||||
qSort(rxchange.begin(),rxchange.end());*/
|
qSort(rxchange.begin(),rxchange.end());*/
|
||||||
|
|
||||||
|
html += "<p style=\"page-break-before:always;\"/>";
|
||||||
html += "<div align=center><br/>";
|
html += "<div align=center><br/>";
|
||||||
html += QString("<table class=curved>"); //cellpadding=2 cellspacing=0 border=1
|
html += QString("<table class=curved>"); //cellpadding=2 cellspacing=0 border=1
|
||||||
html += "<tr bgcolor='"+heading_color+"'><td colspan=10 align=center><font size=+3>" + tr("Changes to Prescription Settings") + "</font></td></tr>";
|
html += "<thead>";
|
||||||
|
html += "<tr bgcolor='"+heading_color+"'><th colspan=10 align=center><font size=+3>" + tr("Changes to Prescription Settings") + "</font></th></tr>";
|
||||||
|
|
||||||
QString extratxt;
|
QString extratxt;
|
||||||
|
|
||||||
QString tooltip;
|
QString tooltip;
|
||||||
@ -1058,9 +1068,19 @@ QString Statistics::GenerateHTML()
|
|||||||
|
|
||||||
html+="<tr>\n";
|
html+="<tr>\n";
|
||||||
for (int i=0; i < hdrlist.size(); ++i) {
|
for (int i=0; i < hdrlist.size(); ++i) {
|
||||||
html+=QString(" <td><b>%1</b></td>\n").arg(hdrlist.at(i));
|
html+=QString(" <th align=left><b>%1</b></th>\n").arg(hdrlist.at(i));
|
||||||
}
|
}
|
||||||
html+="</tr>\n";
|
html+="</tr>\n";
|
||||||
|
html += "</thead>";
|
||||||
|
html += "<tfoot>";
|
||||||
|
html += "<tr><td colspan=10 align=center>";
|
||||||
|
html += QString("<i>") +
|
||||||
|
tr("Efficacy highlighting ignores prescription settings with less than %1 days of recorded data.").
|
||||||
|
arg(rxthresh) + QString("</i><br/>");
|
||||||
|
|
||||||
|
html += "</td></tr>";
|
||||||
|
html += "</tfoot>";
|
||||||
|
|
||||||
|
|
||||||
for (int i = 0; i < rxchange.size(); i++) {
|
for (int i = 0; i < rxchange.size(); i++) {
|
||||||
RXChange rx = rxchange.at(i);
|
RXChange rx = rxchange.at(i);
|
||||||
@ -1185,9 +1205,6 @@ QString Statistics::GenerateHTML()
|
|||||||
}
|
}
|
||||||
|
|
||||||
html += "</table>";
|
html += "</table>";
|
||||||
html += QString("<i>") +
|
|
||||||
tr("Efficacy highlighting ignores prescription settings with less than %1 days of recorded data.").arg(
|
|
||||||
rxthresh) + QString("</i><br/>");
|
|
||||||
html += "</div>";
|
html += "</div>";
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -1195,7 +1212,9 @@ QString Statistics::GenerateHTML()
|
|||||||
if (mach.size() > 0) {
|
if (mach.size() > 0) {
|
||||||
html += "<div align=center><br/>";
|
html += "<div align=center><br/>";
|
||||||
|
|
||||||
html += QString("<table class=curved>"); // cellpadding=2 cellspacing=0 border=1 width=90%>");
|
html += QString("<table class=curved style=\"page-break-before:auto;\">");
|
||||||
|
|
||||||
|
html += "<thead>";
|
||||||
html += "<tr bgcolor='"+heading_color+"'><td colspan=5 align=center><font size=+3>" + tr("Machine Information") + "</font></td></tr>";
|
html += "<tr bgcolor='"+heading_color+"'><td colspan=5 align=center><font size=+3>" + tr("Machine Information") + "</font></td></tr>";
|
||||||
|
|
||||||
html += QString("<tr><td><b>%1</b></td><td><b>%2</b></td><td><b>%3</b></td><td><b>%4</b></td><td><b>%5</b></td></tr>")
|
html += QString("<tr><td><b>%1</b></td><td><b>%2</b></td><td><b>%3</b></td><td><b>%4</b></td><td><b>%5</b></td></tr>")
|
||||||
@ -1204,6 +1223,9 @@ QString Statistics::GenerateHTML()
|
|||||||
.arg(STR_TR_Serial)
|
.arg(STR_TR_Serial)
|
||||||
.arg(tr("First Use"))
|
.arg(tr("First Use"))
|
||||||
.arg(tr("Last Use"));
|
.arg(tr("Last Use"));
|
||||||
|
|
||||||
|
html += "</thead>";
|
||||||
|
|
||||||
Machine *m;
|
Machine *m;
|
||||||
|
|
||||||
for (int i = 0; i < mach.size(); i++) {
|
for (int i = 0; i < mach.size(); i++) {
|
||||||
|
Loading…
Reference in New Issue
Block a user