mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
CMS50i file import support, CMS50D+ swap pulse and spo2, added AHI to machine list
This commit is contained in:
parent
e956038571
commit
22ea3868f1
@ -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);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -337,7 +337,7 @@ void gXAxis::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
|||||||
}
|
}
|
||||||
|
|
||||||
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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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");
|
||||||
|
@ -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;
|
||||||
|
@ -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();
|
||||||
|
@ -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();
|
||||||
|
|
||||||
|
@ -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());
|
||||||
|
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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);
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -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>";
|
||||||
|
Loading…
Reference in New Issue
Block a user