2018-05-07 15:11:02 +00:00
# include " welcome.h "
# include "ui_welcome.h"
2014-07-25 07:53:48 +00:00
2018-05-07 15:11:02 +00:00
# include "mainwindow.h"
extern MainWindow * mainwin ;
2014-07-25 07:53:48 +00:00
2018-05-07 15:11:02 +00:00
Welcome : : Welcome ( QWidget * parent ) :
QWidget ( parent ) ,
ui ( new Ui : : Welcome )
{
ui - > setupUi ( this ) ;
if ( ! p_profile ) {
return ;
}
// 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
const auto & mlist = p_profile - > GetMachines ( MT_CPAP ) ;
bool showCardWarning = ( mlist . size ( ) = = 0 ) ;
for ( auto & mach : mlist ) {
if ( mach - > series ( ) . compare ( " S9 " ) = = 0 ) showCardWarning = true ;
}
ui - > S9Warning - > setVisible ( showCardWarning ) ;
ui - > cpapInfo - > setHtml ( GenerateCPAPHTML ( ) ) ;
ui - > oxiInfo - > setHtml ( GenerateOxiHTML ( ) ) ;
}
Welcome : : ~ Welcome ( )
{
delete ui ;
}
void Welcome : : on_dailyButton_clicked ( )
{
mainwin - > JumpDaily ( ) ;
}
void Welcome : : on_overviewButton_clicked ( )
{
mainwin - > JumpOverview ( ) ;
}
void Welcome : : on_statisticsButton_clicked ( )
{
mainwin - > JumpStatistics ( ) ;
}
void Welcome : : on_oximetryButton_clicked ( )
{
mainwin - > JumpOxiWizard ( ) ;
}
void Welcome : : on_importButton_clicked ( )
{
mainwin - > JumpImport ( ) ;
}
2014-07-25 07:53:48 +00:00
2018-04-17 04:11:28 +00:00
extern EventDataType calcAHI ( QDate start , QDate end ) ;
extern EventDataType calcFL ( QDate start , QDate end ) ;
2018-05-07 15:11:02 +00:00
QString Welcome : : GenerateCPAPHTML ( )
2014-07-25 07:53:48 +00:00
{
2018-05-07 15:11:02 +00:00
auto cpap_machines = p_profile - > GetMachines ( MT_CPAP ) ;
auto oximeters = p_profile - > GetMachines ( MT_OXIMETER ) ;
2014-07-25 07:53:48 +00:00
QList < Machine * > mach ;
mach . append ( cpap_machines ) ;
mach . append ( oximeters ) ;
bool havecpapdata = false ;
bool haveoximeterdata = false ;
2018-05-07 15:11:02 +00:00
for ( auto & mach : cpap_machines ) {
int daysize = mach - > day . size ( ) ;
2014-07-25 07:53:48 +00:00
if ( daysize > 0 ) {
havecpapdata = true ;
break ;
}
}
2018-05-07 15:11:02 +00:00
for ( auto & mach : oximeters ) {
int daysize = mach - > day . size ( ) ;
2014-07-25 07:53:48 +00:00
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; } "
" </style> "
" </head> "
2018-05-07 15:11:02 +00:00
" <body leftmargin=5 topmargin=5 rightmargin=5 bottommargin=5 valign=center align=center> " ;
2018-04-17 23:47:04 +00:00
Machine * cpap = nullptr ;
2014-07-25 07:53:48 +00:00
if ( ! havecpapdata & & ! haveoximeterdata ) {
2018-05-07 15:11:02 +00:00
html + = " <p> " + tr ( " It might be a good idea to check preferences first,</br>as there are some options that affect import. " ) + " </p> "
" <p> " + tr ( " First import can take a few minutes. " ) + " </p> " ;
2014-07-25 07:53:48 +00:00
} else {
2014-08-20 17:17:13 +00:00
QDate date = p_profile - > LastDay ( MT_CPAP ) ;
Day * day = p_profile - > GetDay ( date , MT_CPAP ) ;
if ( havecpapdata & & day ) {
2018-03-28 07:10:52 +00:00
cpap = day - > machine ( MT_CPAP ) ;
}
if ( day & & ( cpap ! = nullptr ) ) {
2018-05-07 15:11:02 +00:00
QString cpapimage = cpap - > getPixmapPath ( ) ;
2014-08-20 17:17:13 +00:00
2018-05-07 15:11:02 +00:00
ui - > cpapIcon - > setPixmap ( QPixmap ( cpapimage ) ) ;
2018-04-17 03:02:24 +00:00
2018-05-07 15:11:02 +00:00
html + = " <b> " + tr ( " The last time you used your %1... " ) . arg ( cpap - > brand ( ) + " " + cpap - > series ( ) + " " + cpap - > model ( ) ) + " </b><br/> " ;
2014-07-25 07:53:48 +00:00
int daysto = date . daysTo ( QDate : : currentDate ( ) ) ;
QString daystring ;
2018-05-07 15:11:02 +00:00
if ( daysto = = 1 ) daystring + = tr ( " last night " ) ;
else if ( daysto = = 2 ) daystring + = tr ( " yesterday " ) ;
else daystring + = tr ( " %2 days ago " ) . arg ( date . daysTo ( QDate : : currentDate ( ) ) ) ;
2014-07-25 07:53:48 +00:00
2018-05-07 15:11:02 +00:00
html + = tr ( " was %1 (on %2) " ) . arg ( daystring ) . arg ( date . toString ( Qt : : SystemLocaleLongDate ) ) + " <br/> " ;
2014-07-25 07:53:48 +00:00
EventDataType hours = day - > hours ( ) ;
2018-04-17 03:02:24 +00:00
html + = " <br/> " ;
2014-07-25 07:53:48 +00:00
int seconds = int ( hours * 3600.0 ) % 60 ;
int minutes = int ( hours * 60 ) % 60 ;
int hour = hours ;
2018-05-07 15:11:02 +00:00
QString timestr = tr ( " %1 hours, %2 minutes and %3 seconds " ) . arg ( hour ) . arg ( minutes ) . arg ( seconds ) ;
2014-07-25 07:53:48 +00:00
2018-04-17 04:11:28 +00:00
const EventDataType compliance_min = 4.0 ;
2018-05-07 15:11:02 +00:00
if ( hours > compliance_min ) html + = tr ( " Your machine was on for %1. " ) . arg ( timestr ) + " <br/> " ;
else html + = tr ( " <font color = red>You only had the mask on for %1.</font> " ) . arg ( timestr ) + " <br/> " ;
2014-07-25 07:53:48 +00:00
2018-04-17 03:02:24 +00:00
2018-04-17 04:11:28 +00:00
int averagedays = 7 ; // how many days to look back
2018-04-17 03:02:24 +00:00
2018-04-17 04:11:28 +00:00
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 ) ;
2018-05-07 15:11:02 +00:00
const QString under = tr ( " under " ) ;
const QString over = tr ( " over " ) ;
const QString close = tr ( " reasonably close to " ) ;
const QString equal = tr ( " equal to " ) ;
2018-04-17 04:11:28 +00:00
QString comp ;
if ( ( ahi < ahidays ) & & ( ( ahidays - ahi ) > = 0.1 ) ) {
comp = under ;
} else if ( ( ahi > ahidays ) & & ( ( ahi - ahidays ) > = 0.1 ) ) {
comp = over ;
2018-04-17 23:47:04 +00:00
} else if ( ( fabs ( ahi > ahidays ) > = 0.01 ) ) {
comp = close ;
2018-04-17 04:11:28 +00:00
} else {
2018-04-17 23:47:04 +00:00
comp = equal ;
2018-04-17 04:11:28 +00:00
}
2018-04-17 03:02:24 +00:00
2018-05-07 15:11:02 +00:00
html + = 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 ) ;
2018-04-17 04:11:28 +00:00
html + = " <br/> " ;
2014-07-25 07:53:48 +00:00
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 ) ;
2018-05-07 15:11:02 +00:00
html + = tr ( " Your CPAP machine blasted you with a constant %1%2 of air " ) . arg ( pressure ) . arg ( schema : : channel [ CPAP_Pressure ] . units ( ) ) ;
2014-07-25 07:53:48 +00:00
} else if ( cpapmode = = MODE_APAP ) {
EventDataType pressure = day - > percentile ( CPAP_Pressure , perc / 100.0 ) ;
2018-05-07 15:11:02 +00:00
html + = tr ( " Your pressure was under %1%2 for %3% of the time. " ) . arg ( pressure ) . arg ( schema : : channel [ CPAP_Pressure ] . units ( ) ) . arg ( perc ) ;
2014-07-25 07:53:48 +00:00
} else if ( cpapmode = = MODE_BILEVEL_FIXED ) {
EventDataType ipap = day - > settings_max ( CPAP_IPAP ) ;
EventDataType epap = day - > settings_min ( CPAP_EPAP ) ;
2018-05-07 15:11:02 +00:00
html + = tr ( " Your machine blasted you with a constant %1-%2 %3 of air. " ) . arg ( epap ) . arg ( ipap ) . arg ( schema : : channel [ CPAP_Pressure ] . units ( ) ) ;
2014-07-25 07:53:48 +00:00
} 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 ) ;
2018-05-07 15:11:02 +00:00
html + = 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 ) ;
2014-07-28 13:56:29 +00:00
} else if ( cpapmode = = MODE_ASV ) {
EventDataType ipap = day - > percentile ( CPAP_IPAP , perc / 100.0 ) ;
EventDataType epap = qRound ( day - > settings_wavg ( CPAP_EPAP ) ) ;
2018-05-07 15:11:02 +00:00
html + = tr ( " Your EPAP pressure fixed at %1%2. " ) . arg ( epap ) . arg ( schema : : channel [ CPAP_EPAP ] . units ( ) ) + " <br/> " ;
html + = tr ( " Your IPAP pressure was under %1%2 for %3% of the time. " ) . arg ( ipap ) . arg ( schema : : channel [ CPAP_IPAP ] . units ( ) ) . arg ( perc ) ;
2014-07-28 13:56:29 +00:00
} 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 ) ;
2018-05-07 15:11:02 +00:00
html + = 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 + = tr ( " Your IPAP pressure was under %1%2 for %3% of the time. " ) . arg ( ipap ) . arg ( schema : : channel [ CPAP_IPAP ] . units ( ) ) . arg ( perc ) ;
2014-07-25 07:53:48 +00:00
}
2018-04-17 04:11:28 +00:00
html + = " <br/> " ;
//EventDataType lat = day->timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline())/ 60.0;
//EventDataType leaks = 1.0/hours * lat;
2014-07-25 07:53:48 +00:00
2018-04-17 04:11:28 +00:00
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 ;
2018-04-17 23:47:04 +00:00
} else if ( ( fabs ( ahi > ahidays ) > = 0.01 ) ) {
comp = close ;
2018-04-17 04:11:28 +00:00
} else {
2018-04-17 23:47:04 +00:00
comp = equal ;
2018-04-17 04:11:28 +00:00
}
2018-05-07 15:11:02 +00:00
html + = 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 ) ;
2018-04-17 04:11:28 +00:00
html + = " <br/> " ;
2014-07-25 07:53:48 +00:00
} else {
2018-05-07 15:11:02 +00:00
html + = " <p> " + tr ( " No CPAP data has been imported yet. " ) + " </p> " ;
2014-07-25 07:53:48 +00:00
}
2018-04-17 23:47:04 +00:00
}
2018-05-07 15:11:02 +00:00
/* html += QString("<div align=center><table class=curved cellpadding=3 width=75%>")+
2018-04-17 23:47:04 +00:00
" <tr> "
2018-04-25 04:46:48 +00:00
" <td align=center colspan=2><b> " + QObject : : tr ( " Very Important Warning For ResMed S9 Users " ) + " </b></font></td></tr> "
" <tr><td align=center> " +
" <p> " + QObject : : tr ( " ALWAYS <font color=red><b>write protect</b></font> CPAP SDCards before inserting them into your computer. " ) + " </p><p> " +
+ " <span title= \" " + QObject : : tr ( " Mac OSX and Win8.1 " ) + " \" onmouseover='ChangeColor(this, \" #eeeeee \" );' onmouseout='ChangeColor(this, \" #ffffff \" );'> " +
QObject : : tr ( " <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. " ) + " " +
QObject : : tr ( " As a second line of protection, make sure to <font color=red><b>unmount</b></font> the data card properly before removing it! " ) + " </p> "
2018-04-17 23:47:04 +00:00
" </td> "
2018-04-25 04:46:48 +00:00
" <td><img src= \" qrc:/icons/sdcard-lock.png \" width=64px></td> "
2018-04-17 23:47:04 +00:00
" </tr> "
" </table> "
2018-05-07 15:11:02 +00:00
" </td></tr></table></div> " ; */
// } else {
// }
html + = " </body></html> " ;
return html ;
}
QString Welcome : : GenerateOxiHTML ( )
{
auto oximeters = p_profile - > GetMachines ( MT_OXIMETER ) ;
bool haveoximeterdata = false ;
for ( auto & mach : oximeters ) {
int daysize = mach - > 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; } "
" </style> "
" </head> "
" <body leftmargin=5 topmargin=5 rightmargin=5 bottommargin=5 valign=center align=center> " ;
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> " ;
ui - > oxiIcon - > setVisible ( true ) ;
ui - > oxiInfo - > setVisible ( true ) ;
2018-04-17 23:47:04 +00:00
} else {
2018-05-07 15:11:02 +00:00
html + = " <p> " + QObject : : tr ( " No oximetery data has been imported yet. " ) + " </p> " ;
ui - > oxiIcon - > setVisible ( false ) ;
ui - > oxiInfo - > setVisible ( false ) ;
2018-04-17 23:47:04 +00:00
}
2018-05-07 15:11:02 +00:00
html + = " </body></html> " ;
2014-07-25 07:53:48 +00:00
return html ;
}