CMS50i file import support, CMS50D+ swap pulse and spo2, added AHI to machine list

This commit is contained in:
Mark Watkins 2014-07-31 03:14:28 +10:00
parent e956038571
commit 22ea3868f1
12 changed files with 151 additions and 20 deletions

View File

@ -861,12 +861,14 @@ bool gGraphView::renderGraphs(QPainter &painter)
if ((m_graphs.size() > 1) && m_showsplitter) { if ((m_graphs.size() > 1) && m_showsplitter) {
// draw the splitter handle // draw the splitter handle
painter.setPen(QColor(158,158,158,255));
painter.drawLine(0, py + h, w, py + h);
painter.setPen(QColor(220, 220, 220, 255)); painter.setPen(QColor(220, 220, 220, 255));
painter.drawLine(0, py + h, w, py + h);
painter.setPen(QColor(158,158,158,255));
painter.drawLine(0, py + h + 1, w, py + h + 1); painter.drawLine(0, py + h + 1, w, py + h + 1);
painter.setPen(QColor(140, 140, 140, 255)); painter.setPen(QColor(240, 240, 240, 255));
painter.drawLine(0, py + h + 2, w, py + h + 2); painter.drawLine(0, py + h + 2, w, py + h + 2);
} }
} }

View File

@ -337,7 +337,7 @@ void gXAxis::paint(QPainter &painter, gGraph &w, const QRegion &region)
} }
if (usepixmap && !m_image.isNull()) { if (usepixmap && !m_image.isNull()) {
painter.drawImage(QPoint(left - 20, top + height - m_image.height() + 4), m_image); painter.drawImage(QPoint(left - 20, top + height - m_image.height() + 5), m_image);
} }
} }

View File

@ -276,6 +276,7 @@ QString STR_TR_Mode;
QString STR_TR_Model; QString STR_TR_Model;
QString STR_TR_Brand; QString STR_TR_Brand;
QString STR_TR_Serial; QString STR_TR_Serial;
QString STR_TR_Series;
QString STR_TR_Machine; QString STR_TR_Machine;
QString STR_TR_Channel; QString STR_TR_Channel;
QString STR_TR_Settings; QString STR_TR_Settings;
@ -455,6 +456,7 @@ void initializeStrings()
STR_TR_Model = QObject::tr("Model"); STR_TR_Model = QObject::tr("Model");
STR_TR_Brand = QObject::tr("Brand"); STR_TR_Brand = QObject::tr("Brand");
STR_TR_Serial = QObject::tr("Serial"); STR_TR_Serial = QObject::tr("Serial");
STR_TR_Series = QObject::tr("Series");
STR_TR_Machine = QObject::tr("Machine"); STR_TR_Machine = QObject::tr("Machine");
STR_TR_Channel = QObject::tr("Channel"); STR_TR_Channel = QObject::tr("Channel");
STR_TR_Settings = QObject::tr("Settings"); STR_TR_Settings = QObject::tr("Settings");

View File

@ -256,6 +256,7 @@ extern QString STR_TR_SleepyHead;
extern QString STR_TR_Mode; extern QString STR_TR_Mode;
extern QString STR_TR_Model; extern QString STR_TR_Model;
extern QString STR_TR_Brand; extern QString STR_TR_Brand;
extern QString STR_TR_Series;
extern QString STR_TR_Serial; extern QString STR_TR_Serial;
extern QString STR_TR_Machine; extern QString STR_TR_Machine;
extern QString STR_TR_Channel; extern QString STR_TR_Channel;

View File

