mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
307 lines
14 KiB
C++
307 lines
14 KiB
C++
/* Welcome Page Implementation
|
|
*
|
|
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
|
*
|
|
* 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 <QString>
|
|
#include <QApplication>
|
|
#include <QFont>
|
|
#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<Machine *> cpap_machines = p_profile->GetMachines(MT_CPAP);
|
|
QList<Machine *> oximeters = p_profile->GetMachines(MT_OXIMETER);
|
|
QList<Machine *> 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><head>")+
|
|
"</head>"
|
|
"<style type='text/css'>"
|
|
"p,a,td,body { font-family: '"+QApplication::font().family()+"'; }"
|
|
"p,a,td,body { font-size: "+QString::number(QApplication::font().pointSize() + 2)+"px; }"
|
|
|
|
"table.curved {"
|
|
"border: 1px solid gray;"
|
|
"border-radius:10px;"
|
|
"-moz-border-radius:10px;"
|
|
"-webkit-border-radius:10px;"
|
|
"page-break-after:auto;"
|
|
"-fs-table-paginate: paginate;"
|
|
"}"
|
|
"table.curved2 {"
|
|
"border: 1px solid gray;"
|
|
"border-radius:10px;"
|
|
"background:#ffffc0;"
|
|
"-moz-border-radius:10px;"
|
|
"-webkit-border-radius:10px;"
|
|
"page-break-after:auto;"
|
|
"-fs-table-paginate: paginate;"
|
|
"}"
|
|
"tr.datarow:nth-child(even) {"
|
|
"background-color: #f8f8f8;"
|
|
"}"
|
|
"table { page-break-after:auto; -fs-table-paginate: paginate; }"
|
|
"tr { page-break-inside:avoid; page-break-after:auto }"
|
|
"td { page-break-inside:avoid; page-break-after:auto }"
|
|
"thead { display:table-header-group; }"
|
|
"tfoot { display:table-footer-group; }"
|
|
|
|
|
|
"</style>"
|
|
"<link rel='stylesheet' type='text/css' href='qrc:/docs/tooltips.css' />"
|
|
"<script type='text/javascript'>"
|
|
"function ChangeColor(tableRow, highLight)"
|
|
"{ tableRow.style.backgroundColor = highLight; }"
|
|
"function Go(url) { throw(url); }"
|
|
"</script>"
|
|
"</head>"
|
|
"<body leftmargin=0 topmargin=5 rightmargin=0>";
|
|
|
|
html += "<div align=center><table border=0 height=100% width=99%>";
|
|
html += QString("<tr><td colspan=2 align=center>") +
|
|
"<img src='qrc:/icons/bob-v3.0.png' height=100px>"
|
|
|
|
"<h1>" + QObject::tr("Welcome to SleepyHead") + "</h1>" +
|
|
|
|
"<table cellpadding=4 border=0>";
|
|
|
|
int cols=2;
|
|
if (havecpapdata || haveoximeterdata) cols=4;
|
|
|
|
|
|
html+=QString("<tr><td colspan=%1 align=center>").arg(cols)+
|
|
"<font size=+1>"+((havecpapdata || haveoximeterdata) ? QObject::tr("What would you like to do?") : QObject::tr("Please Import Some Data")) +"</font></td>"
|
|
"</tr>"
|
|
"<tr>"
|
|
" <td align=center>"
|
|
" <table class=curved cellpadding=4>"
|
|
" <tr><td align=center onmouseover='ChangeColor(this, \"#eeeeee\");' onmouseout='ChangeColor(this, \"#ffffff\");' onclick='alert(\"import=cpap\");'><font size=+1><img src='qrc:/icons/sdcard.png' width=128px><br/>" + QObject::tr("CPAP<br/>Importer")+"</font></td></tr>"
|
|
" </table>"
|
|
" </td>"
|
|
"<td align=center>"
|
|
" <table class=curved cellpadding=4>"
|
|
" <tr><td align=center onmouseover='ChangeColor(this, \"#eeeeee\");' onmouseout='ChangeColor(this, \"#ffffff\");' onclick='alert(\"import=oximeter\");'><font size=+1><img src='qrc:/icons/cms50f.png' width=128px><br/>" + QObject::tr("Oximetery<br/>Wizard")+"</font></td></tr>"
|
|
" </table>"
|
|
"</td>";
|
|
if (havecpapdata || haveoximeterdata) {
|
|
html += "<td align=center><font size=+2>"+QObject::tr("or")+"</font></td>"
|
|
"<td align=center>"
|
|
" <table class=curved cellpadding=4>"
|
|
" <tr><td align=center onmouseover='ChangeColor(this, \"#eeeeee\");' onmouseout='ChangeColor(this, \"#ffffff\");' onclick='alert(\"statistics=1\");'><font size=+1><img src='qrc:/icons/statistics.png' width=128px><br/>" + QObject::tr("View<br/>Statistics")+"</font></td></tr>"
|
|
" </table>"
|
|
"</td>"
|
|
"<td align=center>"
|
|
" <table class=curved cellpadding=4>"
|
|
" <tr><td align=center onmouseover='ChangeColor(this, \"#eeeeee\");' onmouseout='ChangeColor(this, \"#ffffff\");' onclick='alert(\"daily=1\");'><font size=+1><img src='qrc:/icons/daily.png' width=128px><br/>" + QObject::tr("View<br/>Daily")+"</font></td></tr>"
|
|
" </table>"
|
|
"</td>";
|
|
}
|
|
html += "</tr></table>";
|
|
|
|
Machine * cpap = nullptr;
|
|
if (!havecpapdata && !haveoximeterdata) {
|
|
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 {
|
|
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 += "<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(cpap->brand()+" "+cpap->model())+"</b><br/>";
|
|
|
|
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)) + "<br/>";
|
|
|
|
EventDataType hours = day->hours();
|
|
html += "<br/>";
|
|
|
|
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)+"<br/>";
|
|
else html += QObject::tr("<font color = red>You only had the mask on for %1.</font>").arg(timestr)+"<br/>";
|
|
|
|
|
|
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 <b>%2</b> your %3 day average of %4.").arg(ahi,0,'f',2).arg(comp).arg(averagedays).arg(ahidays,0,'f',2);
|
|
|
|
html += "<br/>";
|
|
|
|
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())+"<br/>";
|
|
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)+"<br/>";
|
|
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 += "<br/>";
|
|
|
|
//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 <b>%3</b> 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 += "<br/>";
|
|
|
|
html += "</td></tr></table></td></tr></table>";
|
|
|
|
} else {
|
|
html += "<p>"+QObject::tr("No CPAP data has been imported yet.")+"</p>";
|
|
}
|
|
if (haveoximeterdata) {
|
|
QDate oxidate=p_profile->LastDay(MT_OXIMETER);
|
|
int daysto = oxidate.daysTo(QDate::currentDate());
|
|
|
|
html += "<p>"+QObject::tr("Most recent Oximetery data: <a onclick='alert(\"daily=%2\");'>%1</a> ").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+="</p>";
|
|
} else {
|
|
html += "<p>"+QObject::tr("No oximetery data has been imported yet.")+"</p>";
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// 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<Machine *> mlist = p_profile->GetMachines(MT_CPAP);
|
|
for (int i=0; i<mlist.size(); ++i) {
|
|
Machine *mach = mlist.at(i);
|
|
if (mach->series().compare("S9") == 0) showCardWarning = true;
|
|
}
|
|
|
|
|
|
if (showCardWarning) {
|
|
html += QString("<div align=center><table class=curved cellpadding=3 width=45%>")+
|
|
"<tr>"
|
|
"<td align=center colspan=2><font size=+1><b>"+QObject::tr("Very Important Warning For ResMed S9 Users")+"</b></font></td></tr>"
|
|
"<tr><td align=left>"+
|
|
QObject::tr("<p>ALWAYS <font size=+1 color=red><b>write protect</b></font> CPAP SDCards before inserting them into your computer.")+"</p>"+
|
|
QObject::tr("<p><span title=\"Mac OSX and Win8.1\" onmouseover='ChangeColor(this, \"#eeeeee\");' onmouseout='ChangeColor(this, \"#ffffff\");'><font color=blue>Certain operating systems</font></span> write index files to the card without asking, which can render your card unreadable by your cpap machine.")+"</p>"+
|
|
QObject::tr("<p>As a second line of protection, ALWAYS UNMOUNT the data card properly before removing it!</p>")+
|
|
"</td>"
|
|
"<td><img src=\"qrc:/icons/sdcard-lock.png\" width=128px></td>"
|
|
"</tr>"
|
|
"</table>"
|
|
"</td></tr></table></div>";
|
|
} else {
|
|
}
|
|
|
|
html += "<script type='text/javascript' language='javascript' src='qrc:/docs/script.js'></script>"
|
|
"</body></html>";
|
|
return html;
|
|
}
|
|
|