mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 02:30:44 +00:00
Smarten up Statistics page a bit, and make it more flexible
This is in preperation for making it editable by users. Also added compliance calculations to profile
This commit is contained in:
parent
95fd5b9a7f
commit
6ea8de1515
@ -629,6 +629,40 @@ int Profile::countDays(MachineType mt, QDate start, QDate end)
|
||||
|
||||
}
|
||||
|
||||
int Profile::countCompliantDays(MachineType mt, QDate start, QDate end)
|
||||
{
|
||||
EventDataType compliance = cpap->complianceHours();
|
||||
|
||||
if (!start.isValid()) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (!end.isValid()) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
QDate date = start;
|
||||
|
||||
if (date.isNull()) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
int days = 0;
|
||||
|
||||
do {
|
||||
Day *day = GetGoodDay(date, mt);
|
||||
|
||||
if (day) {
|
||||
if ((day->machine->GetType() == mt) && (day->hours() > compliance)) { days++; }
|
||||
}
|
||||
|
||||
date = date.addDays(1);
|
||||
} while (date <= end);
|
||||
|
||||
return days;
|
||||
}
|
||||
|
||||
|
||||
EventDataType Profile::calcCount(ChannelID code, MachineType mt, QDate start, QDate end)
|
||||
{
|
||||
if (!start.isValid()) {
|
||||
|
@ -98,6 +98,9 @@ class Profile : public Preferences
|
||||
bool contains(QString key) { return p_preferences.contains(key); }
|
||||
|
||||
int countDays(MachineType mt = MT_UNKNOWN, QDate start = QDate(), QDate end = QDate());
|
||||
|
||||
int countCompliantDays(MachineType mt, QDate start, QDate end);
|
||||
|
||||
EventDataType calcCount(ChannelID code, MachineType mt = MT_CPAP, QDate start = QDate(),
|
||||
QDate end = QDate());
|
||||
double calcSum(ChannelID code, MachineType mt = MT_CPAP, QDate start = QDate(),
|
||||
|
@ -1932,7 +1932,8 @@ void MainWindow::on_actionImport_Somnopose_Data_triggered()
|
||||
|
||||
void MainWindow::GenerateStatistics()
|
||||
{
|
||||
QString html = Statistics::GenerateHTML();
|
||||
Statistics stats;
|
||||
QString html = stats.GenerateHTML();
|
||||
|
||||
updateFavourites();
|
||||
|
||||
|
@ -212,7 +212,8 @@ OTHER_FILES += \
|
||||
../update.xml \
|
||||
docs/changelog.txt \
|
||||
docs/update_notes.html \
|
||||
docs/intro.html
|
||||
docs/intro.html \
|
||||
docs/statistics.xml
|
||||
|
||||
win32 {
|
||||
CONFIG(debug, debug|release) {
|
||||
|
@ -14,15 +14,93 @@
|
||||
|
||||
#include "mainwindow.h"
|
||||
#include "statistics.h"
|
||||
#include "SleepLib/schema.h"
|
||||
|
||||
|
||||
extern MainWindow *mainwin;
|
||||
|
||||
QString formatTime(float time)
|
||||
{
|
||||
int hours = time;
|
||||
int seconds = time * 3600.0;
|
||||
int minutes = (seconds / 60) % 60;
|
||||
seconds %= 60;
|
||||
return QString().sprintf("%02i:%02i", hours, minutes); //,seconds);
|
||||
}
|
||||
|
||||
|
||||
Statistics::Statistics(QObject *parent) :
|
||||
QObject(parent)
|
||||
{
|
||||
rows = {
|
||||
{ tr("CPAP Statistics"), SC_HEADING, MT_CPAP },
|
||||
{ "", SC_DAYS, MT_CPAP },
|
||||
{ "", SC_COLUMNHEADERS, MT_CPAP },
|
||||
{ tr("CPAP Usage"), SC_SUBHEADING, MT_CPAP },
|
||||
{ tr("Average Hours per Night"), SC_HOURS, MT_CPAP },
|
||||
{ tr("Compliancy"), SC_COMPLIANCE, MT_CPAP },
|
||||
|
||||
{ tr("Therapy Efficiacy"), SC_SUBHEADING, MT_CPAP },
|
||||
{ "AHI", SC_AHI, MT_CPAP },
|
||||
{ "Obstructive", SC_CPH, MT_CPAP },
|
||||
{ "Hypopnea", SC_CPH, MT_CPAP },
|
||||
{ "ClearAirway", SC_CPH, MT_CPAP },
|
||||
{ "FlowLimit", SC_CPH, MT_CPAP },
|
||||
{ "RERA", SC_CPH, MT_CPAP },
|
||||
|
||||
{ tr("Leak Statistics"), SC_SUBHEADING, MT_CPAP },
|
||||
{ "Leak", SC_WAVG, MT_CPAP },
|
||||
{ "Leak", SC_90P, MT_CPAP },
|
||||
|
||||
{ tr("Pressure Statistics"), SC_SUBHEADING, MT_CPAP },
|
||||
{ "Pressure", SC_WAVG, MT_CPAP },
|
||||
{ "Pressure", SC_MIN, MT_CPAP },
|
||||
{ "Pressure", SC_MAX, MT_CPAP },
|
||||
{ "Pressure", SC_90P, MT_CPAP },
|
||||
{ "EPAP", SC_WAVG, MT_CPAP },
|
||||
{ "EPAP", SC_MIN, MT_CPAP },
|
||||
{ "EPAP", SC_MAX, MT_CPAP },
|
||||
{ "IPAP", SC_WAVG, MT_CPAP },
|
||||
{ "IPAP", SC_90P, MT_CPAP },
|
||||
{ "IPAP", SC_MIN, MT_CPAP },
|
||||
{ "IPAP", SC_MAX, MT_CPAP },
|
||||
|
||||
{ tr("Oximeter Statistics"), SC_HEADING, MT_OXIMETER },
|
||||
{ "", SC_DAYS, MT_OXIMETER },
|
||||
{ "", SC_COLUMNHEADERS, MT_OXIMETER },
|
||||
|
||||
{ tr("Blood Oxygen Saturation"), SC_SUBHEADING, MT_CPAP },
|
||||
{ "SPO2", SC_WAVG, MT_OXIMETER },
|
||||
{ "SPO2", SC_MIN, MT_OXIMETER },
|
||||
{ "SPO2Drop", SC_CPH, MT_OXIMETER },
|
||||
{ "SPO2Drop", SC_SPH, MT_OXIMETER },
|
||||
{ tr("Pulse Rate"), SC_SUBHEADING, MT_CPAP },
|
||||
{ "Pulse", SC_WAVG, MT_OXIMETER },
|
||||
{ "Pulse", SC_MIN, MT_OXIMETER },
|
||||
{ "Pulse", SC_MAX, MT_OXIMETER },
|
||||
{ "PulseChange", SC_CPH, MT_OXIMETER },
|
||||
};
|
||||
|
||||
// These are for formatting the headers for the first column
|
||||
calcnames = {
|
||||
{ SC_UNDEFINED, "" },
|
||||
{ SC_MEDIAN, tr("%1 Median") },
|
||||
{ SC_AVG, tr("Average %1") },
|
||||
{ SC_WAVG, tr("Average %1") },
|
||||
{ SC_90P, tr("90% %1") }, // this gets converted to whatever the upper percentile is set to
|
||||
{ SC_MIN, tr("Min %1") },
|
||||
{ SC_MAX, tr("Max %1") },
|
||||
{ SC_CPH, tr("%1 Index") },
|
||||
{ SC_SPH, tr("% of night in %1") },
|
||||
};
|
||||
machinenames = {
|
||||
{ MT_UNKNOWN, STR_TR_Unknown },
|
||||
{ MT_CPAP, STR_TR_CPAP },
|
||||
{ MT_OXIMETER, STR_TR_Oximetry },
|
||||
{ MT_SLEEPSTAGE, STR_TR_SleepStage },
|
||||
// { MT_JOURNAL, STR_TR_Journal },
|
||||
// { MT_POSITION, STR_TR_Position },
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
|
||||
@ -48,17 +126,6 @@ QString htmlHeader()
|
||||
"</head>"
|
||||
"<body leftmargin=0 topmargin=0 rightmargin=0>"
|
||||
"<div align=center><table cellpadding=3 cellspacing=0 border=0 width=100%>"
|
||||
//"<tr>"
|
||||
//"<td bgcolor='red' width=100% colspan=3 align=center><font color='yellow'><b>"+QObject::tr("Warning:")+"</b> "+
|
||||
// #ifdef TEST_BUILD
|
||||
// QObject::tr("This is an testing build so expect the possibility things will go wrong.")+"<br/>"+
|
||||
// QObject::tr("Please report bugs you find here to SleepyHead's developer mailing list.")+
|
||||
// #else
|
||||
// QObject::tr("This is a beta software and some functionality may not work as intended yet.")+"<br/>"+
|
||||
// QObject::tr("Please report any bugs you find to SleepyHead's SourceForge page.")+
|
||||
// #endif
|
||||
//"</font></td>"
|
||||
//"</tr><tr>"
|
||||
"<td>" +
|
||||
QString(QObject::tr("Name: %1, %2")).arg(PROFILE.user->lastName()).arg(
|
||||
PROFILE.user->firstName()) + "<br/>" +
|
||||
@ -89,14 +156,6 @@ QString htmlFooter()
|
||||
"</body></html>";
|
||||
}
|
||||
|
||||
QString formatTime(float time)
|
||||
{
|
||||
int hours = time;
|
||||
int seconds = time * 3600.0;
|
||||
int minutes = (seconds / 60) % 60;
|
||||
seconds %= 60;
|
||||
return QString().sprintf("%02i:%02i", hours, minutes); //,seconds);
|
||||
}
|
||||
|
||||
|
||||
EventDataType calcAHI(QDate start, QDate end)
|
||||
@ -384,24 +443,27 @@ bool operator <(const UsageData &c1, const UsageData &c2)
|
||||
|
||||
QString Statistics::GenerateHTML()
|
||||
{
|
||||
|
||||
QString heading_color="#f8f0ff";
|
||||
QString subheading_color="#efefef";
|
||||
|
||||
QString html = htmlHeader();
|
||||
|
||||
// Find first and last days with valid CPAP data
|
||||
QDate lastcpap = p_profile->LastGoodDay(MT_CPAP);
|
||||
QDate firstcpap = p_profile->FirstGoodDay(MT_CPAP);
|
||||
QDate cpapweek = lastcpap.addDays(-7);
|
||||
QDate cpapmonth = lastcpap.addDays(-30);
|
||||
|
||||
|
||||
QDate cpapweek = lastcpap.addDays(-6);
|
||||
QDate cpapmonth = lastcpap.addDays(-29);
|
||||
QDate cpap6month = lastcpap.addMonths(-6);
|
||||
QDate cpapyear = lastcpap.addYears(-12);
|
||||
QDate cpapyear = lastcpap.addMonths(-12);
|
||||
|
||||
if (cpapweek < firstcpap) { cpapweek = firstcpap; }
|
||||
|
||||
if (cpapmonth < firstcpap) { cpapmonth = firstcpap; }
|
||||
|
||||
if (cpap6month < firstcpap) { cpap6month = firstcpap; }
|
||||
|
||||
if (cpapyear < firstcpap) { cpapyear = firstcpap; }
|
||||
|
||||
|
||||
QList<Machine *> cpap_machines = PROFILE.GetMachines(MT_CPAP);
|
||||
QList<Machine *> oximeters = PROFILE.GetMachines(MT_OXIMETER);
|
||||
QList<Machine *> mach;
|
||||
@ -420,10 +482,6 @@ QString Statistics::GenerateHTML()
|
||||
}
|
||||
|
||||
int cpapdays = PROFILE.countDays(MT_CPAP, firstcpap, lastcpap);
|
||||
int cpapweekdays = PROFILE.countDays(MT_CPAP, cpapweek, lastcpap);
|
||||
int cpapmonthdays = PROFILE.countDays(MT_CPAP, cpapmonth, lastcpap);
|
||||
int cpapyeardays = PROFILE.countDays(MT_CPAP, cpapyear, lastcpap);
|
||||
int cpap6monthdays = PROFILE.countDays(MT_CPAP, cpap6month, lastcpap);
|
||||
|
||||
CPAPMode cpapmode = (CPAPMode)(int)p_profile->calcSettingsMax(CPAP_Mode, MT_CPAP, firstcpap,
|
||||
lastcpap);
|
||||
@ -448,283 +506,95 @@ QString Statistics::GenerateHTML()
|
||||
html += "<div align=center>";
|
||||
html += QString("<table cellpadding=2 cellspacing=0 border=1 width=90%>");
|
||||
|
||||
if (cpapdays == 0) {
|
||||
html += "<tr><td colspan=6 align=center>" + tr("No CPAP Machine Data Imported") + "</td></tr>";
|
||||
} else {
|
||||
html += QString("<tr><td colspan=6 align=center><b>") + tr("CPAP Statistics as of") +
|
||||
QString(" %1</b></td></tr>").arg(lastcpap.toString(Qt::SystemLocaleLongDate));
|
||||
|
||||
if (cpap_machines.size() > 0) {
|
||||
// html+=QString("<tr><td colspan=6 align=center><b>%1</b></td></tr>").arg(tr("CPAP Statistics"));
|
||||
QDate last = lastcpap, first = lastcpap, week = lastcpap, month = lastcpap, sixmonth = lastcpap, year = lastcpap;
|
||||
|
||||
if (!cpapdays) {
|
||||
html += QString("<tr><td colspan=6 align=center><b>%1</b></td></tr>").arg(
|
||||
tr("No CPAP data available."));
|
||||
} else if (cpapdays == 1) {
|
||||
html += QString("<tr><td colspan=6 align=center>%1</td></tr>").arg(QString(
|
||||
tr("%1 day of CPAP Data, on %2.")).arg(cpapdays).arg(firstcpap.toString(
|
||||
Qt::SystemLocaleShortDate)));
|
||||
} else {
|
||||
html += QString("<tr><td colspan=6 align=center>%1</td></tr>").arg(QString(
|
||||
tr("%1 days of CPAP Data, between %2 and %3")).arg(cpapdays).arg(firstcpap.toString(
|
||||
Qt::SystemLocaleShortDate)).arg(lastcpap.toString(Qt::SystemLocaleShortDate)));
|
||||
bool skipsection = false;;
|
||||
for (auto i = rows.begin(); i != rows.end(); ++i) {
|
||||
StatisticsRow &row = (*i);
|
||||
QString name;
|
||||
|
||||
if (row.calc == SC_HEADING) { // All sections begin with a heading
|
||||
last = p_profile->LastGoodDay(row.type);
|
||||
first = p_profile->FirstGoodDay(row.type);
|
||||
|
||||
week = last.addDays(-6);
|
||||
month = last.addDays(-29);
|
||||
sixmonth = last.addMonths(-6);
|
||||
year = last.addMonths(-12);
|
||||
|
||||
if (week < first) { week = first; }
|
||||
if (month < first) { month = first; }
|
||||
if (sixmonth < first) { sixmonth = first; }
|
||||
if (year < first) { year = first; }
|
||||
|
||||
int days = PROFILE.countDays(row.type, first, last);
|
||||
skipsection = (days == 0);
|
||||
if (days > 0) {
|
||||
html+=QString("<tr bgcolor='"+heading_color+"'><td colspan=6 align=center><font size=+3>%1</font></td></tr>\n").arg(row.src);
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
// Bypass this entire section if no data is present
|
||||
if (skipsection) continue;
|
||||
|
||||
if (row.calc == SC_AHI) {
|
||||
name = ahitxt;
|
||||
} else if ((row.calc == SC_HOURS) || (row.calc == SC_COMPLIANCE)) {
|
||||
name = row.src;
|
||||
} else if (row.calc == SC_COLUMNHEADERS) {
|
||||
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</td></tr>")
|
||||
.arg(tr("Details")).arg(tr("Most Recent")).arg(tr("Last 7 Days")).arg(tr("Last 30 Days")).arg(
|
||||
tr("Last 6 months")).arg(tr("Last Year"));
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(ahitxt)
|
||||
.arg(calcAHI(lastcpap, lastcpap), 0, 'f', decimals)
|
||||
.arg(calcAHI(cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(calcAHI(cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(calcAHI(cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(calcAHI(cpapyear, lastcpap), 0, 'f', decimals);
|
||||
continue;
|
||||
} else if (row.calc == SC_DAYS) {
|
||||
QDate first=p_profile->FirstGoodDay(row.type);
|
||||
QDate last=p_profile->LastGoodDay(row.type);
|
||||
QString & machine = machinenames[row.type];
|
||||
int value=p_profile->countDays(row.type, first, last);
|
||||
|
||||
if (PROFILE.calcCount(CPAP_RERA, MT_CPAP, cpapyear, lastcpap)) {
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("RERA Index"))
|
||||
.arg(PROFILE.calcCount(CPAP_RERA, MT_CPAP, lastcpap, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, lastcpap, lastcpap), 0, 'f', decimals)
|
||||
.arg(PROFILE.calcCount(CPAP_RERA, MT_CPAP, cpapweek, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(PROFILE.calcCount(CPAP_RERA, MT_CPAP, cpapmonth, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(PROFILE.calcCount(CPAP_RERA, MT_CPAP, cpap6month, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(PROFILE.calcCount(CPAP_RERA, MT_CPAP, cpapyear, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, cpapyear, lastcpap), 0, 'f', decimals);
|
||||
}
|
||||
|
||||
if (PROFILE.calcCount(CPAP_FlowLimit, MT_CPAP, cpapyear, lastcpap)) {
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Flow Limit Index"))
|
||||
.arg(PROFILE.calcCount(CPAP_FlowLimit, MT_CPAP, lastcpap, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, lastcpap, lastcpap), 0, 'f', decimals)
|
||||
.arg(PROFILE.calcCount(CPAP_FlowLimit, MT_CPAP, cpapweek, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(PROFILE.calcCount(CPAP_FlowLimit, MT_CPAP, cpapmonth, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(PROFILE.calcCount(CPAP_FlowLimit, MT_CPAP, cpap6month, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(PROFILE.calcCount(CPAP_FlowLimit, MT_CPAP, cpapyear, lastcpap)
|
||||
/ PROFILE.calcHours(MT_CPAP, cpapyear, lastcpap), 0, 'f', decimals);
|
||||
}
|
||||
|
||||
|
||||
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Hours per Night"))
|
||||
.arg(formatTime(p_profile->calcHours(MT_CPAP)))
|
||||
.arg(formatTime(p_profile->calcHours(MT_CPAP, cpapweek, lastcpap) / float(cpapweekdays)))
|
||||
.arg(formatTime(p_profile->calcHours(MT_CPAP, cpapmonth, lastcpap) / float(cpapmonthdays)))
|
||||
.arg(formatTime(p_profile->calcHours(MT_CPAP, cpap6month, lastcpap) / float(cpap6monthdays)))
|
||||
.arg(formatTime(p_profile->calcHours(MT_CPAP, cpapyear, lastcpap) / float(cpapyeardays)));
|
||||
|
||||
if (cpapmode >= MODE_BIPAP) {
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Min EPAP"))
|
||||
.arg(p_profile->calcMin(CPAP_EPAP, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(CPAP_EPAP, MT_CPAP, cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(CPAP_EPAP, MT_CPAP, cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(CPAP_EPAP, MT_CPAP, cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(CPAP_EPAP, MT_CPAP, cpapyear, lastcpap), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(QString("%1% " + STR_TR_EPAP).arg(percentile * 100.0, 0, 'f', 0))
|
||||
.arg(p_profile->calcPercentile(CPAP_EPAP, percentile, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_EPAP, percentile, MT_CPAP, cpapweek, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_EPAP, percentile, MT_CPAP, cpapmonth, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_EPAP, percentile, MT_CPAP, cpap6month, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_EPAP, percentile, MT_CPAP, cpapyear, lastcpap), 0, 'f',
|
||||
decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Max IPAP"))
|
||||
.arg(p_profile->calcMax(CPAP_IPAP, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMax(CPAP_IPAP, MT_CPAP, cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMax(CPAP_IPAP, MT_CPAP, cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMax(CPAP_IPAP, MT_CPAP, cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMax(CPAP_IPAP, MT_CPAP, cpapyear, lastcpap), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(QString("%1% " + STR_TR_IPAP).arg(percentile * 100.0, 0, 'f', 0))
|
||||
.arg(p_profile->calcPercentile(CPAP_IPAP, percentile, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_IPAP, percentile, MT_CPAP, cpapweek, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_IPAP, percentile, MT_CPAP, cpapmonth, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_IPAP, percentile, MT_CPAP, cpap6month, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_IPAP, percentile, MT_CPAP, cpapyear, lastcpap), 0, 'f',
|
||||
decimals);
|
||||
} else if (cpapmode >= MODE_APAP) {
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Average Pressure"))
|
||||
.arg(p_profile->calcWavg(CPAP_Pressure, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(CPAP_Pressure, MT_CPAP, cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(CPAP_Pressure, MT_CPAP, cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(CPAP_Pressure, MT_CPAP, cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(CPAP_Pressure, MT_CPAP, cpapyear, lastcpap), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("%1% Pressure").arg(percentile * 100.0, 0, 'f', 0))
|
||||
.arg(p_profile->calcPercentile(CPAP_Pressure, percentile, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_Pressure, percentile, MT_CPAP, cpapweek, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_Pressure, percentile, MT_CPAP, cpapmonth, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_Pressure, percentile, MT_CPAP, cpap6month, lastcpap), 0, 'f',
|
||||
decimals)
|
||||
.arg(p_profile->calcPercentile(CPAP_Pressure, percentile, MT_CPAP, cpapyear, lastcpap), 0, 'f',
|
||||
decimals);
|
||||
if (value == 0) {
|
||||
html+="<tr><td colspan=6 align=center>"+
|
||||
QString(tr("No %1 data available.")).arg(machine)
|
||||
+"</td></tr>";
|
||||
} else if (value == 1) {
|
||||
html+="<tr><td colspan=6 align=center>"+
|
||||
QString("%1 day of %2 Data on %3")
|
||||
.arg(value)
|
||||
.arg(machine)
|
||||
.arg(last.toString())
|
||||
+"</td></tr>\n";
|
||||
} else {
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Pressure"))
|
||||
.arg(p_profile->calcSettingsMin(CPAP_Pressure, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcSettingsMin(CPAP_Pressure, MT_CPAP, cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcSettingsMin(CPAP_Pressure, MT_CPAP, cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcSettingsMin(CPAP_Pressure, MT_CPAP, cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcSettingsMin(CPAP_Pressure, MT_CPAP, cpapyear, lastcpap), 0, 'f', decimals);
|
||||
html+="<tr><td colspan=6 align=center>"+
|
||||
QString("%1 days of %2 Data, between %3 and %4")
|
||||
.arg(value)
|
||||
.arg(machine)
|
||||
.arg(first.toString())
|
||||
.arg(last.toString())
|
||||
+"</td></tr>\n";
|
||||
}
|
||||
|
||||
//html+="<tr><td colspan=6>TODO: 90% pressure.. Any point showing if this is all CPAP?</td></tr>";
|
||||
|
||||
|
||||
ChannelID leak;
|
||||
|
||||
if (p_profile->calcCount(CPAP_LeakTotal, MT_CPAP, cpapyear, lastcpap) > 0) {
|
||||
leak = CPAP_LeakTotal;
|
||||
} else { leak = CPAP_Leak; }
|
||||
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Average %1").arg(schema::channel[leak].label()))
|
||||
.arg(p_profile->calcWavg(leak, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(leak, MT_CPAP, cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(leak, MT_CPAP, cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(leak, MT_CPAP, cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(leak, MT_CPAP, cpapyear, lastcpap), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("%1% %2").arg(percentile * 100.0f, 0, 'f', 0).arg(schema::channel[leak].label()))
|
||||
.arg(p_profile->calcPercentile(leak, percentile, MT_CPAP), 0, 'f', decimals)
|
||||
.arg(p_profile->calcPercentile(leak, percentile, MT_CPAP, cpapweek, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcPercentile(leak, percentile, MT_CPAP, cpapmonth, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcPercentile(leak, percentile, MT_CPAP, cpap6month, lastcpap), 0, 'f', decimals)
|
||||
.arg(p_profile->calcPercentile(leak, percentile, MT_CPAP, cpapyear, lastcpap), 0, 'f', decimals);
|
||||
}
|
||||
}
|
||||
|
||||
int oxisize = oximeters.size();
|
||||
|
||||
if (oxisize > 0) {
|
||||
QDate lastoxi = p_profile->LastGoodDay(MT_OXIMETER);
|
||||
QDate firstoxi = p_profile->FirstGoodDay(MT_OXIMETER);
|
||||
int days = PROFILE.countDays(MT_OXIMETER, firstoxi, lastoxi);
|
||||
|
||||
if (days > 0) {
|
||||
html += QString("<tr><td colspan=6 align=center><b>%1</b></td></tr>").arg(tr("Oximetry Statistics"));
|
||||
|
||||
if (days == 1) {
|
||||
html += QString("<tr><td colspan=6 align=center>%1</td></tr>").arg(QString(
|
||||
tr("%1 day of Oximetry Data, on %2.")).arg(days).arg(firstoxi.toString(
|
||||
Qt::SystemLocaleShortDate)));
|
||||
} else {
|
||||
html += QString("<tr><td colspan=6 align=center>%1</td></tr>").arg(QString(
|
||||
tr("%1 days of Oximetry Data, between %2 and %3")).arg(days).arg(firstoxi.toString(
|
||||
Qt::SystemLocaleShortDate)).arg(lastoxi.toString(Qt::SystemLocaleShortDate)));
|
||||
continue;
|
||||
} else if (row.calc == SC_SUBHEADING) { // subheading..
|
||||
html+=QString("<tr bgcolor='"+subheading_color+"'><td colspan=6 align=center><b>%1</b></td></tr>\n").arg(row.src);
|
||||
continue;
|
||||
} else if (row.calc == SC_UNDEFINED) {
|
||||
continue;
|
||||
} else {
|
||||
ChannelID id = schema::channel[row.src].id();
|
||||
if ((id == NoChannel) || (!PROFILE.hasChannel(id))) {
|
||||
continue;
|
||||
}
|
||||
|
||||
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</td></tr>")
|
||||
.arg(tr("Details")).arg(tr("Most Recent")).arg(tr("Last 7 Days")).arg(tr("Last 30 Days")).arg(
|
||||
tr("Last 6 months")).arg(tr("Last Year"));
|
||||
QDate oxiweek = lastoxi.addDays(-7);
|
||||
QDate oximonth = lastoxi.addDays(-30);
|
||||
QDate oxi6month = lastoxi.addMonths(-6);
|
||||
QDate oxiyear = lastoxi.addYears(-12);
|
||||
|
||||
if (oxiweek < firstoxi) { oxiweek = firstoxi; }
|
||||
|
||||
if (oximonth < firstoxi) { oximonth = firstoxi; }
|
||||
|
||||
if (oxi6month < firstoxi) { oxi6month = firstoxi; }
|
||||
|
||||
if (oxiyear < firstoxi) { oxiyear = firstoxi; }
|
||||
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Average SpO2"))
|
||||
.arg(p_profile->calcWavg(OXI_SPO2, MT_OXIMETER), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(OXI_SPO2, MT_OXIMETER, oxiweek, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(OXI_SPO2, MT_OXIMETER, oximonth, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(OXI_SPO2, MT_OXIMETER, oxi6month, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(OXI_SPO2, MT_OXIMETER, oxiyear, lastoxi), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Minimum SpO2"))
|
||||
.arg(p_profile->calcMin(OXI_SPO2, MT_OXIMETER), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(OXI_SPO2, MT_OXIMETER, oxiweek, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(OXI_SPO2, MT_OXIMETER, oximonth, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(OXI_SPO2, MT_OXIMETER, oxi6month, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(OXI_SPO2, MT_OXIMETER, oxiyear, lastoxi), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("SpO2 Events / Hour"))
|
||||
.arg(p_profile->calcCount(OXI_SPO2Drop, MT_OXIMETER)
|
||||
/ p_profile->calcHours(MT_OXIMETER), 0, 'f', decimals)
|
||||
.arg(p_profile->calcCount(OXI_SPO2Drop, MT_OXIMETER, oxiweek, lastoxi)
|
||||
/ p_profile->calcHours(MT_OXIMETER, oxiweek, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcCount(OXI_SPO2Drop, MT_OXIMETER, oximonth, lastoxi)
|
||||
/ p_profile->calcHours(MT_OXIMETER, oximonth, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcCount(OXI_SPO2Drop, MT_OXIMETER, oxi6month, lastoxi)
|
||||
/ p_profile->calcHours(MT_OXIMETER, oxi6month, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcCount(OXI_SPO2Drop, MT_OXIMETER, oxiyear, lastoxi)
|
||||
/ p_profile->calcHours(MT_OXIMETER, oxiyear, lastoxi), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2\%</td><td>%3\%</td><td>%4\%</td><td>%5\%</td><td>%6\%</td></tr>")
|
||||
.arg(tr("% of time in SpO2 Events"))
|
||||
.arg(100.0 / p_profile->calcHours(MT_OXIMETER) * p_profile->calcSum(OXI_SPO2Drop,
|
||||
MT_OXIMETER) / 3600.0, 0, 'f', decimals)
|
||||
.arg(100.0 / p_profile->calcHours(MT_OXIMETER, oxiweek, lastoxi)
|
||||
* p_profile->calcSum(OXI_SPO2Drop, MT_OXIMETER, oxiweek, lastoxi)
|
||||
/ 3600.0, 0, 'f', decimals)
|
||||
.arg(100.0 / p_profile->calcHours(MT_OXIMETER, oximonth, lastoxi)
|
||||
* p_profile->calcSum(OXI_SPO2Drop, MT_OXIMETER, oximonth, lastoxi)
|
||||
/ 3600.0, 0, 'f', decimals)
|
||||
.arg(100.0 / p_profile->calcHours(MT_OXIMETER, oxi6month, lastoxi)
|
||||
* p_profile->calcSum(OXI_SPO2Drop, MT_OXIMETER, oxi6month, lastoxi)
|
||||
/ 3600.0, 0, 'f',decimals)
|
||||
.arg(100.0 / p_profile->calcHours(MT_OXIMETER, oxiyear, lastoxi)
|
||||
* p_profile->calcSum(OXI_SPO2Drop, MT_OXIMETER, oxiyear, lastoxi)
|
||||
/ 3600.0, 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Average Pulse Rate"))
|
||||
.arg(p_profile->calcWavg(OXI_Pulse, MT_OXIMETER), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(OXI_Pulse, MT_OXIMETER, oxiweek, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(OXI_Pulse, MT_OXIMETER, oximonth, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(OXI_Pulse, MT_OXIMETER, oxi6month, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcWavg(OXI_Pulse, MT_OXIMETER, oxiyear, lastoxi), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Minimum Pulse Rate"))
|
||||
.arg(p_profile->calcMin(OXI_Pulse, MT_OXIMETER), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(OXI_Pulse, MT_OXIMETER, oxiweek, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(OXI_Pulse, MT_OXIMETER, oximonth, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(OXI_Pulse, MT_OXIMETER, oxi6month, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMin(OXI_Pulse, MT_OXIMETER, oxiyear, lastoxi), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Maximum Pulse Rate"))
|
||||
.arg(p_profile->calcMax(OXI_Pulse, MT_OXIMETER), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMax(OXI_Pulse, MT_OXIMETER, oxiweek, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMax(OXI_Pulse, MT_OXIMETER, oximonth, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMax(OXI_Pulse, MT_OXIMETER, oxi6month, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcMax(OXI_Pulse, MT_OXIMETER, oxiyear, lastoxi), 0, 'f', decimals);
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
|
||||
.arg(tr("Pulse Change Events / Hour"))
|
||||
.arg(p_profile->calcCount(OXI_PulseChange, MT_OXIMETER) / p_profile->calcHours(MT_OXIMETER), 0,
|
||||
'f', decimals)
|
||||
.arg(p_profile->calcCount(OXI_PulseChange, MT_OXIMETER, oxiweek,
|
||||
lastoxi) / p_profile->calcHours(MT_OXIMETER, oxiweek, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcCount(OXI_PulseChange, MT_OXIMETER, oximonth,
|
||||
lastoxi) / p_profile->calcHours(MT_OXIMETER, oximonth, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcCount(OXI_PulseChange, MT_OXIMETER, oxi6month,
|
||||
lastoxi) / p_profile->calcHours(MT_OXIMETER, oxi6month, lastoxi), 0, 'f', decimals)
|
||||
.arg(p_profile->calcCount(OXI_PulseChange, MT_OXIMETER, oxiyear,
|
||||
lastoxi) / p_profile->calcHours(MT_OXIMETER, oxiyear, lastoxi), 0, 'f', decimals);
|
||||
name = calcnames[row.calc].arg(row.src);
|
||||
}
|
||||
|
||||
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>\n")
|
||||
.arg(name)
|
||||
.arg(row.value(last,last))
|
||||
.arg(row.value(week,last))
|
||||
.arg(row.value(month,last))
|
||||
.arg(row.value(sixmonth,last))
|
||||
.arg(row.value(year,last));
|
||||
}
|
||||
|
||||
html += "</table>";
|
||||
@ -732,8 +602,6 @@ QString Statistics::GenerateHTML()
|
||||
|
||||
QList<UsageData> AHI;
|
||||
|
||||
//QDate bestAHIdate, worstAHIdate;
|
||||
//EventDataType bestAHI=999.0, worstAHI=0;
|
||||
if (cpapdays > 0) {
|
||||
QDate first, last = lastcpap;
|
||||
CPAPMode mode = MODE_UNKNOWN, cmode = MODE_UNKNOWN;
|
||||
@ -1087,9 +955,9 @@ QString Statistics::GenerateHTML()
|
||||
/*RXsort=RX_min;
|
||||
RXorder=true;
|
||||
qSort(rxchange.begin(),rxchange.end());*/
|
||||
html += "<div align=center>";
|
||||
html += QString("<br/><b>") + tr("Changes to Prescription Settings") + "</b>";
|
||||
html += "<div align=center><br/>";
|
||||
html += QString("<table cellpadding=2 cellspacing=0 border=1 width=90%>");
|
||||
html += "<tr bgcolor='"+heading_color+"'><td colspan=9 align=center><font size=+3>" + tr("Changes to Prescription Settings") + "</font></td></tr>";
|
||||
QString extratxt;
|
||||
|
||||
QString tooltip;
|
||||
@ -1230,10 +1098,11 @@ QString Statistics::GenerateHTML()
|
||||
}
|
||||
|
||||
if (mach.size() > 0) {
|
||||
html += "<div align=center>";
|
||||
html += "<div align=center><br/>";
|
||||
|
||||
html += QString("<br/><b>") + tr("Machine Information") + "</b>";
|
||||
html += QString("<table cellpadding=2 cellspacing=0 border=1 width=90%>");
|
||||
html += "<tr bgcolor='"+heading_color+"'><td colspan=5 align=center><font size=+3>" + tr("Machine Information") + "</font></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></tr>")
|
||||
.arg(STR_TR_Brand)
|
||||
.arg(STR_TR_Model)
|
||||
@ -1267,3 +1136,64 @@ QString Statistics::GenerateHTML()
|
||||
html += htmlFooter();
|
||||
return html;
|
||||
}
|
||||
|
||||
|
||||
QString StatisticsRow::value(QDate start, QDate end)
|
||||
{
|
||||
const int decimals=2;
|
||||
QString value = "???";
|
||||
qDebug() << "Calculating " << src << calc << "for" << start << end;
|
||||
float days = PROFILE.countDays(type, start, end);
|
||||
|
||||
// Handle special data sources first
|
||||
if (calc == SC_AHI) {
|
||||
value = QString("%1").arg(calcAHI(start, end), 0, 'f', decimals);
|
||||
} else if (calc == SC_HOURS) {
|
||||
value = QString("%1").arg(formatTime(p_profile->calcHours(type, start, end) / days));
|
||||
} else if (calc == SC_COMPLIANCE) {
|
||||
float c = p_profile->countCompliantDays(type, start, end);
|
||||
float p = (100.0 / days) * c;
|
||||
value = QString("%1%").arg(p, 0, 'f', 0);
|
||||
} else if (calc == SC_DAYS) {
|
||||
value = QString("%1").arg(p_profile->countDays(type, start, end));
|
||||
} else if ((calc == SC_COLUMNHEADERS) || (calc == SC_SUBHEADING) || (calc == SC_UNDEFINED)) {
|
||||
} else {
|
||||
//
|
||||
ChannelID code=channel();
|
||||
if (code != NoChannel) {
|
||||
switch(calc) {
|
||||
case SC_AVG:
|
||||
value = QString("%1").arg(p_profile->calcAvg(code, type, start, end), 0, 'f', decimals);
|
||||
break;
|
||||
case SC_WAVG:
|
||||
value = QString("%1").arg(p_profile->calcWavg(code, type, start, end), 0, 'f', decimals);
|
||||
break;
|
||||
case SC_MEDIAN:
|
||||
value = QString("%1").arg(p_profile->calcPercentile(code, 0.5, type, start, end), 0, 'f', decimals);
|
||||
break;
|
||||
case SC_90P:
|
||||
value = QString("%1").arg(p_profile->calcPercentile(code, 0.9, type, start, end), 0, 'f', decimals);
|
||||
break;
|
||||
case SC_MIN:
|
||||
value = QString("%1").arg(p_profile->calcMin(code, type, start, end), 0, 'f', decimals);
|
||||
break;
|
||||
case SC_MAX:
|
||||
value = QString("%1").arg(p_profile->calcMax(code, type, start, end), 0, 'f', decimals);
|
||||
break;
|
||||
case SC_CPH:
|
||||
value = QString("%1").arg(PROFILE.calcCount(code, type, start, end)
|
||||
/ PROFILE.calcHours(type, start, end), 0, 'f', decimals);
|
||||
break;
|
||||
case SC_SPH:
|
||||
value = QString("%1%").arg(100.0 / p_profile->calcHours(type, start, end)
|
||||
* p_profile->calcSum(code, type, start, end)
|
||||
/ 3600.0, 0, 'f', decimals);
|
||||
|
||||
default:
|
||||
break;
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
return value;
|
||||
}
|
||||
|
@ -13,6 +13,88 @@
|
||||
#define SUMMARY_H
|
||||
|
||||
#include <QObject>
|
||||
#include <QHash>
|
||||
#include <QList>
|
||||
#include "SleepLib/schema.h"
|
||||
|
||||
enum StatCalcType {
|
||||
SC_UNDEFINED=0, SC_COLUMNHEADERS, SC_HEADING, SC_SUBHEADING, SC_MEDIAN, SC_AVG, SC_WAVG, SC_90P, SC_MIN, SC_MAX, SC_CPH, SC_SPH, SC_AHI, SC_HOURS, SC_COMPLIANCE, SC_DAYS
|
||||
};
|
||||
|
||||
struct StatisticsRow {
|
||||
StatisticsRow() { calc=SC_UNDEFINED; }
|
||||
StatisticsRow(QString src, QString calc, QString type) {
|
||||
this->src = src;
|
||||
this->calc = lookupCalc(calc);
|
||||
this->type = lookupType(type);
|
||||
}
|
||||
StatisticsRow(QString src, StatCalcType calc, MachineType type) {
|
||||
this->src = src;
|
||||
this->calc = calc;
|
||||
this->type = type;
|
||||
}
|
||||
StatisticsRow(const StatisticsRow ©) {
|
||||
src=copy.src;
|
||||
calc=copy.calc;
|
||||
type=copy.type;
|
||||
}
|
||||
QString src;
|
||||
StatCalcType calc;
|
||||
MachineType type;
|
||||
|
||||
StatCalcType lookupCalc(QString calc)
|
||||
{
|
||||
if (calc.compare("avg",Qt::CaseInsensitive)==0) {
|
||||
return SC_AVG;
|
||||
} else if (calc.compare("w-avg",Qt::CaseInsensitive)==0) {
|
||||
return SC_WAVG;
|
||||
} else if (calc.compare("median",Qt::CaseInsensitive)==0) {
|
||||
return SC_MEDIAN;
|
||||
} else if (calc.compare("90%",Qt::CaseInsensitive)==0) {
|
||||
return SC_90P;
|
||||
} else if (calc.compare("min", Qt::CaseInsensitive)==0) {
|
||||
return SC_MIN;
|
||||
} else if (calc.compare("max", Qt::CaseInsensitive)==0) {
|
||||
return SC_MAX;
|
||||
} else if (calc.compare("cph", Qt::CaseInsensitive)==0) {
|
||||
return SC_CPH;
|
||||
} else if (calc.compare("sph", Qt::CaseInsensitive)==0) {
|
||||
return SC_SPH;
|
||||
} else if (calc.compare("ahi", Qt::CaseInsensitive)==0) {
|
||||
return SC_AHI;
|
||||
} else if (calc.compare("hours", Qt::CaseInsensitive)==0) {
|
||||
return SC_HOURS;
|
||||
} else if (calc.compare("compliance", Qt::CaseInsensitive)==0) {
|
||||
return SC_COMPLIANCE;
|
||||
} else if (calc.compare("days", Qt::CaseInsensitive)==0) {
|
||||
return SC_DAYS;
|
||||
} else if (calc.compare("heading", Qt::CaseInsensitive)==0) {
|
||||
return SC_HEADING;
|
||||
} else if (calc.compare("subheading", Qt::CaseInsensitive)==0) {
|
||||
return SC_SUBHEADING;
|
||||
}
|
||||
return SC_UNDEFINED;
|
||||
}
|
||||
|
||||
MachineType lookupType(QString type)
|
||||
{
|
||||
if (type.compare("cpap", Qt::CaseInsensitive)==0) {
|
||||
return MT_CPAP;
|
||||
} else if (type.compare("oximeter", Qt::CaseInsensitive)==0) {
|
||||
return MT_OXIMETER;
|
||||
} else if (type.compare("sleepstage", Qt::CaseInsensitive)==0) {
|
||||
return MT_SLEEPSTAGE;
|
||||
}
|
||||
return MT_UNKNOWN;
|
||||
}
|
||||
|
||||
|
||||
ChannelID channel() {
|
||||
return schema::channel[src].id();
|
||||
}
|
||||
|
||||
QString value(QDate start, QDate end);
|
||||
};
|
||||
|
||||
class Statistics : public QObject
|
||||
{
|
||||
@ -20,7 +102,13 @@ class Statistics : public QObject
|
||||
public:
|
||||
explicit Statistics(QObject *parent = 0);
|
||||
|
||||
static QString GenerateHTML();
|
||||
QString GenerateHTML();
|
||||
|
||||
protected:
|
||||
// Using a map to maintain order
|
||||
QList<StatisticsRow> rows;
|
||||
QMap<StatCalcType, QString> calcnames;
|
||||
QMap<MachineType, QString> machinenames;
|
||||
|
||||
signals:
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user