diff --git a/sleepyhead/SleepLib/profiles.cpp b/sleepyhead/SleepLib/profiles.cpp index 786052da..a647de14 100644 --- a/sleepyhead/SleepLib/profiles.cpp +++ b/sleepyhead/SleepLib/profiles.cpp @@ -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()) { diff --git a/sleepyhead/SleepLib/profiles.h b/sleepyhead/SleepLib/profiles.h index 973407f5..dc8b6287 100644 --- a/sleepyhead/SleepLib/profiles.h +++ b/sleepyhead/SleepLib/profiles.h @@ -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(), diff --git a/sleepyhead/mainwindow.cpp b/sleepyhead/mainwindow.cpp index c1ab8798..b8f06fa4 100644 --- a/sleepyhead/mainwindow.cpp +++ b/sleepyhead/mainwindow.cpp @@ -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(); diff --git a/sleepyhead/sleepyhead.pro b/sleepyhead/sleepyhead.pro index 7117a62f..a62b3bb0 100644 --- a/sleepyhead/sleepyhead.pro +++ b/sleepyhead/sleepyhead.pro @@ -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) { diff --git a/sleepyhead/statistics.cpp b/sleepyhead/statistics.cpp index e190140e..84f9926c 100644 --- a/sleepyhead/statistics.cpp +++ b/sleepyhead/statistics.cpp @@ -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() "" "
" ""+QObject::tr("Warning:")+" "+
- // #ifdef TEST_BUILD
- // QObject::tr("This is an testing build so expect the possibility things will go wrong.")+" "+ - // 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.")+" "+ - // QObject::tr("Please report any bugs you find to SleepyHead's SourceForge page.")+ - // #endif - //" | "
- //"||
" +
QString(QObject::tr("Name: %1, %2")).arg(PROFILE.user->lastName()).arg(
PROFILE.user->firstName()) + " " + @@ -89,14 +156,6 @@ QString htmlFooter() " |