mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 18:50:44 +00:00
Restructure Day object to allow for multiple machine sessions
This commit is contained in:
parent
ba8b8f12fa
commit
e27232423e
@ -48,17 +48,18 @@ void MinutesAtPressure::SetDay(Day *day)
|
||||
Layer::SetDay(day);
|
||||
|
||||
// look at session summaryValues.
|
||||
if (day) {
|
||||
Machine * cpap = nullptr;
|
||||
if (day) cpap = day->machine(MT_CPAP);
|
||||
if (cpap) {
|
||||
QList<Session *>::iterator sit;
|
||||
EventDataType minpressure = 40;
|
||||
EventDataType maxpressure = 0;
|
||||
|
||||
Machine * mach = day->machine;
|
||||
QMap<QDate, Day *>::iterator it;
|
||||
QMap<QDate, Day *>::iterator day_end = mach->day.end();
|
||||
QMap<QDate, Day *>::iterator day_end = cpap->day.end();
|
||||
// look at overall pressure ranges and find the max
|
||||
|
||||
for (it = mach->day.begin(); it != day_end; ++it) {
|
||||
for (it = cpap->day.begin(); it != day_end; ++it) {
|
||||
Day * d = it.value();
|
||||
QList<Session *>::iterator sess_end = d->end();
|
||||
for (sit = d->begin(); sit != sess_end; ++sit) {
|
||||
|
@ -124,7 +124,7 @@ void SummaryChart::SetDay(Day * nullday)
|
||||
bool first = true;
|
||||
|
||||
// For each day in the main profile daylist
|
||||
QMap<QDate, QList<Day *> >::iterator d;
|
||||
QMap<QDate, Day *>::iterator d;
|
||||
|
||||
for (d = p_profile->daylist.begin(); d != p_profile->daylist.end(); d++) {
|
||||
|
||||
@ -154,64 +154,61 @@ void SummaryChart::SetDay(Day * nullday)
|
||||
}
|
||||
|
||||
// for each day object on record for this date
|
||||
int dlistsize = d.value().size();
|
||||
for (int i = 0; i < dlistsize; ++i) {
|
||||
day = d.value().at(i);
|
||||
day = d.value();
|
||||
|
||||
// skip any empty or irrelevant day records
|
||||
if (!day || (day->machine_type() != m_machinetype)) { continue; }
|
||||
// skip any empty or irrelevant day records
|
||||
if (!day || (day->machine(m_machinetype) == nullptr)) { continue; }
|
||||
|
||||
int ft = qint64(day->first()) / 1000L;
|
||||
ft += tz_offset; // convert to local time
|
||||
int ft = qint64(day->first()) / 1000L;
|
||||
ft += tz_offset; // convert to local time
|
||||
|
||||
int dz2 = ft / 86400;
|
||||
dz2 *= 86400;
|
||||
// ft = first sessions time, rounded back to midnight..
|
||||
int dz2 = ft / 86400;
|
||||
dz2 *= 86400;
|
||||
// ft = first sessions time, rounded back to midnight..
|
||||
|
||||
// For each session in this day record
|
||||
for (int s = 0; s < day->size(); s++) {
|
||||
Session *sess = (*day)[s];
|
||||
// For each session in this day record
|
||||
for (int s = 0; s < day->size(); s++) {
|
||||
Session *sess = (*day)[s];
|
||||
|
||||
if (!sess->enabled()) { continue; }
|
||||
if (!sess->enabled()) { continue; }
|
||||
|
||||
// Get session duration
|
||||
tmp = sess->hours();
|
||||
m_values[dn][s] = tmp;
|
||||
// Get session duration
|
||||
tmp = sess->hours();
|
||||
m_values[dn][s] = tmp;
|
||||
|
||||
total += tmp;
|
||||
total += tmp;
|
||||
|
||||
// Get session start timestamp
|
||||
zt = qint64(sess->first()) / 1000L;
|
||||
zt += tz_offset;
|
||||
// Get session start timestamp
|
||||
zt = qint64(sess->first()) / 1000L;
|
||||
zt += tz_offset;
|
||||
|
||||
// Calculate the starting hour
|
||||
tmp2 = zt - dn * 86400;
|
||||
tmp2 /= 3600.0;
|
||||
// Calculate the starting hour
|
||||
tmp2 = zt - dn * 86400;
|
||||
tmp2 /= 3600.0;
|
||||
|
||||
m_times[dn][s] = tmp2;
|
||||
m_times[dn][s] = tmp2;
|
||||
|
||||
// Update min & max Y values
|
||||
if (first) {
|
||||
// Update min & max Y values
|
||||
if (first) {
|
||||
m_miny = tmp2;
|
||||
m_maxy = tmp2 + tmp;
|
||||
first = false;
|
||||
} else {
|
||||
if (tmp2 < m_miny) {
|
||||
m_miny = tmp2;
|
||||
m_maxy = tmp2 + tmp;
|
||||
first = false;
|
||||
} else {
|
||||
if (tmp2 < m_miny) {
|
||||
m_miny = tmp2;
|
||||
}
|
||||
|
||||
if (tmp2 + tmp > m_maxy) {
|
||||
m_maxy = tmp2 + tmp;
|
||||
}
|
||||
}
|
||||
} // for each session
|
||||
|
||||
// if total hours for all sessions more than 0, register the day as valid
|
||||
if (total > 0) {
|
||||
m_days[dn] = day;
|
||||
m_hours[dn] = total;
|
||||
m_empty = false;
|
||||
if (tmp2 + tmp > m_maxy) {
|
||||
m_maxy = tmp2 + tmp;
|
||||
}
|
||||
}
|
||||
} // for each session
|
||||
|
||||
// if total hours for all sessions more than 0, register the day as valid
|
||||
if (total > 0) {
|
||||
m_days[dn] = day;
|
||||
m_hours[dn] = total;
|
||||
m_empty = false;
|
||||
}
|
||||
} else {
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
@ -225,139 +222,136 @@ void SummaryChart::SetDay(Day * nullday)
|
||||
type = m_type[j];
|
||||
EventDataType typeval = m_typeval[j];
|
||||
|
||||
// for each machine object for this day
|
||||
for (int i = 0; i < d.value().size(); i++) {
|
||||
day = d.value()[i];
|
||||
day = d.value();
|
||||
|
||||
CPAPMode mode = (CPAPMode)(int)day->settings_max(CPAP_Mode);
|
||||
CPAPMode mode = (CPAPMode)(int)day->settings_max(CPAP_Mode);
|
||||
|
||||
// ignore irrelevent day objects
|
||||
if (day->machine_type() != m_machinetype) { continue; }
|
||||
// ignore irrelevent day objects
|
||||
if (day->machine(m_machinetype) == nullptr) { continue; }
|
||||
|
||||
bool hascode = //day->channelHasData(code) ||
|
||||
type == ST_HOURS ||
|
||||
type == ST_SESSIONS ||
|
||||
day->settingExists(code) ||
|
||||
day->hasData(code, type);
|
||||
bool hascode = //day->channelHasData(code) ||
|
||||
type == ST_HOURS ||
|
||||
type == ST_SESSIONS ||
|
||||
day->settingExists(code) ||
|
||||
day->hasData(code, type);
|
||||
|
||||
if (code == CPAP_Pressure) {
|
||||
if ((cpapmode > MODE_CPAP) && (mode == MODE_CPAP)) {
|
||||
hascode = false;
|
||||
if (code == CPAP_Pressure) {
|
||||
if ((cpapmode > MODE_CPAP) && (mode == MODE_CPAP)) {
|
||||
hascode = false;
|
||||
|
||||
if ((type == ST_WAVG) || (type == ST_AVG) || ((type == ST_PERC) && (typeval == 0.5))) {
|
||||
type = ST_SETWAVG;
|
||||
hascode = true;
|
||||
}
|
||||
} else {
|
||||
type = m_type[j];
|
||||
if ((type == ST_WAVG) || (type == ST_AVG) || ((type == ST_PERC) && (typeval == 0.5))) {
|
||||
type = ST_SETWAVG;
|
||||
hascode = true;
|
||||
}
|
||||
}
|
||||
|
||||
//if (code==CPAP_Hypopnea) { // Make sure at least one of the CPAP data gets through with 0
|
||||
// hascode=true;
|
||||
//}
|
||||
if (hascode) {
|
||||
m_days[dn] = day;
|
||||
|
||||
switch (type) {
|
||||
case ST_AVG:
|
||||
tmp = day->avg(code);
|
||||
break;
|
||||
|
||||
case ST_SUM:
|
||||
tmp = day->sum(code);
|
||||
break;
|
||||
|
||||
case ST_WAVG:
|
||||
tmp = day->wavg(code);
|
||||
break;
|
||||
|
||||
case ST_90P:
|
||||
tmp = day->p90(code);
|
||||
break;
|
||||
|
||||
case ST_PERC:
|
||||
tmp = day->percentile(code, typeval);
|
||||
break;
|
||||
|
||||
case ST_MIN:
|
||||
tmp = day->Min(code);
|
||||
break;
|
||||
|
||||
case ST_MAX:
|
||||
tmp = day->Max(code);
|
||||
break;
|
||||
|
||||
case ST_CNT:
|
||||
tmp = day->count(code);
|
||||
break;
|
||||
|
||||
case ST_CPH:
|
||||
tmp = day->count(code) / day->hours();
|
||||
break;
|
||||
|
||||
case ST_SPH:
|
||||
tmp = day->sph(code);
|
||||
break;
|
||||
|
||||
case ST_HOURS:
|
||||
tmp = day->hours();
|
||||
break;
|
||||
|
||||
case ST_SESSIONS:
|
||||
tmp = day->size();
|
||||
break;
|
||||
|
||||
case ST_SETMIN:
|
||||
tmp = day->settings_min(code);
|
||||
break;
|
||||
|
||||
case ST_SETMAX:
|
||||
tmp = day->settings_max(code);
|
||||
break;
|
||||
|
||||
case ST_SETAVG:
|
||||
tmp = day->settings_avg(code);
|
||||
break;
|
||||
|
||||
case ST_SETWAVG:
|
||||
tmp = day->settings_wavg(code);
|
||||
break;
|
||||
|
||||
case ST_SETSUM:
|
||||
tmp = day->settings_sum(code);
|
||||
break;
|
||||
|
||||
default:
|
||||
tmp = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
if (suboffset > 0) {
|
||||
tmp -= suboffset;
|
||||
|
||||
if (tmp < 0) { tmp = 0; }
|
||||
}
|
||||
|
||||
total += tmp;
|
||||
m_values[dn][j + 1] = tmp;
|
||||
|
||||
if (tmp < m_miny) { m_miny = tmp; }
|
||||
|
||||
if (tmp > m_maxy) { m_maxy = tmp; }
|
||||
|
||||
m_goodcodes[j] = true;
|
||||
fnd = true;
|
||||
break;
|
||||
} else {
|
||||
type = m_type[j];
|
||||
}
|
||||
}
|
||||
|
||||
//if (code==CPAP_Hypopnea) { // Make sure at least one of the CPAP data gets through with 0
|
||||
// hascode=true;
|
||||
//}
|
||||
if (hascode) {
|
||||
m_days[dn] = day;
|
||||
|
||||
switch (type) {
|
||||
case ST_AVG:
|
||||
tmp = day->avg(code);
|
||||
break;
|
||||
|
||||
case ST_SUM:
|
||||
tmp = day->sum(code);
|
||||
break;
|
||||
|
||||
case ST_WAVG:
|
||||
tmp = day->wavg(code);
|
||||
break;
|
||||
|
||||
case ST_90P:
|
||||
tmp = day->p90(code);
|
||||
break;
|
||||
|
||||
case ST_PERC:
|
||||
tmp = day->percentile(code, typeval);
|
||||
break;
|
||||
|
||||
case ST_MIN:
|
||||
tmp = day->Min(code);
|
||||
break;
|
||||
|
||||
case ST_MAX:
|
||||
tmp = day->Max(code);
|
||||
break;
|
||||
|
||||
case ST_CNT:
|
||||
tmp = day->count(code);
|
||||
break;
|
||||
|
||||
case ST_CPH:
|
||||
tmp = day->count(code) / day->hours(m_machinetype);
|
||||
break;
|
||||
|
||||
case ST_SPH:
|
||||
tmp = day->sph(code);
|
||||
break;
|
||||
|
||||
case ST_HOURS:
|
||||
tmp = day->hours(m_machinetype);
|
||||
break;
|
||||
|
||||
case ST_SESSIONS:
|
||||
tmp = day->size();
|
||||
break;
|
||||
|
||||
case ST_SETMIN:
|
||||
tmp = day->settings_min(code);
|
||||
break;
|
||||
|
||||
case ST_SETMAX:
|
||||
tmp = day->settings_max(code);
|
||||
break;
|
||||
|
||||
case ST_SETAVG:
|
||||
tmp = day->settings_avg(code);
|
||||
break;
|
||||
|
||||
case ST_SETWAVG:
|
||||
tmp = day->settings_wavg(code);
|
||||
break;
|
||||
|
||||
case ST_SETSUM:
|
||||
tmp = day->settings_sum(code);
|
||||
break;
|
||||
|
||||
default:
|
||||
tmp = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
if (suboffset > 0) {
|
||||
tmp -= suboffset;
|
||||
|
||||
if (tmp < 0) { tmp = 0; }
|
||||
}
|
||||
|
||||
total += tmp;
|
||||
m_values[dn][j + 1] = tmp;
|
||||
|
||||
if (tmp < m_miny) { m_miny = tmp; }
|
||||
|
||||
if (tmp > m_maxy) { m_maxy = tmp; }
|
||||
|
||||
m_goodcodes[j] = true;
|
||||
fnd = true;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
if (fnd) {
|
||||
if (!m_fday) { m_fday = dn; }
|
||||
|
||||
m_values[dn][0] = total;
|
||||
m_hours[dn] = day->hours();
|
||||
m_hours[dn] = day->hours(m_machinetype);
|
||||
|
||||
if (m_graphtype == GT_BAR) {
|
||||
if (total < m_miny) { m_miny = total; }
|
||||
|
@ -34,7 +34,11 @@ void gDailySummary::SetDay(Day *day)
|
||||
pie_total = 0;
|
||||
|
||||
m_day = day;
|
||||
if (day) {
|
||||
Machine * cpap = nullptr;
|
||||
if (day) cpap = day->machine(MT_CPAP);
|
||||
|
||||
if (cpap) {
|
||||
|
||||
m_minx = m_day->first();
|
||||
m_maxx = m_day->last();;
|
||||
|
||||
@ -103,8 +107,9 @@ void gDailySummary::SetDay(Day *day)
|
||||
info.append(QObject::tr("%1: %2").arg(STR_TR_AHI).arg(day->calcAHI(),0,'f',2));
|
||||
info_background.append(QColor("orange"));
|
||||
|
||||
settings.append(day->machine->brand()+ " " + day->machine->series());
|
||||
settings.append(day->machine->model()+ " " + day->machine->modelnumber());
|
||||
|
||||
settings.append(cpap->brand()+ " " + cpap->series());
|
||||
settings.append(cpap->model()+ " " + cpap->modelnumber());
|
||||
settings.append(schema::channel[CPAP_Mode].option(mode));
|
||||
|
||||
if (mode == MODE_CPAP) {
|
||||
|
@ -15,8 +15,7 @@
|
||||
#include "day.h"
|
||||
#include "profiles.h"
|
||||
|
||||
Day::Day(Machine *m)
|
||||
: machine(m)
|
||||
Day::Day()
|
||||
{
|
||||
d_firstsession = true;
|
||||
}
|
||||
@ -26,10 +25,53 @@ Day::~Day()
|
||||
delete(*s);
|
||||
}
|
||||
}
|
||||
MachineType Day::machine_type() const
|
||||
|
||||
Session * Day::firstSession(MachineType type)
|
||||
{
|
||||
return machine->type();
|
||||
for (int i=0; i<sessions.size(); i++) {
|
||||
Session * sess = sessions.at(i);
|
||||
if (!sess->enabled()) continue;
|
||||
if (sess->machine()->type() == type) {
|
||||
return sess;
|
||||
}
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
bool Day::addMachine(Machine *mach)
|
||||
{
|
||||
if (!machines.contains(mach->type())) {
|
||||
machines[mach->type()] = mach;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
Machine *Day::machine(MachineType type)
|
||||
{
|
||||
QHash<MachineType,Machine *>::iterator it = machines.find(type);
|
||||
if (it != machines.end())
|
||||
return it.value();
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
QList<Session *> Day::getSessions(MachineType type)
|
||||
{
|
||||
QList<Session *>::iterator it;
|
||||
QList<Session *>::iterator sess_end = sessions.end();
|
||||
|
||||
QList<Session *> newlist;
|
||||
|
||||
for (it = sessions.begin(); it != sess_end; ++it) {
|
||||
if (!(*it)->enabled())
|
||||
continue;
|
||||
|
||||
if ((*it)->machine()->type() == type)
|
||||
newlist.append((*it));
|
||||
}
|
||||
|
||||
return newlist;
|
||||
}
|
||||
|
||||
Session *Day::find(SessionID sessid)
|
||||
{
|
||||
QList<Session *>::iterator end=sessions.end();
|
||||
@ -42,11 +84,18 @@ Session *Day::find(SessionID sessid)
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
void Day::AddSession(Session *s)
|
||||
void Day::addSession(Session *s)
|
||||
{
|
||||
if (!s) {
|
||||
qWarning("Day::AddSession called with nullptr session object");
|
||||
return;
|
||||
Q_ASSERT(s!=nullptr);
|
||||
QHash<MachineType, Machine *>::iterator mi = machines.find(s->machine()->type());
|
||||
|
||||
if (mi != machines.end()) {
|
||||
if (mi.value() != s->machine()) {
|
||||
qDebug() << "SleepyHead can't add session" << s->session() << "to this day record, as it already contains a different machine of the same MachineType";
|
||||
return;
|
||||
}
|
||||
} else {
|
||||
machines[s->machine()->type()] = s->machine();
|
||||
}
|
||||
|
||||
sessions.push_back(s);
|
||||
@ -489,7 +538,7 @@ qint64 Day::total_time()
|
||||
for (QList<Session *>::iterator it = sessions.begin(); it != end; ++it) {
|
||||
Session &sess = *(*it);
|
||||
|
||||
if (sess.enabled()) {
|
||||
if (sess.enabled() && (sess.machine()->type() != MT_JOURNAL)) {
|
||||
first = sess.first();
|
||||
last = sess.last();
|
||||
|
||||
@ -536,6 +585,68 @@ qint64 Day::total_time()
|
||||
return total; //d_totaltime;
|
||||
}
|
||||
|
||||
// Total session time in milliseconds, only considering machinetype
|
||||
qint64 Day::total_time(MachineType type)
|
||||
{
|
||||
qint64 d_totaltime = 0;
|
||||
QMultiMap<qint64, bool> range;
|
||||
//range.reserve(size()*2);
|
||||
|
||||
// Remember sessions may overlap..
|
||||
|
||||
qint64 first, last;
|
||||
QList<Session *>::iterator end = sessions.end();
|
||||
for (QList<Session *>::iterator it = sessions.begin(); it != end; ++it) {
|
||||
Session &sess = *(*it);
|
||||
|
||||
if ((sess.machine()->type() == type) && sess.enabled()) {
|
||||
first = sess.first();
|
||||
last = sess.last();
|
||||
|
||||
// This algorithm relies on non zero length, and correctly ordered sessions
|
||||
if (last > first) {
|
||||
range.insert(first, 0);
|
||||
range.insert(last, 1);
|
||||
d_totaltime += sess.length();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bool b;
|
||||
int nest = 0;
|
||||
qint64 ti = 0;
|
||||
qint64 total = 0;
|
||||
|
||||
// This is my implementation of a typical "brace counting" algorithm mentioned here:
|
||||
// http://stackoverflow.com/questions/7468948/problem-calculating-overlapping-date-ranges
|
||||
|
||||
QMultiMap<qint64, bool>::iterator rend = range.end();
|
||||
for (QMultiMap<qint64, bool>::iterator rit = range.begin(); rit != rend; ++rit) {
|
||||
b = rit.value();
|
||||
|
||||
if (!b) {
|
||||
if (!ti) {
|
||||
ti = rit.key();
|
||||
}
|
||||
|
||||
nest++;
|
||||
} else {
|
||||
if (--nest <= 0) {
|
||||
total += rit.key() - ti;
|
||||
ti = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (total != d_totaltime) {
|
||||
// They can overlap.. tough.
|
||||
// qDebug() << "Sessions Times overlaps!" << total << d_totaltime;
|
||||
}
|
||||
|
||||
return total; //d_totaltime;
|
||||
}
|
||||
|
||||
|
||||
bool Day::hasEnabledSessions()
|
||||
{
|
||||
QList<Session *>::iterator end = sessions.end();
|
||||
@ -549,6 +660,19 @@ bool Day::hasEnabledSessions()
|
||||
return false;
|
||||
}
|
||||
|
||||
bool Day::hasEnabledSessions(MachineType type)
|
||||
{
|
||||
QList<Session *>::iterator end = sessions.end();
|
||||
|
||||
for (QList<Session *>::iterator it = sessions.begin(); it != end; ++it) {
|
||||
if (((*it)->machine()->type() == type) && (*it)->enabled()) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/*EventDataType Day::percentile(ChannelID code,double percent)
|
||||
{
|
||||
double val=0;
|
||||
@ -946,7 +1070,12 @@ void Day::CloseEvents()
|
||||
|
||||
QList<ChannelID> Day::getSortedMachineChannels(quint32 chantype)
|
||||
{
|
||||
QList<ChannelID> available = machine->availableChannels(chantype);
|
||||
QList<ChannelID> available;
|
||||
QHash<MachineType, Machine *>::iterator mi_end = machines.end();
|
||||
for (QHash<MachineType, Machine *>::iterator mi = machines.begin(); mi != mi_end; mi++) {
|
||||
if (mi.key() == MT_JOURNAL) continue;
|
||||
available.append(mi.value()->availableChannels(chantype));
|
||||
}
|
||||
|
||||
QMultiMap<int, ChannelID> order;
|
||||
|
||||
@ -965,6 +1094,31 @@ QList<ChannelID> Day::getSortedMachineChannels(quint32 chantype)
|
||||
return channels;
|
||||
}
|
||||
|
||||
qint64 Day::first(MachineType type)
|
||||
{
|
||||
qint64 date = 0;
|
||||
qint64 tmp;
|
||||
|
||||
QList<Session *>::iterator end = sessions.end();
|
||||
for (QList<Session *>::iterator it = sessions.begin(); it != end; ++it) {
|
||||
Session & sess = *(*it);
|
||||
|
||||
if ((sess.machine()->type() == type) && sess.enabled()) {
|
||||
tmp = sess.first();
|
||||
|
||||
if (!tmp) { continue; }
|
||||
|
||||
if (!date) {
|
||||
date = tmp;
|
||||
} else {
|
||||
if (tmp < date) { date = tmp; }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return date;
|
||||
}
|
||||
|
||||
qint64 Day::first()
|
||||
{
|
||||
qint64 date = 0;
|
||||
@ -1017,6 +1171,33 @@ qint64 Day::last()
|
||||
return date;
|
||||
}
|
||||
|
||||
qint64 Day::last(MachineType type)
|
||||
{
|
||||
qint64 date = 0;
|
||||
qint64 tmp;
|
||||
|
||||
QList<Session *>::iterator end = sessions.end();
|
||||
|
||||
|
||||
for (QList<Session *>::iterator it = sessions.begin(); it != end; ++it) {
|
||||
Session & sess = *(*it);
|
||||
|
||||
if ((sess.machine()->type() == type) && sess.enabled()) {
|
||||
tmp = sess.last();
|
||||
|
||||
if (!tmp) { continue; }
|
||||
|
||||
if (!date) {
|
||||
date = tmp;
|
||||
} else {
|
||||
if (tmp > date) { date = tmp; }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return date;
|
||||
}
|
||||
|
||||
bool Day::removeSession(Session *sess)
|
||||
{
|
||||
return sessions.removeAll(sess) > 0;
|
||||
@ -1024,7 +1205,7 @@ bool Day::removeSession(Session *sess)
|
||||
|
||||
QString Day::getCPAPMode()
|
||||
{
|
||||
Q_ASSERT(machine_type() == MT_CPAP);
|
||||
Q_ASSERT(machine(MT_CPAP) != nullptr);
|
||||
|
||||
CPAPMode mode = (CPAPMode)(int)qRound(settings_wavg(CPAP_Mode));
|
||||
if (mode == MODE_CPAP) {
|
||||
@ -1047,7 +1228,10 @@ QString Day::getCPAPMode()
|
||||
|
||||
QString Day::getPressureRelief()
|
||||
{
|
||||
CPAPLoader * loader = qobject_cast<CPAPLoader *>(machine->loader());
|
||||
Machine * mach = machine(MT_CPAP);
|
||||
if (!mach) return STR_MessageBox_Error;
|
||||
|
||||
CPAPLoader * loader = qobject_cast<CPAPLoader *>(mach->loader());
|
||||
|
||||
if (!loader) return STR_MessageBox_Error;
|
||||
|
||||
@ -1072,7 +1256,7 @@ QString Day::getPressureRelief()
|
||||
|
||||
QString Day::getPressureSettings()
|
||||
{
|
||||
Q_ASSERT(machine_type() == MT_CPAP);
|
||||
Q_ASSERT(machine(MT_CPAP) != nullptr);
|
||||
|
||||
CPAPMode mode = (CPAPMode)(int)settings_max(CPAP_Mode);
|
||||
QString units = schema::channel[CPAP_Pressure].units();
|
||||
|
@ -31,14 +31,20 @@ class Session;
|
||||
class Day
|
||||
{
|
||||
public:
|
||||
Day(Machine *m);
|
||||
Day();
|
||||
~Day();
|
||||
|
||||
//! \brief Add Session to this Day object (called during Load)
|
||||
void AddSession(Session *s);
|
||||
//! \brief Add a new machine to this day record
|
||||
bool addMachine(Machine *m);
|
||||
|
||||
//! \brief Returns this machines type
|
||||
MachineType machine_type() const;
|
||||
//! \brief Returns a machine record if present of specified machine type
|
||||
Machine *machine(MachineType type);
|
||||
|
||||
//! \brief Returns a list of sessions for the specified machine type
|
||||
QList<Session *> getSessions(MachineType type);
|
||||
|
||||
//! \brief Add Session to this Day object (called during Load)
|
||||
void addSession(Session *s);
|
||||
|
||||
//! \brief Returns the count of all this days sessions' events for this day
|
||||
EventDataType count(ChannelID code);
|
||||
@ -113,6 +119,13 @@ class Day
|
||||
//! \brief Returns the last session time of this day
|
||||
qint64 last();
|
||||
|
||||
//! \brief Returns the first session time of this machine type for this day
|
||||
qint64 first(MachineType type);
|
||||
|
||||
//! \brief Returns the last session time of this machine type for this day
|
||||
qint64 last(MachineType type);
|
||||
|
||||
|
||||
// //! \brief Sets the first session time of this day
|
||||
// void setFirst(qint64 val) { d_first=val; }
|
||||
|
||||
@ -128,11 +141,18 @@ class Day
|
||||
//! \brief Returns the total time in milliseconds for this day
|
||||
qint64 total_time();
|
||||
|
||||
//! \brief Returns the total time in milliseconds for this day for given machine type
|
||||
qint64 total_time(MachineType type);
|
||||
|
||||
//! \brief Returns true if this day has enabled sessions for supplied machine type
|
||||
bool hasEnabledSessions(MachineType);
|
||||
|
||||
//! \brief Returns true if this day has enabled sessions
|
||||
bool hasEnabledSessions();
|
||||
|
||||
//! \brief Return the total time in decimal hours for this day
|
||||
EventDataType hours() { return double(total_time()) / 3600000.0; }
|
||||
EventDataType hours(MachineType type) { return double(total_time(type)) / 3600000.0; }
|
||||
|
||||
//! \brief Return the session indexed by i
|
||||
Session *operator [](int i) { return sessions[i]; }
|
||||
@ -153,8 +173,6 @@ class Day
|
||||
//! \brief Returns the number of Sessions in this day record
|
||||
int size() { return sessions.size(); }
|
||||
|
||||
Machine *machine;
|
||||
|
||||
//! \brief Loads all Events files for this Days Sessions
|
||||
void OpenEvents();
|
||||
|
||||
@ -239,8 +257,13 @@ class Day
|
||||
|
||||
EventDataType calc(ChannelID code, ChannelCalcType type);
|
||||
|
||||
Session * firstSession(MachineType type);
|
||||
|
||||
//! \brief A QList containing all Sessions objects for this day
|
||||
QList<Session *> sessions;
|
||||
|
||||
QHash<MachineType, Machine *> machines;
|
||||
|
||||
protected:
|
||||
//! \brief A Vector containing all sessions for this day
|
||||
QHash<ChannelID, QHash<EventDataType, EventDataType> > perc_cache;
|
||||
|
@ -64,7 +64,7 @@ JournalEntry::JournalEntry(QDate date)
|
||||
session = nullptr;
|
||||
day = p_profile->GetDay(date, MT_JOURNAL);
|
||||
if (!day) {
|
||||
session = day->sessions[0];
|
||||
session = day->firstSession(MT_JOURNAL);
|
||||
} else {
|
||||
// Doesn't exist.. create a new one..
|
||||
session = new Session(jmach,0);
|
||||
@ -227,7 +227,7 @@ void BackupJournal(QString filename)
|
||||
|
||||
Day * journal = p_profile->GetDay(date, MT_JOURNAL);
|
||||
if (!journal) continue;
|
||||
Session * sess = journal->sessions[0];
|
||||
Session * sess = journal->firstSession(MT_JOURNAL);
|
||||
if (!sess) continue;
|
||||
QDomElement day = doc.createElement("day");
|
||||
day.setAttribute("date", date.toString());
|
||||
|
@ -59,10 +59,6 @@ Machine::Machine(MachineID id)
|
||||
Machine::~Machine()
|
||||
{
|
||||
qDebug() << "Destroy Machine" << info.loadername << hex << m_id;
|
||||
|
||||
for (QMap<QDate, Day *>::iterator d = day.begin(); d != day.end(); d++) {
|
||||
delete d.value();
|
||||
}
|
||||
}
|
||||
Session *Machine::SessionExists(SessionID session)
|
||||
{
|
||||
@ -208,17 +204,11 @@ bool Machine::AddSession(Session *s)
|
||||
dit = day.find(date);
|
||||
|
||||
if (dit == day.end()) {
|
||||
//QString dstr=date.toString("yyyyMMdd");
|
||||
//qDebug("Adding Profile Day %s",dstr.toLatin1().data());
|
||||
dd = new Day(this);
|
||||
day[date] = dd;
|
||||
// Add this Day record to profile
|
||||
p_profile->AddDay(date, dd, m_type);
|
||||
} else {
|
||||
dd = *dit;
|
||||
dit = day.insert(date, p_profile->addDay(date));
|
||||
}
|
||||
dd = dit.value();
|
||||
|
||||
dd->AddSession(s);
|
||||
dd->addSession(s);
|
||||
|
||||
if (combine_next_day) {
|
||||
for (QList<Session *>::iterator i = nextday.value()->begin(); i != nextday.value()->end(); i++) {
|
||||
@ -229,7 +219,7 @@ bool Machine::AddSession(Session *s)
|
||||
|
||||
sessionlist[(*i)->session()] = *i;
|
||||
|
||||
dd->AddSession(*i);
|
||||
dd->addSession(*i);
|
||||
}
|
||||
|
||||
// QMap<QDate, QList<Day *> >::iterator nd = p_profile->daylist.find(date.addDays(1));
|
||||
@ -692,7 +682,6 @@ QList<ChannelID> Machine::availableChannels(quint32 chantype)
|
||||
{
|
||||
QHash<ChannelID, int> chanhash;
|
||||
|
||||
|
||||
// look through the daylist and return a list of available channels for this machine
|
||||
QMap<QDate, Day *>::iterator dit;
|
||||
QMap<QDate, Day *>::iterator day_end = day.end();
|
||||
@ -701,6 +690,8 @@ QList<ChannelID> Machine::availableChannels(quint32 chantype)
|
||||
|
||||
for (QList<Session *>::iterator sit = dit.value()->begin(); sit != sess_end; ++sit) {
|
||||
Session * sess = (*sit);
|
||||
if (sess->machine() != this) continue;
|
||||
|
||||
int size = sess->availableChannels().size();
|
||||
for (int i=0; i < size; ++i) {
|
||||
ChannelID code = sess->availableChannels().at(i);
|
||||
|
@ -103,6 +103,7 @@ Machine * MachineLoader::CreateMachine(MachineInfo info, MachineID id)
|
||||
break;
|
||||
case MT_JOURNAL:
|
||||
m = new Machine(id);
|
||||
m->setType(MT_JOURNAL);
|
||||
break;
|
||||
default:
|
||||
m = new Machine(id);
|
||||
@ -246,7 +247,7 @@ void MachineLoader::runTasks(bool threaded)
|
||||
|
||||
QList<ChannelID> CPAPLoader::eventFlags(Day * day)
|
||||
{
|
||||
Machine * mach = day->machine;
|
||||
Machine * mach = day->machine(MT_CPAP);
|
||||
|
||||
QList<ChannelID> list;
|
||||
|
||||
|
@ -84,6 +84,11 @@ Profile::~Profile()
|
||||
}
|
||||
m_opened=false;
|
||||
}
|
||||
|
||||
for (QMap<QDate, Day *>::iterator d = daylist.begin(); d != daylist.end(); d++) {
|
||||
delete d.value();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
bool Profile::Save(QString filename)
|
||||
@ -572,16 +577,16 @@ QDomElement Profile::ExtraSave(QDomDocument &doc)
|
||||
}
|
||||
|
||||
return mach;
|
||||
|
||||
}
|
||||
|
||||
void Profile::AddDay(QDate date, Day *day, MachineType mt)
|
||||
|
||||
Day *Profile::addDay(QDate date)
|
||||
{
|
||||
//date+=wxTimeSpan::Day();
|
||||
if (!day) {
|
||||
qDebug() << "Profile::AddDay called with null day object";
|
||||
return;
|
||||
QMap<QDate, Day *>::iterator dit = daylist.find(date);
|
||||
if (dit == daylist.end()) {
|
||||
dit = daylist.insert(date, new Day());
|
||||
}
|
||||
Day * day = dit.value();
|
||||
|
||||
if (is_first_day) {
|
||||
m_first = m_last = date;
|
||||
@ -595,23 +600,7 @@ void Profile::AddDay(QDate date, Day *day, MachineType mt)
|
||||
if (m_last < date) {
|
||||
m_last = date;
|
||||
}
|
||||
|
||||
// Check for any other machines of same type.. Throw an exception if one already exists.
|
||||
QList<Day *> &dl = daylist[date];
|
||||
|
||||
for (QList<Day *>::iterator a = dl.begin(); a != dl.end(); a++) {
|
||||
if ((*a)->machine->type() == mt) {
|
||||
|
||||
// disabled this because two machines isn't all that bad
|
||||
// if (QMessageBox::question(nullptr,"Different Machine Detected","This data comes from another machine to what's usually imported, and has overlapping data.\nThis new data will override any older data from the old machine. Are you sure you want to do this?",QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) {
|
||||
// throw OneTypePerDay();
|
||||
// }
|
||||
daylist[date].erase(a);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
daylist[date].push_back(day);
|
||||
return day;
|
||||
}
|
||||
|
||||
// Get Day record if data available for date and machine type,
|
||||
@ -622,14 +611,10 @@ Day *Profile::GetGoodDay(QDate date, MachineType type)
|
||||
if (!day)
|
||||
return nullptr;
|
||||
|
||||
// Just return the day if not matching for a machine.
|
||||
if (type == MT_UNKNOWN)
|
||||
return day;
|
||||
|
||||
// For a machine match, find at least one enabled Session.
|
||||
Q_ASSERT(day->machine_type() == type);
|
||||
for (int i = 0; i < day->size(); ++i) {
|
||||
if ((*day)[i]->enabled())
|
||||
Session * sess = (*day)[i];
|
||||
if (((type == MT_UNKNOWN) || (sess->machine()->type() == type)) && sess->enabled())
|
||||
return day;
|
||||
}
|
||||
|
||||
@ -639,24 +624,14 @@ Day *Profile::GetGoodDay(QDate date, MachineType type)
|
||||
|
||||
Day *Profile::GetDay(QDate date, MachineType type)
|
||||
{
|
||||
if (!daylist.contains(date))
|
||||
return nullptr;
|
||||
QMap<QDate, Day *>::iterator di = daylist.find(date);
|
||||
if (di == daylist.end()) return nullptr;
|
||||
|
||||
QList<Day *> list(daylist.value(date));
|
||||
Day * day = di.value();
|
||||
|
||||
QList<Day *>::iterator it = list.begin();
|
||||
QList<Day *>::iterator list_end = list.end();
|
||||
if (type == MT_UNKNOWN) return day; // just want the day record
|
||||
|
||||
for (; it != list_end; ++it) {
|
||||
Day * day = (*it);
|
||||
|
||||
Q_ASSERT(day != nullptr);
|
||||
|
||||
// Just return the day if not matching for a machine.
|
||||
if (day->machine_type() == type || type == MT_UNKNOWN) {
|
||||
return day;
|
||||
}
|
||||
}
|
||||
if (day->machines.contains(type)) return day;
|
||||
|
||||
return nullptr;
|
||||
}
|
||||
@ -765,33 +740,17 @@ Machine *Profile::GetMachine(MachineType t)
|
||||
|
||||
bool Profile::unlinkDay(Day * day)
|
||||
{
|
||||
bool b=false;
|
||||
|
||||
QList<QDate> dates;
|
||||
|
||||
QMap<QDate, QList<Day *> >::iterator it;
|
||||
QMap<QDate, QList<Day *> >::iterator it_end = daylist.end();
|
||||
QMap<QDate, Day *>::iterator it;
|
||||
QMap<QDate, Day *>::iterator it_end = daylist.end();
|
||||
|
||||
// Find the key...
|
||||
for (it = daylist.begin(); it != it_end; ++it) {
|
||||
if (it.value().contains(day)) {
|
||||
dates.push_back(it.key());
|
||||
if (it.value() == day) {
|
||||
daylist.erase(it);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
for (int i=0; i < dates.size(); ++i) {
|
||||
it = daylist.find(dates.at(i));
|
||||
|
||||
if (it != daylist.end()) {
|
||||
it.value().removeAll(day);
|
||||
// TODO: Check it doesn't change from the above...
|
||||
|
||||
if (it.value().size() == 0) {
|
||||
daylist.erase(it);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return b;
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
@ -847,6 +806,7 @@ Profile *Create(QString name)
|
||||
p_profile->Set(STR_GEN_DataFolder, QString("{home}/Profiles/{") + QString(STR_UI_UserName) + QString("}"));
|
||||
|
||||
Machine *m = new Machine(0);
|
||||
m->setType(MT_JOURNAL);
|
||||
MachineInfo info(MT_JOURNAL, 0, STR_MACH_Journal, "SleepyHead", STR_MACH_Journal, QString(), m->hexid(), QString(), QDateTime::currentDateTime(), 0);
|
||||
|
||||
m->setInfo(info);
|
||||
@ -906,35 +866,42 @@ void Scan()
|
||||
// Returns a list of all days records matching machine type between start and end date
|
||||
QList<Day *> Profile::getDays(MachineType mt, QDate start, QDate end)
|
||||
{
|
||||
QList<Day *> daylist;
|
||||
QList<Day *> list;
|
||||
|
||||
if (!start.isValid()) {
|
||||
return daylist;
|
||||
return list;
|
||||
}
|
||||
|
||||
if (!end.isValid()) {
|
||||
return daylist;
|
||||
return list;
|
||||
}
|
||||
|
||||
QDate date = start;
|
||||
|
||||
if (date.isNull()) {
|
||||
return daylist;
|
||||
return list;
|
||||
}
|
||||
|
||||
do {
|
||||
Day *day = GetGoodDay(date, mt);
|
||||
QMap<QDate, Day *>::iterator it;
|
||||
|
||||
if (day) {
|
||||
if ((mt == MT_UNKNOWN) || (day->machine->type() == mt)) {
|
||||
daylist.push_back(day);
|
||||
do {
|
||||
it = daylist.find(date);
|
||||
if (it != daylist.end()) {
|
||||
Day *day = it.value();
|
||||
if (mt != MT_UNKNOWN) {
|
||||
if (day->hasEnabledSessions(mt)) {
|
||||
list.push_back(day);
|
||||
}
|
||||
} else {
|
||||
if (day->hasEnabledSessions()) {
|
||||
list.push_back(day);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
date = date.addDays(1);
|
||||
} while (date <= end);
|
||||
|
||||
return daylist;
|
||||
return list;
|
||||
}
|
||||
|
||||
int Profile::countDays(MachineType mt, QDate start, QDate end)
|
||||
@ -959,7 +926,7 @@ int Profile::countDays(MachineType mt, QDate start, QDate end)
|
||||
Day *day = GetGoodDay(date, mt);
|
||||
|
||||
if (day) {
|
||||
if ((mt == MT_UNKNOWN) || (day->machine->type() == mt)) { days++; }
|
||||
days++;
|
||||
}
|
||||
|
||||
date = date.addDays(1);
|
||||
@ -993,7 +960,7 @@ int Profile::countCompliantDays(MachineType mt, QDate start, QDate end)
|
||||
Day *day = GetGoodDay(date, mt);
|
||||
|
||||
if (day) {
|
||||
if ((day->machine->type() == mt) && (day->hours() > compliance)) { days++; }
|
||||
if (day->hours(mt) > compliance) { days++; }
|
||||
}
|
||||
|
||||
date = date.addDays(1);
|
||||
@ -1671,9 +1638,7 @@ bool Profile::hasChannel(ChannelID code)
|
||||
return false;
|
||||
}
|
||||
|
||||
QMap<QDate, QList<Day *> >::iterator dit;
|
||||
QList<Day *>::iterator di;
|
||||
QList<Day *>::iterator di_end;
|
||||
QMap<QDate, Day *>::iterator dit;
|
||||
|
||||
bool found = false;
|
||||
|
||||
@ -1681,23 +1646,14 @@ bool Profile::hasChannel(ChannelID code)
|
||||
dit = daylist.find(d);
|
||||
|
||||
if (dit != daylist.end()) {
|
||||
Day *day = dit.value();
|
||||
|
||||
di = dit.value().begin();
|
||||
di_end = dit.value().end();
|
||||
for (; di != di_end; ++di) {
|
||||
Day *day = (*di);
|
||||
|
||||
if (day->channelHasData(code)) {
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
if (day->channelHasData(code)) {
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (found) {
|
||||
break;
|
||||
}
|
||||
|
||||
d = d.addDays(-1);
|
||||
} while (d >= f);
|
||||
|
||||
|
@ -82,7 +82,7 @@ class Profile : public Preferences
|
||||
// bool trashMachine(Machine * mach);
|
||||
|
||||
//! \brief Add Day record to Profile Day list
|
||||
void AddDay(QDate date, Day *day, MachineType mt);
|
||||
Day *addDay(QDate date);
|
||||
|
||||
//! \brief Get Day record if data available for date and machine type, else return nullptr
|
||||
Day *GetDay(QDate date, MachineType type = MT_UNKNOWN);
|
||||
@ -186,8 +186,8 @@ class Profile : public Preferences
|
||||
//! \brief Return if this profile has been opened or not
|
||||
bool isOpen() { return m_opened; }
|
||||
|
||||
//! \brief Red-Black tree of Days (iterates in order).
|
||||
QMap<QDate, QList<Day *> > daylist;
|
||||
//! \brief QMap of day records (iterates in order).
|
||||
QMap<QDate, Day *> daylist;
|
||||
|
||||
//! \brief List of machines, indexed by MachineID.
|
||||
QHash<MachineID, Machine *> machlist;
|
||||
|
@ -168,7 +168,9 @@ bool Session::Store(QString path)
|
||||
|
||||
bool Session::StoreSummary(QString filename)
|
||||
{
|
||||
|
||||
if (filename.isEmpty()) {
|
||||
filename = machine()->getDataPath() + QString().sprintf("%08lx.000", s_session);
|
||||
}
|
||||
QFile file(filename);
|
||||
file.open(QIODevice::WriteOnly);
|
||||
|
||||
@ -476,6 +478,10 @@ const quint16 compress_method = 1;
|
||||
|
||||
bool Session::StoreEvents(QString filename)
|
||||
{
|
||||
if (filename.isEmpty()) {
|
||||
filename = machine()->getDataPath() + QString().sprintf("%08lx.001", s_session);
|
||||
}
|
||||
|
||||
QFile file(filename);
|
||||
file.open(QIODevice::WriteOnly);
|
||||
|
||||
|
@ -40,10 +40,10 @@ class Session
|
||||
bool Store(QString path);
|
||||
|
||||
//! \brief Writes the Sessions Summary Indexes to filename, in SleepLibs custom data format.
|
||||
bool StoreSummary(QString filename);
|
||||
bool StoreSummary(QString filename = QString());
|
||||
|
||||
//! \brief Writes the Sessions EventLists to filename, in SleepLibs custom data format.
|
||||
bool StoreEvents(QString filename);
|
||||
bool StoreEvents(QString filename = QString());
|
||||
|
||||
//bool Load(QString path);
|
||||
|
||||
|
@ -126,7 +126,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared)
|
||||
GraphView=new gGraphView(ui->graphFrame,shared);
|
||||
GraphView->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding);
|
||||
|
||||
snapGV=new gGraphView(GraphView); //ui->graphMainArea);
|
||||
snapGV=new gGraphView(GraphView);
|
||||
snapGV->setMinimumSize(172,172);
|
||||
snapGV->hideSplitter();
|
||||
snapGV->hide();
|
||||
@ -520,15 +520,10 @@ void Daily::closeEvent(QCloseEvent *event)
|
||||
|
||||
void Daily::doToggleSession(Session * sess)
|
||||
{
|
||||
Q_UNUSED(sess)
|
||||
sess->setEnabled(!sess->enabled());
|
||||
sess->StoreSummary();
|
||||
|
||||
// sess->StoreSummary();
|
||||
Day *day=p_profile->GetDay(previous_date,MT_CPAP);
|
||||
if (day) {
|
||||
day->machine->Save();
|
||||
this->LoadDate(previous_date);
|
||||
}
|
||||
LoadDate(previous_date);
|
||||
}
|
||||
|
||||
void Daily::Link_clicked(const QUrl &url)
|
||||
@ -537,21 +532,20 @@ void Daily::Link_clicked(const QUrl &url)
|
||||
QString data=url.toString().section("=",1);
|
||||
int sid=data.toInt();
|
||||
Day *day=nullptr;
|
||||
|
||||
if (code=="togglecpapsession") { // Enable/Disable CPAP session
|
||||
day=p_profile->GetDay(previous_date,MT_CPAP);
|
||||
if (!day) return;
|
||||
Session *sess=day->find(sid);
|
||||
if (!sess)
|
||||
return;
|
||||
int i=webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-webView->page()->mainFrame()->scrollBarValue(Qt::Vertical);
|
||||
sess->setEnabled(!sess->enabled());
|
||||
|
||||
// Messy, this rewrites both summary & events.. TODO: Write just the session summary file
|
||||
day->machine->Save();
|
||||
sess->StoreSummary();
|
||||
|
||||
// Reload day
|
||||
this->LoadDate(previous_date);
|
||||
LoadDate(previous_date);
|
||||
webView->page()->mainFrame()->setScrollBarValue(Qt::Vertical, webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-i);
|
||||
return;
|
||||
} else if (code=="toggleoxisession") { // Enable/Disable Oximetry session
|
||||
day=p_profile->GetDay(previous_date,MT_OXIMETER);
|
||||
Session *sess=day->find(sid);
|
||||
@ -559,19 +553,29 @@ void Daily::Link_clicked(const QUrl &url)
|
||||
return;
|
||||
int i=webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-webView->page()->mainFrame()->scrollBarValue(Qt::Vertical);
|
||||
sess->setEnabled(!sess->enabled());
|
||||
// Messy, this rewrites both summary & events.. TODO: Write just the session summary file
|
||||
day->machine->Save();
|
||||
|
||||
sess->StoreSummary();
|
||||
|
||||
// Reload day
|
||||
this->LoadDate(previous_date);
|
||||
LoadDate(previous_date);
|
||||
webView->page()->mainFrame()->setScrollBarValue(Qt::Vertical, webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-i);
|
||||
return;
|
||||
} else if (code=="cpap") {
|
||||
day=p_profile->GetDay(previous_date,MT_CPAP);
|
||||
if (day) {
|
||||
Session *sess=day->machine(MT_CPAP)->sessionlist[sid];
|
||||
if (sess && sess->enabled()) {
|
||||
GraphView->SetXBounds(sess->first(),sess->last());
|
||||
}
|
||||
}
|
||||
} else if (code=="oxi") {
|
||||
//day=p_profile->GetDay(previous_date,MT_OXIMETER);
|
||||
//Session *sess=day->machine->sessionlist[sid];
|
||||
return;
|
||||
day=p_profile->GetDay(previous_date,MT_OXIMETER);
|
||||
if (day) {
|
||||
Session *sess=day->machine(MT_OXIMETER)->sessionlist[sid];
|
||||
if (sess && sess->enabled()) {
|
||||
GraphView->SetXBounds(sess->first(),sess->last());
|
||||
}
|
||||
}
|
||||
|
||||
} else if (code=="event") {
|
||||
QList<QTreeWidgetItem *> list=ui->treeWidget->findItems(schema::channel[sid].fullname(),Qt::MatchContains);
|
||||
if (list.size()>0) {
|
||||
@ -588,13 +592,6 @@ void Daily::Link_clicked(const QUrl &url)
|
||||
} else {
|
||||
qDebug() << "Clicked on" << code << data;
|
||||
}
|
||||
if (day) {
|
||||
|
||||
Session *sess=day->machine->sessionlist[sid];
|
||||
if (sess && sess->enabled()) {
|
||||
GraphView->SetXBounds(sess->first(),sess->last());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void Daily::ReloadGraphs()
|
||||
@ -916,24 +913,19 @@ MyWebView::MyWebView(QWidget *parent):
|
||||
}
|
||||
|
||||
|
||||
QString Daily::getSessionInformation(Day * cpap, Day * oxi, Day * stage, Day * posit)
|
||||
QString Daily::getSessionInformation(Day * day)
|
||||
{
|
||||
QString html;
|
||||
QList<Day *> list;
|
||||
if (cpap) list.push_back(cpap);
|
||||
if (oxi) list.push_back(oxi);
|
||||
if (stage) list.push_back(stage);
|
||||
if (posit) list.push_back(posit);
|
||||
|
||||
|
||||
if (list.isEmpty())
|
||||
return html;
|
||||
if (!day) return html;
|
||||
|
||||
html="<table cellpadding=0 cellspacing=0 border=0 width=100%>";
|
||||
html+=QString("<tr><td colspan=5 align=center><b>"+tr("Session Information")+"</b></td></tr>");
|
||||
html+="<tr><td colspan=5 align=center> </td></tr>";
|
||||
QFontMetrics FM(*defaultfont);
|
||||
QRect r=FM.boundingRect('@');
|
||||
|
||||
Machine * cpap = day->machine(MT_CPAP);
|
||||
|
||||
if (cpap) {
|
||||
html+=QString("<tr><td colspan=5 align=center>"
|
||||
"<object type=\"application/x-qt-plugin\" classid=\"SessionBar\" name=\"sessbar\" height=%1 width=100%></object>"
|
||||
@ -949,30 +941,32 @@ QString Daily::getSessionInformation(Day * cpap, Day * oxi, Day * stage, Day * p
|
||||
bool corrupted_waveform=false;
|
||||
QString tooltip;
|
||||
|
||||
QList<Day *>::iterator di;
|
||||
QString type;
|
||||
|
||||
for (di=list.begin();di!=list.end();di++) {
|
||||
Day * day=*di;
|
||||
html+="<tr><td colspan=5 align=center><i>";
|
||||
switch (day->machine_type()) {
|
||||
case MT_CPAP: type="cpap";
|
||||
html+=tr("CPAP Sessions");
|
||||
break;
|
||||
case MT_OXIMETER: type="oxi";
|
||||
html+=tr("Oximetery Sessions");
|
||||
break;
|
||||
case MT_SLEEPSTAGE: type="stage";
|
||||
html+=tr("Sleep Stage Sessions");
|
||||
break;
|
||||
case MT_POSITION: type="stage";
|
||||
html+=tr("Position Sensor Sessions");
|
||||
break;
|
||||
QHash<MachineType, Machine *>::iterator mach_end = day->machines.end();
|
||||
QHash<MachineType, Machine *>::iterator mi;
|
||||
|
||||
default:
|
||||
type="unknown";
|
||||
html+=tr("Unknown Session");
|
||||
break;
|
||||
for (mi = day->machines.begin(); mi != mach_end; ++mi) {
|
||||
if (mi.key() == MT_JOURNAL) continue;
|
||||
html += "<tr><td colspan=5 align=center><i>";
|
||||
switch (mi.key()) {
|
||||
case MT_CPAP: type="cpap";
|
||||
html+=tr("CPAP Sessions");
|
||||
break;
|
||||
case MT_OXIMETER: type="oxi";
|
||||
html+=tr("Oximetery Sessions");
|
||||
break;
|
||||
case MT_SLEEPSTAGE: type="stage";
|
||||
html+=tr("Sleep Stage Sessions");
|
||||
break;
|
||||
case MT_POSITION: type="stage";
|
||||
html+=tr("Position Sensor Sessions");
|
||||
break;
|
||||
|
||||
default:
|
||||
type="unknown";
|
||||
html+=tr("Unknown Session");
|
||||
break;
|
||||
}
|
||||
html+="</i></td></tr>\n";
|
||||
html+=QString("<tr>"
|
||||
@ -981,10 +975,12 @@ QString Daily::getSessionInformation(Day * cpap, Day * oxi, Day * stage, Day * p
|
||||
"<th>"+STR_TR_Start+"</th>"
|
||||
"<th>"+STR_TR_End+"</th>"
|
||||
"<th>"+tr("Duration")+"</th></tr>");
|
||||
for (QList<Session *>::iterator s=day->begin();s!=day->end();++s) {
|
||||
|
||||
if ((day->machine_type()==MT_CPAP) &&
|
||||
((*s)->settings.find(CPAP_BrokenWaveform)!=(*s)->settings.end()))
|
||||
QList<Session *> sesslist = day->getSessions(mi.key());
|
||||
|
||||
for (QList<Session *>::iterator s=sesslist.begin(); s != sesslist.end(); ++s) {
|
||||
if (((*s)->machine()->type() == MT_CPAP) &&
|
||||
((*s)->settings.find(CPAP_BrokenWaveform) != (*s)->settings.end()))
|
||||
corrupted_waveform=true;
|
||||
|
||||
fd=QDateTime::fromTime_t((*s)->first()/1000L);
|
||||
@ -993,13 +989,6 @@ QString Daily::getSessionInformation(Day * cpap, Day * oxi, Day * stage, Day * p
|
||||
int h=len/3600;
|
||||
int m=(len/60) % 60;
|
||||
int s1=len % 60;
|
||||
//tooltip=day->machine->loaderName()+QString(":#%1").arg((*s)->session(),8,10,QChar('0'));
|
||||
|
||||
//#define DEBUG_SESSIONS
|
||||
//#ifdef DEBUG_SESSIONS
|
||||
// tooltip += " "+QString::number(len)+"s";
|
||||
//#endif
|
||||
// tooltip needs to lookup language.. :-/
|
||||
|
||||
Session *sess=*s;
|
||||
if (!sess->settings.contains(SESSION_ENABLED)) {
|
||||
@ -1034,23 +1023,32 @@ QString Daily::getSessionInformation(Day * cpap, Day * oxi, Day * stage, Day * p
|
||||
return html;
|
||||
}
|
||||
|
||||
QString Daily::getMachineSettings(Day * cpap) {
|
||||
QString Daily::getMachineSettings(Day * day) {
|
||||
QString html;
|
||||
if (cpap && cpap->hasEnabledSessions()) {
|
||||
|
||||
Machine * cpap = day->machine(MT_CPAP);
|
||||
if (cpap && day->hasEnabledSessions(MT_CPAP)) {
|
||||
html="<table cellpadding=0 cellspacing=0 border=0 width=100%>";
|
||||
html+=QString("<tr><td colspan=5 align=center><b>%1</b></td></tr>").arg(tr("Machine Settings"));
|
||||
html+="<tr><td colspan=5> </td></tr>";
|
||||
|
||||
if ((cpap && cpap->settingExists(CPAP_BrokenSummary))) {
|
||||
if ((day->settingExists(CPAP_BrokenSummary))) {
|
||||
html+="<tr><td colspan=5 align=center><i>"+tr("Machine Settings Unavailable")+"</i></td></tr></table><hr/>\n";
|
||||
return html;
|
||||
}
|
||||
|
||||
QMap<QString, QString> other;
|
||||
QHash<ChannelID, QVariant>::iterator it = cpap->sessions.at(0)->settings.begin();
|
||||
QHash<ChannelID, QVariant>::iterator it_end = cpap->sessions.at(0)->settings.end();
|
||||
Session * sess = day->firstSession(MT_CPAP);
|
||||
|
||||
QHash<ChannelID, QVariant>::iterator it;
|
||||
QHash<ChannelID, QVariant>::iterator it_end;
|
||||
if (sess) {
|
||||
it_end = sess->settings.end();
|
||||
it = sess->settings.begin();
|
||||
}
|
||||
QMap<int, QString> first;
|
||||
for (; it != it_end; ++it) {
|
||||
|
||||
if (sess) for (; it != it_end; ++it) {
|
||||
ChannelID code = it.key();
|
||||
|
||||
if ((code <= 1) || (code == RMS9_MaskOnTime)) continue;
|
||||
@ -1137,31 +1135,36 @@ QString Daily::getMachineSettings(Day * cpap) {
|
||||
return html;
|
||||
}
|
||||
|
||||
QString Daily::getOximeterInformation(Day * oxi)
|
||||
QString Daily::getOximeterInformation(Day * day)
|
||||
{
|
||||
QString html;
|
||||
if (oxi && oxi->hasEnabledSessions()) {
|
||||
Machine * oxi = day->machine(MT_OXIMETER);
|
||||
if (oxi && day->hasEnabledSessions(MT_OXIMETER)) {
|
||||
html="<table cellpadding=0 cellspacing=0 border=0 width=100%>";
|
||||
html+=QString("<tr><td colspan=5 align=center><b>%1</b></td></tr>\n").arg(tr("Oximeter Information"));
|
||||
html+="<tr><td colspan=5 align=center> </td></tr>";
|
||||
html+="<tr><td colspan=5 align=center>"+oxi->machine->brand()+" "+oxi->machine->series()+"</td></tr>\n";
|
||||
html+="<tr><td colspan=5 align=center>"+oxi->brand()+" "+oxi->series()+"</td></tr>\n";
|
||||
html+="<tr><td colspan=5 align=center> </td></tr>";
|
||||
html+=QString("<tr><td colspan=5 align=center>%1: %2 (%3%)</td></tr>").arg(tr("SpO2 Desaturations")).arg(oxi->count(OXI_SPO2Drop)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_SPO2Drop)/3600.0),0,'f',2);
|
||||
html+=QString("<tr><td colspan=5 align=center>%1: %2 (%3%)</td></tr>").arg(tr("Pulse Change events")).arg(oxi->count(OXI_PulseChange)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_PulseChange)/3600.0),0,'f',2);
|
||||
html+=QString("<tr><td colspan=5 align=center>%1: %2%</td></tr>").arg(tr("SpO2 Baseline Used")).arg(oxi->settings_wavg(OXI_SPO2Drop),0,'f',2); // CHECKME: Should this value be wavg OXI_SPO2 isntead?
|
||||
html+=QString("<tr><td colspan=5 align=center>%1: %2 (%3%)</td></tr>").arg(tr("SpO2 Desaturations")).arg(day->count(OXI_SPO2Drop)).arg((100.0/day->hours(MT_OXIMETER)) * (day->sum(OXI_SPO2Drop)/3600.0),0,'f',2);
|
||||
html+=QString("<tr><td colspan=5 align=center>%1: %2 (%3%)</td></tr>").arg(tr("Pulse Change events")).arg(day->count(OXI_PulseChange)).arg((100.0/day->hours(MT_OXIMETER)) * (day->sum(OXI_PulseChange)/3600.0),0,'f',2);
|
||||
html+=QString("<tr><td colspan=5 align=center>%1: %2%</td></tr>").arg(tr("SpO2 Baseline Used")).arg(day->settings_wavg(OXI_SPO2Drop),0,'f',2); // CHECKME: Should this value be wavg OXI_SPO2 isntead?
|
||||
html+="</table>\n";
|
||||
html+="<hr/>\n";
|
||||
}
|
||||
return html;
|
||||
}
|
||||
|
||||
QString Daily::getCPAPInformation(Day * cpap)
|
||||
QString Daily::getCPAPInformation(Day * day)
|
||||
{
|
||||
QString html;
|
||||
if (!cpap)
|
||||
if (!day)
|
||||
return html;
|
||||
|
||||
MachineInfo info = cpap->machine->getInfo();
|
||||
Machine * cpap = day->machine(MT_CPAP);
|
||||
if (!cpap) return html;
|
||||
|
||||
|
||||
MachineInfo info = cpap->getInfo();
|
||||
|
||||
html="<table cellspacing=0 cellpadding=0 border=0 width='100%'>\n";
|
||||
|
||||
@ -1174,10 +1177,10 @@ QString Daily::getCPAPInformation(Day * cpap)
|
||||
//CPAPMode mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode);
|
||||
html+="<tr><td align=center>";
|
||||
|
||||
html+=tr("PAP Mode: %1<br/>").arg(cpap->getCPAPMode());
|
||||
html+= cpap->getPressureSettings();
|
||||
html+=tr("PAP Mode: %1<br/>").arg(day->getCPAPMode());
|
||||
html+= day->getPressureSettings();
|
||||
html+="</td></tr>\n";
|
||||
if ((cpap && cpap->settingExists(CPAP_BrokenSummary))) {
|
||||
if ((day->settingExists(CPAP_BrokenSummary))) {
|
||||
html+="<tr><td> </td></tr>\n";
|
||||
html+=QString("<tr><td colspan=2><i>%1</i></td></tr>").arg("<b>"+STR_MessageBox_PleaseNote+":</b> "+ tr("This day has missing pressure, mode and settings data."));
|
||||
}
|
||||
@ -1188,14 +1191,13 @@ QString Daily::getCPAPInformation(Day * cpap)
|
||||
}
|
||||
|
||||
|
||||
QString Daily::getStatisticsInfo(Day * cpap,Day * oxi,Day *pos)
|
||||
QString Daily::getStatisticsInfo(Day * day)
|
||||
{
|
||||
if (!day) return QString();
|
||||
|
||||
QList<Day *> list;
|
||||
|
||||
list.push_back(cpap);
|
||||
list.push_back(oxi);
|
||||
list.push_back(pos);
|
||||
Machine *cpap = day->machine(MT_CPAP),
|
||||
*oxi = day->machine(MT_OXIMETER),
|
||||
*pos = day->machine(MT_POSITION);
|
||||
|
||||
|
||||
int mididx=p_profile->general->prefCalcMiddle();
|
||||
@ -1235,73 +1237,65 @@ QString Daily::getStatisticsInfo(Day * cpap,Day * oxi,Day *pos)
|
||||
int ccnt=0;
|
||||
EventDataType tmp,med,perc,mx,mn;
|
||||
|
||||
QList<Day *>::iterator di;
|
||||
for (int i=0;i<numchans;i++) {
|
||||
|
||||
for (di=list.begin();di!=list.end();di++) {
|
||||
Day * day=*di;
|
||||
ChannelID code=chans[i];
|
||||
|
||||
if (!day)
|
||||
if (!day->channelHasData(code))
|
||||
continue;
|
||||
|
||||
for (int i=0;i<numchans;i++) {
|
||||
QString tooltip=schema::channel[code].description();
|
||||
|
||||
ChannelID code=chans[i];
|
||||
if (!schema::channel[code].units().isEmpty()) tooltip+=" ("+schema::channel[code].units()+")";
|
||||
|
||||
if (!day->channelHasData(code))
|
||||
continue;
|
||||
|
||||
QString tooltip=schema::channel[code].description();
|
||||
|
||||
if (!schema::channel[code].units().isEmpty()) tooltip+=" ("+schema::channel[code].units()+")";
|
||||
|
||||
if (ST_max == ST_MAX) {
|
||||
mx=day->Max(code);
|
||||
} else {
|
||||
mx=day->percentile(code,maxperc);
|
||||
}
|
||||
|
||||
mn=day->Min(code);
|
||||
perc=day->percentile(code,percentile);
|
||||
|
||||
if (ST_mid == ST_PERC) {
|
||||
med=day->percentile(code,0.5);
|
||||
tmp=day->wavg(code);
|
||||
if (tmp>0 || mx==0) {
|
||||
tooltip+=QString("<br/>"+STR_TR_WAvg+": %1").arg(tmp,0,'f',2);
|
||||
}
|
||||
} else if (ST_mid == ST_WAVG) {
|
||||
med=day->wavg(code);
|
||||
tmp=day->percentile(code,0.5);
|
||||
if (tmp>0 || mx==0) {
|
||||
tooltip+=QString("<br/>"+STR_TR_Median+": %1").arg(tmp,0,'f',2);
|
||||
}
|
||||
} else if (ST_mid == ST_AVG) {
|
||||
med=day->avg(code);
|
||||
tmp=day->percentile(code,0.5);
|
||||
if (tmp>0 || mx==0) {
|
||||
tooltip+=QString("<br/>"+STR_TR_Median+": %1").arg(tmp,0,'f',2);
|
||||
}
|
||||
}
|
||||
|
||||
html+=QString("<tr class='datarow'><td align=left class='info' onmouseover=\"style.color='blue';\" onmouseout=\"style.color='"+COLOR_Text.name()+"';\">%1<span>%6</span></td><td>%2</td><td>%3</td><td>%4</td><td>%5</td></tr>")
|
||||
.arg(schema::channel[code].label())
|
||||
.arg(mn,0,'f',2)
|
||||
.arg(med,0,'f',2)
|
||||
.arg(perc,0,'f',2)
|
||||
.arg(mx,0,'f',2)
|
||||
.arg(tooltip);
|
||||
ccnt++;
|
||||
if (ST_max == ST_MAX) {
|
||||
mx=day->Max(code);
|
||||
} else {
|
||||
mx=day->percentile(code,maxperc);
|
||||
}
|
||||
|
||||
mn=day->Min(code);
|
||||
perc=day->percentile(code,percentile);
|
||||
|
||||
if (ST_mid == ST_PERC) {
|
||||
med=day->percentile(code,0.5);
|
||||
tmp=day->wavg(code);
|
||||
if (tmp>0 || mx==0) {
|
||||
tooltip+=QString("<br/>"+STR_TR_WAvg+": %1").arg(tmp,0,'f',2);
|
||||
}
|
||||
} else if (ST_mid == ST_WAVG) {
|
||||
med=day->wavg(code);
|
||||
tmp=day->percentile(code,0.5);
|
||||
if (tmp>0 || mx==0) {
|
||||
tooltip+=QString("<br/>"+STR_TR_Median+": %1").arg(tmp,0,'f',2);
|
||||
}
|
||||
} else if (ST_mid == ST_AVG) {
|
||||
med=day->avg(code);
|
||||
tmp=day->percentile(code,0.5);
|
||||
if (tmp>0 || mx==0) {
|
||||
tooltip+=QString("<br/>"+STR_TR_Median+": %1").arg(tmp,0,'f',2);
|
||||
}
|
||||
}
|
||||
|
||||
html+=QString("<tr class='datarow'><td align=left class='info' onmouseover=\"style.color='blue';\" onmouseout=\"style.color='"+COLOR_Text.name()+"';\">%1<span>%6</span></td><td>%2</td><td>%3</td><td>%4</td><td>%5</td></tr>")
|
||||
.arg(schema::channel[code].label())
|
||||
.arg(mn,0,'f',2)
|
||||
.arg(med,0,'f',2)
|
||||
.arg(perc,0,'f',2)
|
||||
.arg(mx,0,'f',2)
|
||||
.arg(tooltip);
|
||||
ccnt++;
|
||||
|
||||
}
|
||||
|
||||
if (GraphView->isEmpty() && ((ccnt>0) || (cpap && cpap->summaryOnly()))) {
|
||||
if (GraphView->isEmpty() && ((ccnt>0) || (cpap && day->summaryOnly()))) {
|
||||
html+="<tr><td colspan=5> </td></tr>\n";
|
||||
html+=QString("<tr><td colspan=5 align=center><i>%1</i></td></tr>").arg("<b>"+STR_MessageBox_PleaseNote+"</b> "+ tr("This day just contains summary data, only limited information is available ."));
|
||||
} else if (cpap) {
|
||||
html+="<tr><td colspan=5> </td></tr>";
|
||||
|
||||
if ((cpap->machine->loaderName() == STR_MACH_ResMed) || ((cpap->machine->loaderName() == STR_MACH_PRS1) && (p_profile->cpap->resyncFromUserFlagging()))) {
|
||||
int ttia = cpap->sum(CPAP_Obstructive) + cpap->sum(CPAP_ClearAirway) + cpap->sum(CPAP_Apnea) + cpap->sum(CPAP_Hypopnea);
|
||||
if ((cpap->loaderName() == STR_MACH_ResMed) || ((cpap->loaderName() == STR_MACH_PRS1) && (p_profile->cpap->resyncFromUserFlagging()))) {
|
||||
int ttia = day->sum(CPAP_Obstructive) + day->sum(CPAP_ClearAirway) + day->sum(CPAP_Apnea) + day->sum(CPAP_Hypopnea);
|
||||
int h = ttia / 3600;
|
||||
int m = ttia / 60 % 60;
|
||||
int s = ttia % 60;
|
||||
@ -1311,28 +1305,31 @@ QString Daily::getStatisticsInfo(Day * cpap,Day * oxi,Day *pos)
|
||||
}
|
||||
|
||||
}
|
||||
float hours = day->hours(MT_CPAP);
|
||||
|
||||
if (p_profile->cpap->showLeakRedline()) {
|
||||
float rlt = cpap->timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline()) / 60.0;
|
||||
float pc = 100.0 / cpap->hours() * rlt;
|
||||
float rlt = day->timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline()) / 60.0;
|
||||
float pc = 100.0 / hours * rlt;
|
||||
html+="<tr><td colspan=3 align='left' bgcolor='white'><b>"+tr("Time over leak redline")+
|
||||
QString("</b></td><td colspan=2 bgcolor='white'>%1%</td></tr>").arg(pc, 0, 'f', 3);
|
||||
}
|
||||
int l = cpap->sum(CPAP_Ramp);
|
||||
int l = day->sum(CPAP_Ramp);
|
||||
|
||||
if (l > 0) {
|
||||
html+="<tr><td colspan=3 align='left' bgcolor='white'>"+tr("Total ramp time")+
|
||||
QString("</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'));
|
||||
float v = (cpap->hours() - (float(l) / 3600.0));
|
||||
float v = (hours - (float(l) / 3600.0));
|
||||
int q = v * 3600.0;
|
||||
html+="<tr><td colspan=3 align='left' bgcolor='white'>"+tr("Time outside of ramp")+
|
||||
QString("</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 hc = day->count(CPAP_Hypopnea) - day->countInsideSpan(CPAP_Ramp, CPAP_Hypopnea);
|
||||
EventDataType oc = day->count(CPAP_Obstructive) - day->countInsideSpan(CPAP_Ramp, CPAP_Obstructive);
|
||||
|
||||
EventDataType tc = cpap->count(CPAP_Hypopnea) + cpap->count(CPAP_Obstructive);
|
||||
EventDataType tc = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive);
|
||||
EventDataType ahi = (hc+oc) / v;
|
||||
// Not sure if i was trying to be funny, and left on my replication of Devilbiss's bug here... :P
|
||||
|
||||
html+="<tr><td colspan=3 align='left' bgcolor='white'>"+tr("AHI excluding ramp")+
|
||||
QString("</td><td colspan=2 bgcolor='white'>%1</td></tr>").arg(ahi, 0, 'f', 2);
|
||||
}
|
||||
@ -1355,17 +1352,14 @@ QString Daily::getEventBreakdown(Day * cpap)
|
||||
return html;
|
||||
}
|
||||
|
||||
QString Daily::getSleepTime(Day * cpap, Day * oxi)
|
||||
QString Daily::getSleepTime(Day * day)
|
||||
{
|
||||
//cpap, Day * oxi
|
||||
QString html;
|
||||
|
||||
Day * day=nullptr;
|
||||
if (cpap && cpap->hours()>0)
|
||||
day=cpap;
|
||||
else if (oxi && oxi->hours()>0)
|
||||
day=oxi;
|
||||
else
|
||||
if (!day || (day->hours() < 0.0000001))
|
||||
return html;
|
||||
|
||||
html+="<table cellspacing=0 cellpadding=0 border=0 width='100%'>\n";
|
||||
html+="<tr><td align='center'><b>"+STR_TR_Date+"</b></td><td align='center'><b>"+tr("Sleep")+"</b></td><td align='center'><b>"+tr("Wake")+"</b></td><td align='center'><b>"+STR_UNIT_Hours+"</b></td></tr>";
|
||||
int tt=qint64(day->total_time())/1000L;
|
||||
@ -1392,38 +1386,46 @@ void Daily::Load(QDate date)
|
||||
{
|
||||
dateDisplay->setText("<i>"+date.toString(Qt::SystemLocaleLongDate)+"</i>");
|
||||
previous_date=date;
|
||||
Day *cpap=p_profile->GetDay(date,MT_CPAP);
|
||||
Day *oxi=p_profile->GetDay(date,MT_OXIMETER);
|
||||
Day *stage=p_profile->GetDay(date,MT_SLEEPSTAGE);
|
||||
Day *posit=p_profile->GetDay(date,MT_POSITION);
|
||||
|
||||
Day * day = p_profile->GetDay(date);
|
||||
Machine *cpap = nullptr,
|
||||
*oxi = nullptr,
|
||||
*stage = nullptr,
|
||||
*posit = nullptr;
|
||||
|
||||
if (day) {
|
||||
cpap = day->machine(MT_CPAP);
|
||||
oxi = day->machine(MT_OXIMETER);
|
||||
stage = day->machine(MT_SLEEPSTAGE);
|
||||
posit = day->machine(MT_POSITION);
|
||||
}
|
||||
|
||||
if (!p_profile->session->cacheSessions()) {
|
||||
// Getting trashed on purge last day...
|
||||
|
||||
// lastcpapday can get purged and be invalid
|
||||
|
||||
|
||||
if (lastcpapday && (lastcpapday!=cpap)) {
|
||||
if (lastcpapday && (lastcpapday!=day)) {
|
||||
for (QList<Session *>::iterator s=lastcpapday->begin();s!=lastcpapday->end();++s) {
|
||||
(*s)->TrashEvents();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ((cpap && oxi) && oxi->hasEnabledSessions()) {
|
||||
int gr;
|
||||
// Don't really see a point in unlinked oximetery sessions anymore... All I can say is BLEH...
|
||||
// if ((cpap && oxi) && day->hasEnabledSessions(MT_OXIMETER)) {
|
||||
// int gr;
|
||||
|
||||
if (qAbs(cpap->first() - oxi->first())>30000) {
|
||||
mainwin->Notify(tr("Oximetry data exists for this day, but its timestamps are too different, so the Graphs will not be linked."),"",3000);
|
||||
gr=1;
|
||||
} else
|
||||
gr=0;
|
||||
// if (qAbs(day->first(MT_CPAP) - day->first(MT_OXIMETER)) > 30000) {
|
||||
// mainwin->Notify(tr("Oximetry data exists for this day, but its timestamps are too different, so the Graphs will not be linked."),"",3000);
|
||||
// gr=1;
|
||||
// } else
|
||||
// gr=0;
|
||||
|
||||
(*GraphView)[schema::channel[OXI_Pulse].code()]->setGroup(gr);
|
||||
(*GraphView)[schema::channel[OXI_SPO2].code()]->setGroup(gr);
|
||||
(*GraphView)[schema::channel[OXI_Plethy].code()]->setGroup(gr);
|
||||
}
|
||||
lastcpapday=cpap;
|
||||
// (*GraphView)[schema::channel[OXI_Pulse].code()]->setGroup(gr);
|
||||
// (*GraphView)[schema::channel[OXI_SPO2].code()]->setGroup(gr);
|
||||
// (*GraphView)[schema::channel[OXI_Plethy].code()]->setGroup(gr);
|
||||
// }
|
||||
lastcpapday=day;
|
||||
|
||||
QString html="<html><head><style type='text/css'>"
|
||||
"p,a,td,body { font-family: '"+QApplication::font().family()+"'; }"
|
||||
@ -1443,20 +1445,27 @@ void Daily::Load(QDate date)
|
||||
"<body leftmargin=0 rightmargin=0 topmargin=0 marginwidth=0 marginheight=0>";
|
||||
QString tmp;
|
||||
|
||||
UpdateOXIGraphs(oxi);
|
||||
UpdateCPAPGraphs(cpap);
|
||||
UpdateSTAGEGraphs(stage);
|
||||
UpdatePOSGraphs(posit);
|
||||
UpdateEventsTree(ui->treeWidget,cpap);
|
||||
if (day) {
|
||||
day->OpenEvents();
|
||||
}
|
||||
GraphView->setDay(day);
|
||||
|
||||
|
||||
// UpdateOXIGraphs(oxi);
|
||||
// UpdateCPAPGraphs(cpap);
|
||||
// UpdateSTAGEGraphs(stage);
|
||||
// UpdatePOSGraphs(posit);
|
||||
UpdateEventsTree(ui->treeWidget, day);
|
||||
|
||||
// FIXME:
|
||||
// Generating entire statistics because bookmarks may have changed.. (This updates the side panel too)
|
||||
mainwin->GenerateStatistics();
|
||||
|
||||
snapGV->setDay(cpap);
|
||||
snapGV->setDay(day);
|
||||
|
||||
GraphView->ResetBounds(false);
|
||||
// GraphView->ResetBounds(false);
|
||||
|
||||
// wtf is hiding the scrollbars for???
|
||||
if (!cpap && !oxi) {
|
||||
scrollbar->hide();
|
||||
} else {
|
||||
@ -1471,26 +1480,24 @@ void Daily::Load(QDate date)
|
||||
updateGraphCombo();
|
||||
ui->eventsCombo->clear();
|
||||
|
||||
if (cpap) {
|
||||
quint32 chans = schema::SPAN | schema::FLAG | schema::MINOR_FLAG;
|
||||
if (p_profile->general->showUnknownFlags()) chans |= schema::UNKNOWN;
|
||||
quint32 chans = schema::SPAN | schema::FLAG | schema::MINOR_FLAG;
|
||||
if (p_profile->general->showUnknownFlags()) chans |= schema::UNKNOWN;
|
||||
|
||||
QList<ChannelID> available = cpap->getSortedMachineChannels(chans);
|
||||
QList<ChannelID> available = day->getSortedMachineChannels(chans);
|
||||
|
||||
for (int i=0; i < available.size(); ++i) {
|
||||
ChannelID code = available.at(i);
|
||||
schema::Channel & chan = schema::channel[code];
|
||||
ui->eventsCombo->addItem(chan.enabled() ? *icon_on : * icon_off, chan.label(), code);
|
||||
ui->eventsCombo->setItemData(i, chan.fullname(), Qt::ToolTipRole);
|
||||
for (int i=0; i < available.size(); ++i) {
|
||||
ChannelID code = available.at(i);
|
||||
schema::Channel & chan = schema::channel[code];
|
||||
ui->eventsCombo->addItem(chan.enabled() ? *icon_on : * icon_off, chan.label(), code);
|
||||
ui->eventsCombo->setItemData(i, chan.fullname(), Qt::ToolTipRole);
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
if (cpap) {
|
||||
//QHash<schema::ChanType, QList<schema::Channel *> > list;
|
||||
|
||||
|
||||
float hours=cpap->hours();
|
||||
float hours=day->hours(MT_CPAP);
|
||||
if (GraphView->isEmpty() && (hours>0)) {
|
||||
if (!p_profile->hasChannel(CPAP_Obstructive) && !p_profile->hasChannel(CPAP_Hypopnea)) {
|
||||
GraphView->setEmptyText(tr("No Graphs :("));
|
||||
@ -1499,12 +1506,12 @@ void Daily::Load(QDate date)
|
||||
}
|
||||
}
|
||||
|
||||
mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode);
|
||||
mode=(CPAPMode)(int)day->settings_max(CPAP_Mode);
|
||||
|
||||
modestr=schema::channel[CPAP_Mode].m_options[mode];
|
||||
|
||||
EventDataType ahi=(cpap->count(CPAP_Obstructive)+cpap->count(CPAP_Hypopnea)+cpap->count(CPAP_ClearAirway)+cpap->count(CPAP_Apnea));
|
||||
if (p_profile->general->calculateRDI()) ahi+=cpap->count(CPAP_RERA);
|
||||
EventDataType ahi=(day->count(CPAP_Obstructive)+day->count(CPAP_Hypopnea)+day->count(CPAP_ClearAirway)+day->count(CPAP_Apnea));
|
||||
if (p_profile->general->calculateRDI()) ahi+=day->count(CPAP_RERA);
|
||||
ahi/=hours;
|
||||
|
||||
if (!isBrick && hours>0) {
|
||||
@ -1520,8 +1527,8 @@ void Daily::Load(QDate date)
|
||||
.arg("#F88017").arg(COLOR_Text.name()).arg(ahiname).arg(schema::channel[ahichan].fullname()).arg(ahi,0,'f',2);
|
||||
html+="</tr>\n";
|
||||
html+="</table>\n";
|
||||
html+=getCPAPInformation(cpap);
|
||||
html+=getSleepTime(cpap,oxi);
|
||||
html+=getCPAPInformation(day);
|
||||
html+=getSleepTime(day);
|
||||
|
||||
html+="<table cellspacing=0 cellpadding=0 border=0 width='100%'>\n";
|
||||
|
||||
@ -1531,7 +1538,7 @@ void Daily::Load(QDate date)
|
||||
if (p_profile->general->showUnknownFlags()) zchans |= schema::UNKNOWN;
|
||||
|
||||
if (show_minors) zchans |= schema::MINOR_FLAG;
|
||||
QList<ChannelID> available = cpap->getSortedMachineChannels(zchans);
|
||||
QList<ChannelID> available = day->getSortedMachineChannels(zchans);
|
||||
|
||||
EventDataType val;
|
||||
QHash<ChannelID, EventDataType> values;
|
||||
@ -1541,10 +1548,10 @@ void Daily::Load(QDate date)
|
||||
if (!chan.enabled()) continue;
|
||||
QString data;
|
||||
if (chan.type() == schema::SPAN) {
|
||||
val = (100.0 / hours)*(cpap->sum(code)/3600.0);
|
||||
val = (100.0 / hours)*(day->sum(code)/3600.0);
|
||||
data = QString("%1%").arg(val,0,'f',2);
|
||||
} else {
|
||||
val = cpap->count(code) / hours;
|
||||
val = day->count(code) / hours;
|
||||
data = QString("%1").arg(val,0,'f',2);
|
||||
}
|
||||
values[code] = val;
|
||||
@ -1588,13 +1595,13 @@ void Daily::Load(QDate date)
|
||||
} else {
|
||||
html += "<tr><td align=center>Unable to display Pie Chart on this system</td></tr>\n";
|
||||
}
|
||||
} else if (cpap->channelHasData(CPAP_Obstructive)
|
||||
|| cpap->channelHasData(CPAP_Hypopnea)
|
||||
|| cpap->channelHasData(CPAP_ClearAirway)
|
||||
|| cpap->channelHasData(CPAP_RERA)
|
||||
|| cpap->channelHasData(CPAP_Apnea)
|
||||
|| cpap->channelHasData(CPAP_FlowLimit)
|
||||
|| cpap->channelHasData(CPAP_SensAwake)
|
||||
} else if (day->channelHasData(CPAP_Obstructive)
|
||||
|| day->channelHasData(CPAP_Hypopnea)
|
||||
|| day->channelHasData(CPAP_ClearAirway)
|
||||
|| day->channelHasData(CPAP_RERA)
|
||||
|| day->channelHasData(CPAP_Apnea)
|
||||
|| day->channelHasData(CPAP_FlowLimit)
|
||||
|| day->channelHasData(CPAP_SensAwake)
|
||||
) {
|
||||
html += "<tr><td align=center><img src=\"qrc:/docs/0.0.gif\"></td></tr>\n";
|
||||
}
|
||||
@ -1606,7 +1613,7 @@ void Daily::Load(QDate date)
|
||||
html+="<table cellspacing=0 cellpadding=0 border=0 width='100%'>\n";
|
||||
if (!isBrick) {
|
||||
html+="<tr><td colspan='5'> </td></tr>\n";
|
||||
if (cpap->size()>0) {
|
||||
if (day->size()>0) {
|
||||
html+="<tr><td colspan='5' align='center'><b><h2>"+tr("Sessions all off!")+"</h2></b></td></tr>";
|
||||
html+="<tr><td colspan='5' align='center'><i>"+tr("Sessions exist for this day but are switched off.")+"</i></td></tr>\n";
|
||||
} else {
|
||||
@ -1624,14 +1631,14 @@ void Daily::Load(QDate date)
|
||||
html+="<hr/>\n";
|
||||
|
||||
} // if (!CPAP)
|
||||
else html+=getSleepTime(cpap,oxi);
|
||||
else html+=getSleepTime(day);
|
||||
|
||||
if ((cpap && !isBrick && (cpap->hours()>0)) || oxi || posit) {
|
||||
if ((cpap && !isBrick && (day->hours()>0)) || oxi || posit) {
|
||||
|
||||
html+=getStatisticsInfo(cpap,oxi,posit);
|
||||
html+=getStatisticsInfo(day);
|
||||
|
||||
} else {
|
||||
if (cpap && cpap->hours()==0) {
|
||||
if (cpap && day->hours(MT_CPAP)<0.0000001) {
|
||||
} else {
|
||||
html+="<table cellspacing=0 cellpadding=0 border=0 width='100%'>\n";
|
||||
html+="<tr><td colspan=5 align=center><i>"+tr("No data available")+"</i></td></tr>\n";
|
||||
@ -1641,10 +1648,11 @@ void Daily::Load(QDate date)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
html+=getOximeterInformation(oxi);
|
||||
html+=getMachineSettings(cpap);
|
||||
html+=getSessionInformation(cpap,oxi,stage,posit);
|
||||
if (day) {
|
||||
html+=getOximeterInformation(day);
|
||||
html+=getMachineSettings(day);
|
||||
html+=getSessionInformation(day);
|
||||
}
|
||||
|
||||
html+="</body></html>";
|
||||
|
||||
@ -1662,9 +1670,10 @@ void Daily::Load(QDate date)
|
||||
connect(sessbar, SIGNAL(sessionClicked(Session*)), this, SLOT(doToggleSession(Session*)));
|
||||
int c=0;
|
||||
|
||||
for (i=cpap->begin();i!=cpap->end();++i) {
|
||||
for (i=day->begin();i!=day->end();++i) {
|
||||
Session * s=*i;
|
||||
sessbar->add(s, cols[c % maxcolors]);
|
||||
if ((*s).machine()->type() == MT_CPAP)
|
||||
sessbar->add(s, cols[c % maxcolors]);
|
||||
c++;
|
||||
}
|
||||
} else sessbar=nullptr;
|
||||
@ -1932,7 +1941,7 @@ void Daily::on_JournalNotesColour_clicked()
|
||||
}
|
||||
Session * Daily::CreateJournalSession(QDate date)
|
||||
{
|
||||
Machine *m=p_profile->GetMachine(MT_JOURNAL);
|
||||
Machine *m = p_profile->GetMachine(MT_JOURNAL);
|
||||
if (!m) {
|
||||
m=new Machine(0);
|
||||
MachineInfo info;
|
||||
@ -1941,10 +1950,13 @@ Session * Daily::CreateJournalSession(QDate date)
|
||||
info.brand = "Journal";
|
||||
info.type = MT_JOURNAL;
|
||||
m->setInfo(info);
|
||||
m->setType(MT_JOURNAL);
|
||||
p_profile->AddMachine(m);
|
||||
}
|
||||
|
||||
Session *sess=new Session(m,0);
|
||||
qint64 st,et;
|
||||
|
||||
Day *cday=p_profile->GetDay(date,MT_CPAP);
|
||||
if (cday) {
|
||||
st=cday->first();
|
||||
@ -1962,14 +1974,13 @@ Session * Daily::CreateJournalSession(QDate date)
|
||||
}
|
||||
Session * Daily::GetJournalSession(QDate date) // Get the first journal session
|
||||
{
|
||||
Day *journal=p_profile->GetDay(date,MT_JOURNAL);
|
||||
if (!journal)
|
||||
return nullptr; //CreateJournalSession(date);
|
||||
QList<Session *>::iterator s;
|
||||
s=journal->begin();
|
||||
if (s!=journal->end())
|
||||
return *s;
|
||||
return nullptr;
|
||||
Day *day=p_profile->addDay(date);
|
||||
|
||||
Session * session = day->firstSession(MT_JOURNAL);
|
||||
if (!session) {
|
||||
session = CreateJournalSession(date);
|
||||
}
|
||||
return session;
|
||||
}
|
||||
|
||||
void Daily::UpdateCPAPGraphs(Day *day)
|
||||
|
@ -314,13 +314,13 @@ private:
|
||||
void updateGraphCombo();
|
||||
|
||||
|
||||
QString getSessionInformation(Day *cpap, Day *oxi, Day *stage, Day *posit);
|
||||
QString getMachineSettings(Day *cpap);
|
||||
QString getStatisticsInfo(Day *cpap, Day *oxi, Day *pos);
|
||||
QString getCPAPInformation(Day *cpap);
|
||||
QString getOximeterInformation(Day *oxi);
|
||||
QString getEventBreakdown(Day *cpap);
|
||||
QString getSleepTime(Day *cpap, Day *oxi);
|
||||
QString getSessionInformation(Day *);
|
||||
QString getMachineSettings(Day *);
|
||||
QString getStatisticsInfo(Day *);
|
||||
QString getCPAPInformation(Day *);
|
||||
QString getOximeterInformation(Day *);
|
||||
QString getEventBreakdown(Day *);
|
||||
QString getSleepTime(Day *);
|
||||
|
||||
QHash<QString, gGraph *> graphlist;
|
||||
|
||||
|
@ -1180,7 +1180,7 @@ void MainWindow::updateFavourites()
|
||||
|
||||
if (journal) {
|
||||
if (journal->size() > 0) {
|
||||
Session *sess = (*journal)[0];
|
||||
Session *sess = journal->firstSession(MT_JOURNAL);
|
||||
QString tmp;
|
||||
bool filtered = !bookmarkFilter.isEmpty();
|
||||
bool found = !filtered;
|
||||
@ -1939,11 +1939,10 @@ void MainWindow::on_actionPurge_Current_Day_triggered()
|
||||
QDate date = getDaily()->getDate();
|
||||
getDaily()->Unload(date);
|
||||
Day *day = p_profile->GetDay(date, MT_CPAP);
|
||||
Machine *m;
|
||||
|
||||
if (day) {
|
||||
m = day->machine;
|
||||
Machine *cpap = nullptr;
|
||||
if (day) cpap = day->machine(MT_CPAP);
|
||||
|
||||
if (cpap) {
|
||||
QList<Session *>::iterator s;
|
||||
|
||||
QList<Session *> list;
|
||||
@ -1956,13 +1955,13 @@ void MainWindow::on_actionPurge_Current_Day_triggered()
|
||||
QHash<QString, SessionID> skipfiles;
|
||||
// Read the already imported file list
|
||||
|
||||
QFile impfile(m->getDataPath()+"/imported_files.csv");
|
||||
QFile impfile(cpap->getDataPath()+"/imported_files.csv");
|
||||
if (impfile.exists()) {
|
||||
if (impfile.open(QFile::ReadOnly)) {
|
||||
QTextStream impstream(&impfile);
|
||||
QString serial;
|
||||
impstream >> serial;
|
||||
if (m->serial() == serial) {
|
||||
if (cpap->serial() == serial) {
|
||||
QString line, file, str;
|
||||
SessionID sid;
|
||||
bool ok;
|
||||
@ -1984,7 +1983,7 @@ void MainWindow::on_actionPurge_Current_Day_triggered()
|
||||
// Rewrite the file without the sessions being removed.
|
||||
if (impfile.open(QFile::WriteOnly)) {
|
||||
QTextStream out(&impfile);
|
||||
out << m->serial();
|
||||
out << cpap->serial();
|
||||
QHash<QString, SessionID>::iterator skit;
|
||||
QHash<QString, SessionID>::iterator skit_end = skipfiles.end();
|
||||
for (skit = skipfiles.begin(); skit != skit_end; ++skit) {
|
||||
|
@ -433,7 +433,7 @@ void OximeterImport::on_liveImportButton_clicked()
|
||||
oximodule->Open("live");
|
||||
ui->stopButton->setVisible(true);
|
||||
|
||||
dummyday = new Day(mach);
|
||||
dummyday = new Day();
|
||||
|
||||
quint32 starttime = oximodule->startTime().toTime_t();
|
||||
ti = qint64(starttime) * 1000L;
|
||||
@ -445,7 +445,7 @@ void OximeterImport::on_liveImportButton_clicked()
|
||||
|
||||
ELplethy->setFirst(start_ti);
|
||||
session->really_set_first(start_ti);
|
||||
dummyday->AddSession(session);
|
||||
dummyday->addSession(session);
|
||||
|
||||
plethyChart->setMinX(start_ti);
|
||||
plethyGraph->SetMinX(start_ti);
|
||||
|
@ -158,19 +158,21 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
|
||||
if (bounds.height() > maxy) { maxy = bounds.height(); }
|
||||
}
|
||||
|
||||
Day *cpap = nullptr, *oxi = nullptr;
|
||||
Machine *cpap = nullptr, *oxi = nullptr;
|
||||
|
||||
int graph_slots = 0;
|
||||
Day * day = p_profile->GetGoodDay(mainwin->getDaily()->getDate(), MT_CPAP);
|
||||
if (day) cpap = day->machine(MT_CPAP);
|
||||
|
||||
if (name == STR_TR_Daily) {
|
||||
cpap = p_profile->GetGoodDay(date, MT_CPAP);
|
||||
oxi = p_profile->GetGoodDay(date, MT_OXIMETER);
|
||||
|
||||
QString cpapinfo = date.toString(Qt::SystemLocaleLongDate) + "\n\n";
|
||||
|
||||
|
||||
if (cpap) {
|
||||
time_t f = cpap->first() / 1000L;
|
||||
time_t l = cpap->last() / 1000L;
|
||||
int tt = qint64(cpap->total_time()) / 1000L;
|
||||
time_t f = day->first(MT_CPAP) / 1000L;
|
||||
time_t l = day->last(MT_CPAP) / 1000L;
|
||||
int tt = qint64(day->total_time(MT_CPAP)) / 1000L;
|
||||
int h = tt / 3600;
|
||||
int m = (tt / 60) % 60;
|
||||
int s = tt % 60;
|
||||
@ -186,29 +188,29 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
|
||||
// submodel = "\n" + cpap->machine->info.modeproperties[STR_PROP_SubModel];
|
||||
// }
|
||||
|
||||
cpapinfo += cpap->machine->brand() + " " +
|
||||
cpap->machine->model() + submodel;
|
||||
CPAPMode mode = (CPAPMode)(int)cpap->settings_max(CPAP_Mode);
|
||||
cpapinfo += cpap->brand() + " " +
|
||||
cpap->model() + submodel;
|
||||
CPAPMode mode = (CPAPMode)(int)day->settings_max(CPAP_Mode);
|
||||
cpapinfo += "\n" + STR_TR_Mode + ": ";
|
||||
|
||||
if (mode == MODE_CPAP) {
|
||||
EventDataType min = round(cpap->settings_wavg(CPAP_Pressure) * 2) / 2.0;
|
||||
EventDataType min = round(day->settings_wavg(CPAP_Pressure) * 2) / 2.0;
|
||||
cpapinfo += STR_TR_CPAP + " " + QString::number(min) + STR_UNIT_CMH2O;
|
||||
} else if (mode == MODE_APAP) {
|
||||
EventDataType min = cpap->settings_min(CPAP_PressureMin);
|
||||
EventDataType max = cpap->settings_max(CPAP_PressureMax);
|
||||
EventDataType min = day->settings_min(CPAP_PressureMin);
|
||||
EventDataType max = day->settings_max(CPAP_PressureMax);
|
||||
cpapinfo += STR_TR_APAP + " " + QString::number(min) + "-" + QString::number(max) + STR_UNIT_CMH2O;
|
||||
} else if (mode == MODE_BILEVEL_FIXED) {
|
||||
EventDataType epap = cpap->settings_min(CPAP_EPAP);
|
||||
EventDataType ipap = cpap->settings_max(CPAP_IPAP);
|
||||
EventDataType ps = cpap->settings_max(CPAP_PS);
|
||||
EventDataType epap = day->settings_min(CPAP_EPAP);
|
||||
EventDataType ipap = day->settings_max(CPAP_IPAP);
|
||||
EventDataType ps = day->settings_max(CPAP_PS);
|
||||
cpapinfo += STR_TR_BiLevel +
|
||||
QString("\n" + STR_TR_EPAP + ": %1 " + STR_TR_IPAP + ": %2 %3\n" + STR_TR_PS + ": %4")
|
||||
.arg(epap, 0, 'f', 1).arg(ipap, 0, 'f', 1).arg(STR_UNIT_CMH2O).arg(ps, 0, 'f', 1);
|
||||
} else if (mode == MODE_BILEVEL_AUTO_FIXED_PS) {
|
||||
EventDataType epap = cpap->settings_min(CPAP_EPAP);
|
||||
EventDataType ipap = cpap->settings_max(CPAP_IPAP);
|
||||
EventDataType ps = cpap->settings_max(CPAP_PS);
|
||||
EventDataType epap = day->settings_min(CPAP_EPAP);
|
||||
EventDataType ipap = day->settings_max(CPAP_IPAP);
|
||||
EventDataType ps = day->settings_max(CPAP_PS);
|
||||
cpapinfo += STR_TR_BiLevel +
|
||||
QString("\n" + QObject::tr("Range")+ ": %1-%2 %3 " + QObject::tr("Fixed %1").arg(STR_TR_PS) + ": %4")
|
||||
.arg(epap, 0, 'f', 1).arg(ipap, 0, 'f', 1).arg(STR_UNIT_CMH2O).arg(ps, 0, 'f', 1);
|
||||
@ -221,11 +223,11 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
|
||||
QString("\n" + QObject::tr("Fixed %1").arg(STR_TR_EPAP) + ": %1 %3" + QObject::tr("Max %1").arg(STR_TR_IPAP) + ": %2 %3\n" + QObject::tr("Variable %1").arg(STR_TR_PS) + ": %4-%5")
|
||||
.arg(epap, 0, 'f', 1).arg(ipap, 0, 'f', 1).arg(STR_UNIT_CMH2O).arg(psl,0,'f',1).arg(psh,0,'f',1);
|
||||
} */else if (mode == MODE_ASV) {
|
||||
EventDataType epap = cpap->settings_min(CPAP_EPAP);
|
||||
EventDataType low = cpap->settings_min(CPAP_IPAPLo);
|
||||
EventDataType high = cpap->settings_max(CPAP_IPAPHi);
|
||||
EventDataType psl = cpap->settings_min(CPAP_PSMin);
|
||||
EventDataType psh = cpap->settings_max(CPAP_PSMax);
|
||||
EventDataType epap = day->settings_min(CPAP_EPAP);
|
||||
EventDataType low = day->settings_min(CPAP_IPAPLo);
|
||||
EventDataType high = day->settings_max(CPAP_IPAPHi);
|
||||
EventDataType psl = day->settings_min(CPAP_PSMin);
|
||||
EventDataType psh = day->settings_max(CPAP_PSMax);
|
||||
cpapinfo += STR_TR_ASV + QString("\n" + STR_TR_EPAP + ": %1 " + STR_TR_IPAP + ": %2 - %3 %4\n" +
|
||||
STR_TR_PS + ": %5 / %6")
|
||||
.arg(epap, 0, 'f', 1)
|
||||
@ -236,24 +238,25 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
|
||||
.arg(psh, 0, 'f', 1);
|
||||
} else { cpapinfo += STR_TR_Unknown; }
|
||||
|
||||
float ahi = (cpap->count(CPAP_Obstructive) + cpap->count(CPAP_Hypopnea) + cpap->count(
|
||||
CPAP_ClearAirway) + cpap->count(CPAP_Apnea));
|
||||
float ahi = (day->count(CPAP_Obstructive) + day->count(CPAP_Hypopnea) +
|
||||
day->count(CPAP_ClearAirway) + day->count(CPAP_Apnea));
|
||||
|
||||
if (p_profile->general->calculateRDI()) { ahi += cpap->count(CPAP_RERA); }
|
||||
if (p_profile->general->calculateRDI()) { ahi += day->count(CPAP_RERA); }
|
||||
|
||||
ahi /= cpap->hours();
|
||||
float csr = (100.0 / cpap->hours()) * (cpap->sum(CPAP_CSR) / 3600.0);
|
||||
float uai = cpap->count(CPAP_Apnea) / cpap->hours();
|
||||
float oai = cpap->count(CPAP_Obstructive) / cpap->hours();
|
||||
float hi = (cpap->count(CPAP_ExP) + cpap->count(CPAP_Hypopnea)) / cpap->hours();
|
||||
float cai = cpap->count(CPAP_ClearAirway) / cpap->hours();
|
||||
float rei = cpap->count(CPAP_RERA) / cpap->hours();
|
||||
float vsi = cpap->count(CPAP_VSnore) / cpap->hours();
|
||||
float fli = cpap->count(CPAP_FlowLimit) / cpap->hours();
|
||||
// float sai = cpap->count(CPAP_SensAwake) / cpap->hours();
|
||||
float nri = cpap->count(CPAP_NRI) / cpap->hours();
|
||||
float lki = cpap->count(CPAP_LeakFlag) / cpap->hours();
|
||||
float exp = cpap->count(CPAP_ExP) / cpap->hours();
|
||||
float hours = day->hours(MT_CPAP);
|
||||
ahi /= hours;
|
||||
float csr = (100.0 / hours) * (day->sum(CPAP_CSR) / 3600.0);
|
||||
float uai = day->count(CPAP_Apnea) / hours;
|
||||
float oai = day->count(CPAP_Obstructive) / hours;
|
||||
float hi = (day->count(CPAP_ExP) + day->count(CPAP_Hypopnea)) / hours;
|
||||
float cai = day->count(CPAP_ClearAirway) / hours;
|
||||
float rei = day->count(CPAP_RERA) / hours;
|
||||
float vsi = day->count(CPAP_VSnore) / hours;
|
||||
float fli = day->count(CPAP_FlowLimit) / hours;
|
||||
// float sai = day->count(CPAP_SensAwake) / hours;
|
||||
float nri = day->count(CPAP_NRI) / hours;
|
||||
float lki = day->count(CPAP_LeakFlag) / hours;
|
||||
float exp = day->count(CPAP_ExP) / hours;
|
||||
|
||||
int piesize = (2048.0 / 8.0) * 1.3; // 1.5" in size
|
||||
//float fscale=font_scale;
|
||||
@ -303,13 +306,13 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
|
||||
stats = QObject::tr("AI=%1 HI=%2 CAI=%3 ").arg(oai, 0, 'f', 2).arg(hi, 0, 'f', 2).arg(cai, 0, 'f',
|
||||
2);
|
||||
|
||||
if (cpap->machine->loaderName() == STR_MACH_PRS1) {
|
||||
if (cpap->loaderName() == STR_MACH_PRS1) {
|
||||
stats += QObject::tr("REI=%1 VSI=%2 FLI=%3 PB/CSR=%4%%")
|
||||
.arg(rei, 0, 'f', 2).arg(vsi, 0, 'f', 2)
|
||||
.arg(fli, 0, 'f', 2).arg(csr, 0, 'f', 2);
|
||||
} else if (cpap->machine->loaderName() == STR_MACH_ResMed) {
|
||||
} else if (cpap->loaderName() == STR_MACH_ResMed) {
|
||||
stats += QObject::tr("UAI=%1 ").arg(uai, 0, 'f', 2);
|
||||
} else if (cpap->machine->loaderName() == STR_MACH_Intellipap) {
|
||||
} else if (cpap->loaderName() == STR_MACH_Intellipap) {
|
||||
stats += QObject::tr("NRI=%1 LKI=%2 EPI=%3").arg(nri, 0, 'f', 2).arg(lki, 0, 'f', 2).arg(exp, 0,
|
||||
'f', 2);
|
||||
}
|
||||
@ -418,11 +421,11 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
|
||||
if (!g->visible()) { continue; }
|
||||
|
||||
if (cpap) {
|
||||
st = cpap->first();
|
||||
et = cpap->last();
|
||||
st = day->first(MT_CPAP);
|
||||
et = day->last(MT_CPAP);
|
||||
} else if (oxi) {
|
||||
st = oxi->first();
|
||||
et = oxi->last();
|
||||
st = day->first(MT_OXIMETER);
|
||||
et = day->last(MT_OXIMETER);
|
||||
}
|
||||
|
||||
if (g->name() == schema::channel[CPAP_FlowRate].code()) {
|
||||
@ -459,22 +462,22 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
|
||||
|
||||
if (cpap && flow && !flow->isEmpty() && flow->visible()) {
|
||||
labels.push_back(EntireDay);
|
||||
start.push_back(cpap->first());
|
||||
end.push_back(cpap->last());
|
||||
start.push_back(day->first(MT_CPAP));
|
||||
end.push_back(day->last(MT_CPAP));
|
||||
graphs.push_back(flow);
|
||||
}
|
||||
|
||||
if (oxi && spo2 && !spo2->isEmpty() && spo2->visible()) {
|
||||
labels.push_back(EntireDay);
|
||||
start.push_back(oxi->first());
|
||||
end.push_back(oxi->last());
|
||||
start.push_back(day->first(MT_OXIMETER));
|
||||
end.push_back(day->last(MT_OXIMETER));
|
||||
graphs.push_back(spo2);
|
||||
}
|
||||
|
||||
if (oxi && pulse && !pulse->isEmpty() && pulse->visible()) {
|
||||
labels.push_back(EntireDay);
|
||||
start.push_back(oxi->first());
|
||||
end.push_back(oxi->last());
|
||||
start.push_back(day->first(MT_OXIMETER));
|
||||
end.push_back(day->last(MT_OXIMETER));
|
||||
graphs.push_back(pulse);
|
||||
}
|
||||
|
||||
|
@ -756,12 +756,12 @@ QString Statistics::GenerateHTML()
|
||||
|
||||
CPAPLoader * loader = nullptr;
|
||||
|
||||
if (day) loader = dynamic_cast<CPAPLoader *>(day->machine->loader());
|
||||
if (day) loader = dynamic_cast<CPAPLoader *>(day->machine(MT_CPAP)->loader());
|
||||
|
||||
if (day && loader) {
|
||||
lastchanged = false;
|
||||
|
||||
hours = day->hours();
|
||||
hours = day->hours(MT_CPAP);
|
||||
|
||||
if (hours > p_profile->cpap->complianceHours()) {
|
||||
compliant++;
|
||||
@ -781,7 +781,7 @@ QString Statistics::GenerateHTML()
|
||||
if (mode ==0) {
|
||||
mode = (CPAPMode)(int)round(day->settings_wavg(CPAP_Mode));
|
||||
}
|
||||
mach = day->machine;
|
||||
mach = day->machine(MT_CPAP);
|
||||
|
||||
min = max = ps = pshi = maxipap = 0;
|
||||
|
||||
|
@ -122,20 +122,23 @@ QString GenerateWelcomeHTML()
|
||||
html += "<p>" + QObject::tr("It might be a good idea to check preferences first,</br>as there are some options that affect import.")+"</p>"
|
||||
"<p>" + QObject::tr("First import can take a few minutes.") + "</p>";
|
||||
} else {
|
||||
if (havecpapdata) {
|
||||
QDate date = p_profile->LastDay(MT_CPAP);
|
||||
Day *day = p_profile->GetDay(date, MT_CPAP);
|
||||
|
||||
if (havecpapdata && day) {
|
||||
QString cpapimage = "";
|
||||
QDate date = p_profile->LastDay(MT_CPAP);
|
||||
Day *day = p_profile->GetDay(date, MT_CPAP);
|
||||
if (day) {
|
||||
if (day->machine->loaderName() == STR_MACH_ResMed) cpapimage = "qrc:/icons/rms9.png";
|
||||
else if (day->machine->loaderName() == STR_MACH_PRS1) cpapimage = "qrc:/icons/prs1.png";
|
||||
else if (day->machine->loaderName() == STR_MACH_Intellipap) cpapimage = "qrc:/icons/intellipap.png";
|
||||
|
||||
Machine * cpap = day->machine(MT_CPAP);
|
||||
if (cpap) {
|
||||
if (cpap->loaderName() == STR_MACH_ResMed) cpapimage = "qrc:/icons/rms9.png";
|
||||
else if (cpap->loaderName() == STR_MACH_PRS1) cpapimage = "qrc:/icons/prs1.png";
|
||||
else if (cpap->loaderName() == STR_MACH_Intellipap) cpapimage = "qrc:/icons/intellipap.png";
|
||||
}
|
||||
html += "<table cellpadding=4><tr><td><img src='"+cpapimage+"' width=160px><br/>";
|
||||
|
||||
html+="</td><td align=center><table cellpadding=4 class=curved2 title=\""+QObject::tr("Click this box to see this in daily view.")+"\"><tr>"+
|
||||
QString("<td align=center onmouseover='ChangeColor(this, \"#efefa0\");' onmouseout='ChangeColor(this, \"#ffffc0\");' onclick='alert(\"daily=%1\");'>").arg(date.toString(Qt::ISODate))+"<b>"+
|
||||
QObject::tr("The last time you used your %1...").arg(day->machine->brand()+" "+day->machine->model())+"</b><br/>";
|
||||
QObject::tr("The last time you used your %1...").arg(cpap->brand()+" "+cpap->model())+"</b><br/>";
|
||||
|
||||
int daysto = date.daysTo(QDate::currentDate());
|
||||
QString daystring;
|
||||
|
Loading…
Reference in New Issue
Block a user