@ -55,6 +55,21 @@ void Day::AddSession(Session *s)
sessions.push_back(s); sessions.push_back(s);
} }
EventDataType Day::countInsideSpan(ChannelID span, ChannelID code)
{
QList<Session *>::iterator end = sessions.end();
int count = 0;
for (QList<Session *>::iterator it = sessions.begin(); it != end; ++it) {
Session &sess = *(*it);
if (sess.enabled()) {
count += sess.countInsideSpan(span, code);
}
}
return count;
}
EventDataType Day::lookupValue(ChannelID code, qint64 time) EventDataType Day::lookupValue(ChannelID code, qint64 time)
{ {
QList<Session *>::iterator end = sessions.end(); QList<Session *>::iterator end = sessions.end();

View File

@ -106,6 +106,10 @@ class Day
//! \brief Returns the value for Channel code at a given time //! \brief Returns the value for Channel code at a given time
EventDataType lookupValue(ChannelID code, qint64 time); EventDataType lookupValue(ChannelID code, qint64 time);
//! \brief Returns the count of code events inside span flag event durations
EventDataType countInsideSpan(ChannelID span, ChannelID code);
//! \brief Returns the first session time of this day //! \brief Returns the first session time of this day
qint64 first(); qint64 first();

View File

@ -548,6 +548,7 @@ bool CMS50Loader::readSpoRFile(QString path)
QByteArray data; QByteArray data;
qint64 filesize = file.size();
data = file.readAll(); data = file.readAll();
QDataStream in(data); QDataStream in(data);
in.setByteOrder(QDataStream::LittleEndian); in.setByteOrder(QDataStream::LittleEndian);
@ -557,6 +558,7 @@ bool CMS50Loader::readSpoRFile(QString path)
in.skipRawData(pos - 2); in.skipRawData(pos - 2);
//long size = data.size(); //long size = data.size();
int bytes_per_record = 2;
if (!spo2header) { if (!spo2header) {
// next is 0x0002 // next is 0x0002
@ -589,7 +591,7 @@ bool CMS50Loader::readSpoRFile(QString path)
qWarning() << ".spo2 file" << path << "might be a different"; qWarning() << ".spo2 file" << path << "might be a different";
} }
// Unknown cruft... // Unknown cruft header...
in.skipRawData(200); in.skipRawData(200);
in >> year >> month >> day; in >> year >> month >> day;
@ -601,18 +603,40 @@ bool CMS50Loader::readSpoRFile(QString path)
pos += 0x1c + 200; pos += 0x1c + 200;
in >> samples; in >> samples;
int remainder = filesize - pos;
bytes_per_record = remainder / samples;
qDebug() << samples << "samples of" << bytes_per_record << "bytes each";
// CMS50I .spo2 data have 4 digits, a 16bit, followed by spo2 then pulse
} }
oxirec = new QVector<OxiRecord>; oxirec = new QVector<OxiRecord>;
oxisessions[m_startTime] = oxirec; oxisessions[m_startTime] = oxirec;
unsigned char o2, pr; unsigned char o2, pr;
quint16 un;
// Read all Pulse and SPO2 data // Read all Pulse and SPO2 data
do { do {
if (bytes_per_record > 2) {
in >> un;
}
in >> o2; in >> o2;
in >> pr; in >> pr;
oxirec->append(OxiRecord(pr, o2));
if ((o2 == 0x7f) && (pr == 0xff)) {
o2 = pr = 0;
un = 0;
}
if (spo2header) {
oxirec->append(OxiRecord(pr, o2));
} else {
oxirec->append(OxiRecord(o2, pr));
}
} while (!in.atEnd()); } while (!in.atEnd());

View File

