/* Welcome Page Implementation * * Copyright (c) 2011-2018 Mark Watkins * * This file is subject to the terms and conditions of the GNU General Public * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ #include #include #include #include #include "SleepLib/profiles.h" // Pinching a few compounded functions from statistics that may come in useful here extern EventDataType calcAHI(QDate start, QDate end); extern EventDataType calcFL(QDate start, QDate end); QString GenerateWelcomeHTML() { QList cpap_machines = p_profile->GetMachines(MT_CPAP); QList oximeters = p_profile->GetMachines(MT_OXIMETER); QList mach; mach.append(cpap_machines); mach.append(oximeters); bool havecpapdata = false; bool haveoximeterdata = false; for (int i=0; i < cpap_machines.size(); ++i) { int daysize = cpap_machines[i]->day.size(); if (daysize > 0) { havecpapdata = true; break; } } for (int i=0; i < oximeters.size(); ++i) { int daysize = oximeters[i]->day.size(); if (daysize > 0) { haveoximeterdata = true; break; } } QString html = QString("")+ "" "" "" "" "" ""; html += "
"; html += QString("
") + "" "

" + QObject::tr("Welcome to SleepyHead") + "

" + ""; int cols=2; if (havecpapdata || haveoximeterdata) cols=5; html+=QString("" "" "" " " ""; if (havecpapdata || haveoximeterdata) { html += "" "" ""; } html += "
").arg(cols)+ ""+((havecpapdata || haveoximeterdata) ? QObject::tr("What would you like to do?") : QObject::tr("Please Import Some Data")) +"
" " " " " "

" + QObject::tr("CPAP
Importer")+"
" "
" " " " " "

" + QObject::tr("Oximetery
Wizard")+"
" "
"+QObject::tr("or")+"" " " " " "

" + QObject::tr("View
Statistics")+"
" "
" " " " " "

" + QObject::tr("View
Daily")+"
" "
"; Machine * cpap = nullptr; if (!havecpapdata && !haveoximeterdata) { html += "

" + QObject::tr("It might be a good idea to check preferences first,
as there are some options that affect import.")+"

" "

" + QObject::tr("First import can take a few minutes.") + "

"; } else { QDate date = p_profile->LastDay(MT_CPAP); Day *day = p_profile->GetDay(date, MT_CPAP); if (havecpapdata && day) { cpap = day->machine(MT_CPAP); } if (day && (cpap != nullptr)) { QString cpapimage = "qrc"+cpap->getPixmapPath(); html += "

"; html+="
"+ QString("
").arg(date.toString(Qt::ISODate))+""+ QObject::tr("The last time you used your %1...").arg(cpap->brand()+" "+cpap->series()+" "+cpap->model())+"
"; int daysto = date.daysTo(QDate::currentDate()); QString daystring; if (daysto == 1) daystring += QObject::tr("last night"); else if (daysto == 2) daystring += QObject::tr("yesterday"); else daystring += QObject::tr("%2 days ago").arg(date.daysTo(QDate::currentDate())); html += QObject::tr("was %1 (on %2)").arg(daystring).arg(date.toString(Qt::SystemLocaleLongDate)) + "
"; EventDataType hours = day->hours(); html += "
"; int seconds = int(hours * 3600.0) % 60; int minutes = int(hours * 60) % 60; int hour = hours; QString timestr = QObject::tr("%1 hours, %2 minutes and %3 seconds").arg(hour).arg(minutes).arg(seconds); const EventDataType compliance_min = 4.0; if (hours > compliance_min) html += QObject::tr("Your machine was on for %1.").arg(timestr)+"
"; else html += QObject::tr("You only had the mask on for %1.").arg(timestr)+"
"; int averagedays = 7; // how many days to look back QDate starttime = date.addDays(-(averagedays-1)); EventDataType ahi = (day->count(CPAP_Obstructive) + day->count(CPAP_Hypopnea) + day->count(CPAP_ClearAirway) + day->count(CPAP_Apnea)) / hours; EventDataType ahidays = calcAHI(starttime, date); const QString under = QObject::tr("under"); const QString over = QObject::tr("over"); const QString close = QObject::tr("reasonably close to"); const QString equal = QObject::tr("equal to"); QString comp; if ((ahi < ahidays) && ((ahidays - ahi) >= 0.1)) { comp = under; } else if ((ahi > ahidays) && ((ahi - ahidays) >= 0.1)) { comp = over; } else if ((fabs(ahi > ahidays) >= 0.01) ) { comp = close; } else { comp = equal; } html += QObject::tr("You had an AHI of %1, which is %2 your %3 day average of %4.").arg(ahi,0,'f',2).arg(comp).arg(averagedays).arg(ahidays,0,'f',2); html += "
"; CPAPMode cpapmode = (CPAPMode)(int)day->settings_max(CPAP_Mode); double perc= p_profile->general->prefCalcPercentile(); if (cpapmode == MODE_CPAP) { EventDataType pressure = day->settings_max(CPAP_Pressure); html += QObject::tr("Your CPAP machine blasted you with a constant %1%2 of air").arg(pressure).arg(schema::channel[CPAP_Pressure].units()); } else if (cpapmode == MODE_APAP) { EventDataType pressure = day->percentile(CPAP_Pressure, perc/100.0); html += QObject::tr("Your pressure was under %1%2 for %3% of the time.").arg(pressure).arg(schema::channel[CPAP_Pressure].units()).arg(perc); } else if (cpapmode == MODE_BILEVEL_FIXED) { EventDataType ipap = day->settings_max(CPAP_IPAP); EventDataType epap = day->settings_min(CPAP_EPAP); html += QObject::tr("Your machine blasted you with a constant %1-%2 %3 of air.").arg(epap).arg(ipap).arg(schema::channel[CPAP_Pressure].units()); } else if (cpapmode == MODE_BILEVEL_AUTO_FIXED_PS) { EventDataType ipap = day->percentile(CPAP_IPAP, perc/100.0); EventDataType epap = day->percentile(CPAP_EPAP, perc/100.0); html += QObject::tr("Your machine was under %1-%2 %3 for %4% of the time.").arg(epap).arg(ipap).arg(schema::channel[CPAP_Pressure].units()).arg(perc); } else if (cpapmode == MODE_ASV){ EventDataType ipap = day->percentile(CPAP_IPAP, perc/100.0); EventDataType epap = qRound(day->settings_wavg(CPAP_EPAP)); html += QObject::tr("Your EPAP pressure fixed at %1%2.").arg(epap).arg(schema::channel[CPAP_EPAP].units())+"
"; html += QObject::tr("Your IPAP pressure was under %1%2 for %3% of the time.").arg(ipap).arg(schema::channel[CPAP_IPAP].units()).arg(perc); } else if (cpapmode == MODE_ASV_VARIABLE_EPAP){ EventDataType ipap = day->percentile(CPAP_IPAP, perc/100.0); EventDataType epap = day->percentile(CPAP_EPAP, perc/100.0); html += QObject::tr("Your EPAP pressure was under %1%2 for %3% of the time.").arg(epap).arg(schema::channel[CPAP_EPAP].units()).arg(perc)+"
"; html += QObject::tr("Your IPAP pressure was under %1%2 for %3% of the time.").arg(ipap).arg(schema::channel[CPAP_IPAP].units()).arg(perc); } html += "
"; //EventDataType lat = day->timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline())/ 60.0; //EventDataType leaks = 1.0/hours * lat; EventDataType leak = day->avg(CPAP_Leak); EventDataType leakdays = p_profile->calcAvg(CPAP_Leak, MT_CPAP, starttime, date); if ((leak < leakdays) && ((leakdays - leak) >= 0.1)) { comp = under; } else if ((leak > leakdays) && ((leak - leakdays) >= 0.1)) { comp = over; } else if ((fabs(ahi > ahidays) >= 0.01) ) { comp = close; } else { comp = equal; } html += QObject::tr("Your average leaks were %1 %2, which is %3 your %4 day average of %5.").arg(leak,0,'f',2).arg(schema::channel[CPAP_Leak].units()).arg(comp).arg(averagedays).arg(leakdays,0,'f',2); html += "
"; html += "
"; } else { html += "

"+QObject::tr("No CPAP data has been imported yet.")+"

"; } if (haveoximeterdata) { QDate oxidate=p_profile->LastDay(MT_OXIMETER); int daysto = oxidate.daysTo(QDate::currentDate()); html += "

"+QObject::tr("Most recent Oximetery data: %1 ").arg(oxidate.toString(Qt::SystemLocaleLongDate)).arg(oxidate.toString(Qt::ISODate)); if (daysto == 1) html += QObject::tr("(last night)"); else if (daysto == 2) html += QObject::tr("(yesterday)"); else html += QObject::tr("(%2 day ago)").arg(oxidate.daysTo(QDate::currentDate())); html+="

"; } else { html += "

"+QObject::tr("No oximetery data has been imported yet.")+"

"; } } // The SDCard warning does not need to be seen anymore for people who DON'T use ResMed S9's.. show first import and only when S9 is present bool showCardWarning = (cpap == nullptr); QList mlist = p_profile->GetMachines(MT_CPAP); for (int i=0; iseries().compare("S9") == 0) showCardWarning = true; } if (showCardWarning) { html += QString("
")+ "" "" "" "" "" "
"+QObject::tr("Very Important Warning For ResMed S9 Users")+"
"+ QObject::tr("

ALWAYS write protect CPAP SDCards before inserting them into your computer.")+"

"+ QObject::tr("

Certain operating systems write index files to the card without asking, which can render your card unreadable by your cpap machine.")+"

"+ QObject::tr("

As a second line of protection, ALWAYS UNMOUNT the data card properly before removing it!

")+ "
" "
"; } else { } html += "" ""; return html; }