mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 02:30:44 +00:00
Added support for importing .spo2 files from newer CMS50F oximeters, plus a little SleepyHead history
This commit is contained in:
parent
1735d9b51e
commit
fe184a1f4a
9
history/README
Normal file
9
history/README
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
This folder contains the python/GTK script I (jedimark) wrote that eventually turned into SleepyHead
|
||||||
|
|
||||||
|
I can't honestly tell you if this is the latest version, because the computer I originally wrote this on died.
|
||||||
|
|
||||||
|
I put it here for project history, reference (and humour at how sad it is.)
|
||||||
|
|
||||||
|
It requires matplotlib and pygtk to run.. Probably pytz too, I honestly can't remember if it needed any other libs.
|
||||||
|
|
||||||
|
I've mostly forgotten python since then.
|
1783
history/cpap.py
Normal file
1783
history/cpap.py
Normal file
File diff suppressed because it is too large
Load Diff
@ -20,6 +20,7 @@
|
|||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QDataStream>
|
||||||
#include <QDateTime>
|
#include <QDateTime>
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
@ -47,6 +48,7 @@ CMS50Loader::CMS50Loader()
|
|||||||
|
|
||||||
m_vendorID = 0x10c4;
|
m_vendorID = 0x10c4;
|
||||||
m_productID = 0xea60;
|
m_productID = 0xea60;
|
||||||
|
cms50dplus = false;
|
||||||
|
|
||||||
oxirec = nullptr;
|
oxirec = nullptr;
|
||||||
|
|
||||||
@ -124,7 +126,7 @@ int CMS50Loader::Open(QString path, Profile *profile)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
QString ext = path.section(".",1);
|
QString ext = path.section(".",1);
|
||||||
if ((ext.compare("spo", Qt::CaseInsensitive)==0) || (ext.compare("spor", Qt::CaseInsensitive)==0)) {
|
if ((ext.compare("spo2", Qt::CaseInsensitive)==0) || (ext.compare("spo", Qt::CaseInsensitive)==0) || (ext.compare("spor", Qt::CaseInsensitive)==0)) {
|
||||||
// try to read and process SpoR file..
|
// try to read and process SpoR file..
|
||||||
return readSpoRFile(path) ? 1 : 0;
|
return readSpoRFile(path) ? 1 : 0;
|
||||||
}
|
}
|
||||||
@ -471,18 +473,30 @@ bool CMS50Loader::readSpoRFile(QString path)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool spo2header = false;
|
||||||
|
QString ext = path.section('.', -1);
|
||||||
|
if (ext.compare("spo2",Qt::CaseInsensitive) == 0) {
|
||||||
|
spo2header = true;
|
||||||
|
}
|
||||||
|
|
||||||
QByteArray data;
|
QByteArray data;
|
||||||
|
|
||||||
data = file.readAll();
|
data = file.readAll();
|
||||||
long size = data.size();
|
QDataStream in(data);
|
||||||
|
in.setByteOrder(QDataStream::LittleEndian);
|
||||||
|
quint16 pos;
|
||||||
|
in >> pos;
|
||||||
|
|
||||||
// position data stream starts at
|
in.skipRawData(pos - 2);
|
||||||
int pos = ((unsigned char)data.at(1) << 8) | (unsigned char)data.at(0);
|
|
||||||
|
|
||||||
|
//long size = data.size();
|
||||||
|
|
||||||
|
if (!spo2header) {
|
||||||
// next is 0x0002
|
// next is 0x0002
|
||||||
// followed by 16bit duration in seconds
|
// 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;
|
||||||
for (int i = 0; i < 18 * 2; i += 2) {
|
for (int i = 0; i < 18 * 2; i += 2) {
|
||||||
@ -490,9 +504,37 @@ bool CMS50Loader::readSpoRFile(QString path)
|
|||||||
}
|
}
|
||||||
|
|
||||||
dchr[j] = 0;
|
dchr[j] = 0;
|
||||||
|
if (dchr[0]) {
|
||||||
QString dstr(dchr);
|
QString dstr(dchr);
|
||||||
m_startTime = 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 (m_startTime.date().year() < 2000) { m_startTime = m_startTime.addYears(100); }
|
||||||
|
} else {
|
||||||
|
m_startTime = QDateTime(QDate::currentDate(), QTime(0,0,0));
|
||||||
|
}
|
||||||
|
} else { // !spo2header
|
||||||
|
|
||||||
|
quint32 samples = 0; // number of samples
|
||||||
|
|
||||||
|
quint32 year, month, day;
|
||||||
|
quint32 hour, minute, second;
|
||||||
|
|
||||||
|
if (data.at(pos) != 1) {
|
||||||
|
qWarning() << ".spo2 file" << path << "might be a different";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unknown cruft...
|
||||||
|
in.skipRawData(200);
|
||||||
|
|
||||||
|
in >> year >> month >> day;
|
||||||
|
in >> hour >> minute >> second;
|
||||||
|
|
||||||
|
m_startTime = QDateTime(QDate(year, month, day), QTime(hour, minute, second));
|
||||||
|
|
||||||
|
// ignoring it for now
|
||||||
|
pos += 0x1c + 200;
|
||||||
|
|
||||||
|
in >> samples;
|
||||||
|
}
|
||||||
|
|
||||||
oxirec = new QVector<OxiRecord>;
|
oxirec = new QVector<OxiRecord>;
|
||||||
oxisessions[m_startTime] = oxirec;
|
oxisessions[m_startTime] = oxirec;
|
||||||
@ -500,12 +542,19 @@ bool CMS50Loader::readSpoRFile(QString path)
|
|||||||
unsigned char o2, pr;
|
unsigned char o2, pr;
|
||||||
|
|
||||||
// Read all Pulse and SPO2 data
|
// Read all Pulse and SPO2 data
|
||||||
for (int i = pos; i < size - 2;) {
|
do {
|
||||||
o2 = (unsigned char)(data.at(i + 1));
|
in >> o2;
|
||||||
pr = (unsigned char)(data.at(i + 0));
|
in >> pr;
|
||||||
oxirec->append(OxiRecord(pr, o2));
|
oxirec->append(OxiRecord(pr, o2));
|
||||||
i += 2;
|
} while (!in.atEnd());
|
||||||
}
|
|
||||||
|
|
||||||
|
// 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));
|
||||||
|
// i += 2;
|
||||||
|
// }
|
||||||
|
|
||||||
// processing gets done later
|
// processing gets done later
|
||||||
return true;
|
return true;
|
||||||
|
@ -177,7 +177,7 @@ void init()
|
|||||||
QObject::tr("VS"), STR_UNIT_EventsPerHour, DEFAULT, QColor("red")));
|
QObject::tr("VS"), STR_UNIT_EventsPerHour, DEFAULT, QColor("red")));
|
||||||
|
|
||||||
schema::channel.add(GRP_CPAP, new Channel(CPAP_VSnore2 = 0x1008, FLAG, SESSION, "VSnore2",
|
schema::channel.add(GRP_CPAP, new Channel(CPAP_VSnore2 = 0x1008, FLAG, SESSION, "VSnore2",
|
||||||
QObject::tr("Vibratory Snore"),
|
QObject::tr("Vibratory Snore (VS2) "),
|
||||||
QObject::tr("A vibratory snore as detcted by a System One machine"),
|
QObject::tr("A vibratory snore as detcted by a System One machine"),
|
||||||
QObject::tr("VS2"), STR_UNIT_EventsPerHour, DEFAULT, QColor("red")));
|
QObject::tr("VS2"), STR_UNIT_EventsPerHour, DEFAULT, QColor("red")));
|
||||||
|
|
||||||
|
@ -271,7 +271,7 @@ void OximeterImport::on_fileImportButton_clicked()
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
QString filename = QFileDialog::getOpenFileName(nullptr , tr("Select a valid oximetry data file"), documentsFolder, tr("Oximetry Files (*.spo *.spor *.dat)"));
|
QString filename = QFileDialog::getOpenFileName(nullptr , tr("Select a valid oximetry data file"), documentsFolder, tr("Oximetry Files (*.spo *.spor *.spo2 *.dat)"));
|
||||||
|
|
||||||
if (filename.isEmpty())
|
if (filename.isEmpty())
|
||||||
return;
|
return;
|
||||||
|
Loading…
Reference in New Issue
Block a user