@ -511,12 +511,16 @@ int IntellipapLoader::Open(QString path)
if (sid) { if (sid) {
sess = Sessions[sid]; sess = Sessions[sid];
if (!sess) continue; if (!sess) continue;
quint64 first = qint64(sid) * 1000L; // quint64 first = qint64(sid) * 1000L;
quint64 last = qint64(SessionEnd[i]) * 1000L; quint64 last = qint64(SessionEnd[i]) * 1000L;
if (sess->last() > 0) { if (sess->last() > 0) {
sess->really_set_last(last);
sess->settings[CPAP_PresReliefType] = (PRTypes)PR_SMARTFLEX; sess->settings[CPAP_PresReliefType] = (PRTypes)PR_SMARTFLEX;
sess->settings[CPAP_PresReliefSet] = smartflex; sess->settings[CPAP_PresReliefSet] = smartflex;

View File

@ -1308,6 +1308,61 @@ bool Session::channelExists(ChannelID id)
return true; return true;
} }
EventDataType Session::countInsideSpan(ChannelID span, ChannelID code)
{
// TODO: Cache me!
QHash<ChannelID, QVector<EventList *> >::iterator j = eventlist.find(span);
if (j == eventlist.end()) {
return 0;
}
QVector<EventList *> &evec = j.value();
qint64 t1,t2;
int evec_size=evec.size();
QList<qint64> start;
QList<qint64> end;
// Simplify the span flags to start and end times list
for (int el = 0; el < evec_size; ++el) {
EventList &ev = *evec[el];
for (quint32 i=0; i < ev.count(); ++i) {
end.push_back(t2=ev.time(i));
start.push_back(t2 - (qint64(ev.data(i)) * 1000L));
}
}
j = eventlist.find(code);
if (j == eventlist.end()) {
return 0;
}
QVector<EventList *> &evec2 = j.value();
evec_size=evec2.size();
int count = 0;
int spans = start.size();
for (int el = 0; el < evec_size; ++el) {
EventList &ev = *evec2[el];
for (quint32 i=0; i < ev.count(); ++i) {
t1 = ev.time(i);
for (int z=0; z < spans; ++z) {
if ((t1 >= start.at(z)) && (t1 <= end.at(z))) {
count++;
break;
}
}
}
}
return count;
}
EventDataType Session::rangeCount(ChannelID id, qint64 first, qint64 last) EventDataType Session::rangeCount(ChannelID id, qint64 first, qint64 last)
{ {
QHash<ChannelID, QVector<EventList *> >::iterator j = eventlist.find(id); QHash<ChannelID, QVector<EventList *> >::iterator j = eventlist.find(id);

View File

@ -224,6 +224,9 @@ class Session
//! \brief Returns the maximum of events of type id between time range //! \brief Returns the maximum of events of type id between time range
EventDataType rangeMax(ChannelID id, qint64 first, qint64 last); EventDataType rangeMax(ChannelID id, qint64 first, qint64 last);
//! \brief Returns the count of code events inside span flag event durations
EventDataType countInsideSpan(ChannelID span, ChannelID code);
//! \brief Returns (and caches) the Sum of all events of type id //! \brief Returns (and caches) the Sum of all events of type id
double sum(ChannelID id); double sum(ChannelID id);

View File

@ -1166,14 +1166,23 @@ QString Daily::getStatisticsInfo(Day * cpap,Day * oxi,Day *pos)
} }
if (cpap) { if (cpap) {
int l = cpap->sum(CPAP_Ramp); int l = cpap->sum(CPAP_Ramp) - (15*60);
if (l>0) { if (l > 0) {
int h = l / 3600; html+="<tr><td colspan=3 align='left' bgcolor='white'><b>"+tr("Total ramp time")+
int m = (l / 60) % 60; QString("</b></td><td colspan=2 bgcolor='white'>%1:%2:%3</td></tr>").arg(l / 3600, 2, 10, QChar('0')).arg((l / 60) % 60, 2, 10, QChar('0')).arg(l % 60, 2, 10, QChar('0'));
int s = l % 60; float v = (cpap->hours() - (float(l) / 3600.0));
html+="<tr><td colspan=3 align='left' bgcolor='white'><b>"+tr("Time spent in ramp")+ int q = v * 3600.0;
QString("</b></td><td colspan=2 bgcolor='white'>%1:%2:%3</td></tr>").arg(h, 2, 10, QChar('0')).arg(m, 2, 10, QChar('0')).arg(s, 2, 10, QChar('0')); html+="<tr><td colspan=3 align='left' bgcolor='white'><b>"+tr("Time outside of ramp")+
QString("</b></td><td colspan=2 bgcolor='white'>%1:%2:%3</td></tr>").arg(q / 3600, 2, 10, QChar('0')).arg((q / 60) % 60, 2, 10, QChar('0')).arg(q % 60, 2, 10, QChar('0'));
EventDataType hc = cpap->count(CPAP_Hypopnea) - cpap->countInsideSpan(CPAP_Ramp, CPAP_Hypopnea);
EventDataType oc = cpap->count(CPAP_Obstructive) - cpap->countInsideSpan(CPAP_Ramp, CPAP_Obstructive);
EventDataType tc = cpap->count(CPAP_Hypopnea) + cpap->count(CPAP_Obstructive);
EventDataType ahi = (hc+oc) / (float(l)/3600.0);
html+="<tr><td colspan=3 align='left' bgcolor='white'><b>"+tr("AHI excluding ramp")+
QString("</b></td><td colspan=2 bgcolor='white'>%1</td></tr>").arg(ahi, 0, 'f', 2);
} }
} }

View File

@ -1215,14 +1215,16 @@ QString Statistics::GenerateHTML()
html += QString("<table class=curved style=\"page-break-before:auto;\" "+table_width+">"); html += QString("<table class=curved style=\"page-break-before:auto;\" "+table_width+">");
html += "<thead>"; html += "<thead>";
html += "<tr bgcolor='"+heading_color+"'><td colspan=5 align=center><font size=+2>" + tr("Machine Information") + "</font></td></tr>"; html += "<tr bgcolor='"+heading_color+"'><th colspan=7 align=center><font size=+2>" + tr("Machine Information") + "</font></th></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><td><b>%6</b></td><td><b>%7</b></td></tr>")
.arg(STR_TR_Brand) .arg(STR_TR_Brand)
.arg(STR_TR_Series)
.arg(STR_TR_Model) .arg(STR_TR_Model)
.arg(STR_TR_Serial) .arg(STR_TR_Serial)
.arg(tr("First Use")) .arg(tr("First Use"))
.arg(tr("Last Use")); .arg(tr("Last Use"))
.arg(STR_TR_AHI);
html += "</thead>"; html += "</thead>";
@ -1233,14 +1235,24 @@ QString Statistics::GenerateHTML()
if (m->type() == MT_JOURNAL) { continue; } if (m->type() == MT_JOURNAL) { continue; }
QDate d1 = m->FirstDay();
QDate d2 = m->LastDay();
QString ahi;
if (m->type() == MT_CPAP) {
float a = calcAHI(d1,d2);
ahi = QString::number(a,'f',2);
}
QString mn = m->modelnumber(); QString mn = m->modelnumber();
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td></tr>") html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td><td>%7</td></tr>")
.arg(m->brand()) .arg(m->brand())
.arg(m->series())
.arg(m->model() + .arg(m->model() +
(mn.isEmpty() ? "" : QString(" (") + mn + QString(")"))) (mn.isEmpty() ? "" : QString(" (") + mn + QString(")")))
.arg(m->serial()) .arg(m->serial())
.arg(m->FirstDay().toString(Qt::SystemLocaleShortDate)) .arg(d1.toString(Qt::SystemLocaleShortDate))
.arg(m->LastDay().toString(Qt::SystemLocaleShortDate)); .arg(d2.toString(Qt::SystemLocaleShortDate))
.arg(ahi);
} }
html += "</table>"; html += "</table>";