2018-04-27 04:29:03 +00:00
|
|
|
|
/* SleepLib ChoiceMMed MD300W1 Oximeter Loader Implementation
|
2014-05-28 09:35:21 +00:00
|
|
|
|
*
|
2024-01-13 20:27:48 +00:00
|
|
|
|
* Copyright (c) 2019-2024 The OSCAR Team
|
2024-02-01 00:14:19 +00:00
|
|
|
|
* Copyright (C) 2011-2018 Mark Watkins
|
2014-05-28 09:35:21 +00:00
|
|
|
|
*
|
|
|
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
2018-06-04 20:48:38 +00:00
|
|
|
|
* License. See the file COPYING in the main directory of the source code
|
|
|
|
|
* for more details. */
|
2014-05-28 09:35:21 +00:00
|
|
|
|
|
|
|
|
|
//********************************************************************************************
|
|
|
|
|
/// IMPORTANT!!!
|
|
|
|
|
//********************************************************************************************
|
|
|
|
|
// Please INCREMENT the md300w1_data_version in md300w1_loader.h when making changes to this
|
|
|
|
|
// loader that change loader behaviour or modify channels.
|
|
|
|
|
//********************************************************************************************
|
|
|
|
|
|
2019-05-09 17:05:53 +00:00
|
|
|
|
// #include <QProgressBar>
|
2014-05-28 09:35:21 +00:00
|
|
|
|
#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"
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
|
int MD300W1Loader::Open(const QString & path)
|
2014-05-28 09:35:21 +00:00
|
|
|
|
{
|
|
|
|
|
// Only one active Oximeter module at a time, set in preferences
|
|
|
|
|
|
2017-07-31 20:58:36 +00:00
|
|
|
|
qDebug() << "MD300W1 Loader opening " << path;
|
2018-04-03 03:51:53 +00:00
|
|
|
|
|
2014-05-28 09:35:21 +00:00
|
|
|
|
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;
|
|
|
|
|
}
|
2017-11-12 16:40:02 +00:00
|
|
|
|
QString ext = path.section(".", -1); // find the last '.'
|
2014-05-28 09:35:21 +00:00
|
|
|
|
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)
|
|
|
|
|
{
|
2014-06-02 11:22:45 +00:00
|
|
|
|
Q_UNUSED(bytes);
|
2014-05-28 09:35:21 +00:00
|
|
|
|
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.
|
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
|
bool MD300W1Loader::readDATFile(const QString & path)
|
2014-05-28 09:35:21 +00:00
|
|
|
|
{
|
|
|
|
|
QFile file(path);
|
2017-09-21 14:48:15 +00:00
|
|
|
|
|
|
|
|
|
qDebug() << "MD300W Loader attempting to read " << path;
|
|
|
|
|
|
2014-05-28 09:35:21 +00:00
|
|
|
|
if (!file.exists()) {
|
2017-09-21 14:48:15 +00:00
|
|
|
|
qDebug() << "File does not exist: " << path;
|
2019-01-31 15:47:52 +00:00
|
|
|
|
QMessageBox::warning(nullptr, STR_MessageBox_Error, "<h2>"+tr("Could not find the oximeter file:")+"<br/><br/>"+path+"</h2>");
|
2014-05-28 09:35:21 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!file.open(QFile::ReadOnly)) {
|
2017-09-21 14:48:15 +00:00
|
|
|
|
qDebug() << "Can't open file R/O: " << path;
|
2019-01-31 15:47:52 +00:00
|
|
|
|
QMessageBox::warning(nullptr, STR_MessageBox_Error, "<h2>"+tr("Could not open the oximeter file:")+"<br/><br/>"+path+"</h2>");
|
2014-05-28 09:35:21 +00:00
|
|
|
|
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);
|
|
|
|
|
|
2014-06-02 06:40:00 +00:00
|
|
|
|
// CHECKME:
|
|
|
|
|
if (size < (n*11)+3) {
|
|
|
|
|
qDebug() << "Short MD300W1 .dat file" << path;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-05-28 09:35:21 +00:00
|
|
|
|
unsigned char o2, pr;
|
|
|
|
|
|
|
|
|
|
qint32 lasttime=0, ts=0;
|
|
|
|
|
int gap;
|
|
|
|
|
for (int pos = 0; pos < n; ++pos) {
|
|
|
|
|
int i = 3 + (pos * 11);
|
2023-03-01 12:51:45 +00:00
|
|
|
|
QString datestr = QString::asprintf("%02d/%02d/%02d %02d:%02d:%02d",
|
2019-05-09 17:05:53 +00:00
|
|
|
|
(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));
|
2021-10-09 15:47:45 +00:00
|
|
|
|
// Ensure date is correct first to ensure DST is handled correctly
|
|
|
|
|
QDate date = QDate::fromString(datestr.left(8),"MM/dd/yy");
|
|
|
|
|
QTime time = QTime::fromString(datestr.right(8),"HH:mm:ss");
|
|
|
|
|
if (date.year() < 2000) {
|
|
|
|
|
date = date.addYears(100);
|
|
|
|
|
}
|
|
|
|
|
QDateTime datetime = QDateTime(date, time);
|
2014-05-28 09:35:21 +00:00
|
|
|
|
ts = datetime.toTime_t();
|
|
|
|
|
gap = ts - lasttime;
|
2017-09-21 14:48:15 +00:00
|
|
|
|
if (gap > 1) { // always true for first record, b/c time started on 1 Jan 1970
|
2014-05-28 09:35:21 +00:00
|
|
|
|
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 {
|
2017-09-21 14:48:15 +00:00
|
|
|
|
// Create a new session, always for first record
|
2019-05-09 17:05:53 +00:00
|
|
|
|
qDebug() << "Create session for " << datetime.toString("yyyy-MMM-dd HH:mm:ss");
|
2014-05-28 09:35:21 +00:00
|
|
|
|
oxirec = new QVector<OxiRecord>;
|
|
|
|
|
oxisessions[datetime] = oxirec;
|
2017-07-31 20:58:36 +00:00
|
|
|
|
m_startTime = datetime; // works for single session files...
|
2014-05-28 09:35:21 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MD300W1Loader::process()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static bool MD300W1_initialized = false;
|
|
|
|
|
|
|
|
|
|
void MD300W1Loader::Register()
|
|
|
|
|
{
|
|
|
|
|
if (MD300W1_initialized) { return; }
|
|
|
|
|
|
|
|
|
|
qDebug() << "Registering MD300W1Loader";
|
|
|
|
|
RegisterLoader(new MD300W1Loader());
|
|
|
|
|
MD300W1_initialized = true;
|
|
|
|
|
}
|
|
|
|
|
|