2014-04-09 21:01:57 +00:00
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* vim : set ts = 8 sts = 4 et sw = 4 tw = 99 :
*
2014-04-25 05:28:10 +00:00
* Statistics Report Generator
2014-04-09 21:01:57 +00:00
*
* Copyright ( c ) 2011 Mark Watkins < jedimark @ users . sourceforge . 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 . */
2013-09-14 23:32:14 +00:00
# include <QApplication>
# include <cmath>
# include "mainwindow.h"
2014-04-25 05:28:10 +00:00
# include "statistics.h"
2013-09-14 23:32:14 +00:00
extern MainWindow * mainwin ;
2014-05-06 09:11:31 +00:00
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);
}
2013-09-14 23:32:14 +00:00
2014-04-25 05:28:10 +00:00
Statistics : : Statistics ( QObject * parent ) :
2013-09-14 23:32:14 +00:00
QObject ( parent )
{
2014-05-06 17:39:05 +00:00
rows . push_back ( StatisticsRow ( tr ( " CPAP Statistics " ) , SC_HEADING , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " " , SC_DAYS , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " " , SC_COLUMNHEADERS , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( tr ( " CPAP Usage " ) , SC_SUBHEADING , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( tr ( " Average Hours per Night " ) , SC_HOURS , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( tr ( " Compliancy " ) , SC_COMPLIANCE , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( tr ( " Therapy Efficiacy " ) , SC_SUBHEADING , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " AHI " , SC_AHI , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Obstructive " , SC_CPH , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Hypopnea " , SC_CPH , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " ClearAirway " , SC_CPH , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " FlowLimit " , SC_CPH , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " RERA " , SC_CPH , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " CSR " , SC_SPH , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( tr ( " Leak Statistics " ) , SC_SUBHEADING , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Leak " , SC_WAVG , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Leak " , SC_90P , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( tr ( " Pressure Statistics " ) , SC_SUBHEADING , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Pressure " , SC_WAVG , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Pressure " , SC_MIN , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Pressure " , SC_MAX , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Pressure " , SC_90P , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " EPAP " , SC_WAVG , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " EPAP " , SC_MIN , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " EPAP " , SC_MAX , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " IPAP " , SC_WAVG , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " IPAP " , SC_90P , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " IPAP " , SC_MIN , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " IPAP " , SC_MAX , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( tr ( " Oximeter Statistics " ) , SC_HEADING , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( " " , SC_DAYS , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( " " , SC_COLUMNHEADERS , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( tr ( " Blood Oxygen Saturation " ) , SC_SUBHEADING , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " SPO2 " , SC_WAVG , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( " SPO2 " , SC_MIN , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( " SPO2Drop " , SC_CPH , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( " SPO2Drop " , SC_SPH , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( tr ( " Pulse Rate " ) , SC_SUBHEADING , MT_CPAP ) ) ;
rows . push_back ( StatisticsRow ( " Pulse " , SC_WAVG , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( " Pulse " , SC_MIN , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( " Pulse " , SC_MAX , MT_OXIMETER ) ) ;
rows . push_back ( StatisticsRow ( " PulseChange " , SC_CPH , MT_OXIMETER ) ) ;
2014-05-06 09:11:31 +00:00
// These are for formatting the headers for the first column
2014-05-06 17:39:05 +00:00
calcnames [ SC_UNDEFINED ] = " " ;
calcnames [ SC_MEDIAN ] = tr ( " %1 Median " ) ;
calcnames [ SC_AVG ] = tr ( " Average %1 " ) ;
calcnames [ SC_WAVG ] = tr ( " Average %1 " ) ;
calcnames [ SC_90P ] = tr ( " 90% %1 " ) ; // this gets converted to whatever the upper percentile is set to
calcnames [ SC_MIN ] = tr ( " Min %1 " ) ;
calcnames [ SC_MAX ] = tr ( " Max %1 " ) ;
calcnames [ SC_CPH ] = tr ( " %1 Index " ) ;
calcnames [ SC_SPH ] = tr ( " % of time in %1 " ) ;
machinenames [ MT_UNKNOWN ] = STR_TR_Unknown ;
machinenames [ MT_CPAP ] = STR_TR_CPAP ;
machinenames [ MT_OXIMETER ] = STR_TR_Oximetry ;
machinenames [ MT_SLEEPSTAGE ] = STR_TR_SleepStage ;
2014-05-06 09:11:31 +00:00
// { MT_JOURNAL, STR_TR_Journal },
// { MT_POSITION, STR_TR_Position },
2013-09-14 23:32:14 +00:00
}
QString htmlHeader ( )
{
2014-04-17 05:52:25 +00:00
QString address = PROFILE . user - > address ( ) ;
address . replace ( " \n " , " <br/> " ) ;
2013-09-14 23:32:14 +00:00
// "a:link,a:visited { color: '#000020'; text-decoration: none; font-weight: bold;}"
2014-04-17 05:52:25 +00:00
// "a:hover { background-color: inherit; color: red; text-decoration:none; font-weight: bold; }"
2014-05-06 09:33:06 +00:00
QString userinfo ;
if ( ! PROFILE . user - > firstName ( ) . isEmpty ( ) ) {
userinfo = QString ( QObject : : tr ( " Name: %1, %2 " ) ) . arg ( PROFILE . user - > lastName ( ) ) . arg ( PROFILE . user - > firstName ( ) ) + " <br/> " ;
if ( ! PROFILE . user - > DOB ( ) . isNull ( ) ) {
userinfo + = QString ( QObject : : tr ( " DOB: %1 " ) ) . arg ( PROFILE . user - > DOB ( ) . toString ( ) ) + " <br/> " ;
}
2014-05-06 12:37:00 +00:00
if ( ! PROFILE . user - > phone ( ) . isEmpty ( ) ) {
userinfo + = QString ( QObject : : tr ( " Phone: %1 " ) ) . arg ( PROFILE . user - > phone ( ) ) + " <br/> " ;
}
if ( ! PROFILE . user - > email ( ) . isEmpty ( ) ) {
userinfo + = QString ( QObject : : tr ( " Email: %1 " ) ) . arg ( PROFILE . user - > email ( ) ) + " <br/><br/> " ;
}
if ( ! PROFILE . user - > address ( ) . isEmpty ( ) ) {
userinfo + = QObject : : tr ( " Address: " ) + " <br/> " + address ;
}
2014-05-06 09:33:06 +00:00
}
2013-09-14 23:32:14 +00:00
return QString ( " <html><head> "
2014-04-17 05:52:25 +00:00
" </head> "
" <style type='text/css'> "
" <!--h1,p,a,td,body { font-family: 'FreeSans', 'Sans Serif' } --/> "
" p,a,td,body { font-size: 14px } "
" </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=0 rightmargin=0> "
" <div align=center><table cellpadding=3 cellspacing=0 border=0 width=100%> "
2014-05-06 09:33:06 +00:00
" <td> " + userinfo + " </td> "
2014-04-17 05:52:25 +00:00
" <td align='right'> "
" <h1> " + STR_TR_SleepyHead + " </h1><br/> "
" <font size='+3'> " + QObject : : tr ( " Usage Statistics " ) + " </font> "
" </td> "
" <td align='right' width=170px><img src='qrc:/icons/bob-v3.0.png' height=140px><br/> "
" </td></tr></table> "
" </div> "
" <hr/> " ) ;
2013-09-14 23:32:14 +00:00
}
QString htmlFooter ( )
{
2014-04-17 05:52:25 +00:00
return " <hr/><div align=center><font size='-1'><i> " +
QString ( QObject : : tr ( " This report was generated by a pre-release version of SleepyHead (%1), <b>and has not been approved in any way for compliance or medical diagnostic purposes</b>. " ) ) . arg (
FullVersionString ) + " <br/><br/> " +
QObject : : tr ( " SleepyHead is free open-source software available from http://sourceforge.net/projects/SleepyHead " )
+
" </i></i></div> "
" </body></html> " ;
2013-09-14 23:32:14 +00:00
}
EventDataType calcAHI ( QDate start , QDate end )
{
2014-04-17 05:52:25 +00:00
EventDataType val = ( p_profile - > calcCount ( CPAP_Obstructive , MT_CPAP , start , end )
+ p_profile - > calcCount ( CPAP_Hypopnea , MT_CPAP , start , end )
+ p_profile - > calcCount ( CPAP_ClearAirway , MT_CPAP , start , end )
+ p_profile - > calcCount ( CPAP_Apnea , MT_CPAP , start , end ) ) ;
if ( PROFILE . general - > calculateRDI ( ) ) {
val + = p_profile - > calcCount ( CPAP_RERA , MT_CPAP , start , end ) ;
}
EventDataType hours = p_profile - > calcHours ( MT_CPAP , start , end ) ;
if ( hours > 0 ) {
val / = hours ;
} else {
val = 0 ;
}
2013-09-14 23:32:14 +00:00
return val ;
}
EventDataType calcFL ( QDate start , QDate end )
{
2014-04-17 05:52:25 +00:00
EventDataType val = ( p_profile - > calcCount ( CPAP_FlowLimit , MT_CPAP , start , end ) ) ;
EventDataType hours = p_profile - > calcHours ( MT_CPAP , start , end ) ;
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
if ( hours > 0 ) {
val / = hours ;
} else {
val = 0 ;
}
2013-09-14 23:32:14 +00:00
return val ;
}
2014-04-17 05:52:25 +00:00
struct RXChange {
2014-04-23 13:19:56 +00:00
RXChange ( ) { highlight = 0 ; machine = nullptr ; }
2014-04-17 05:52:25 +00:00
RXChange ( const RXChange & copy ) {
first = copy . first ;
last = copy . last ;
days = copy . days ;
ahi = copy . ahi ;
fl = copy . fl ;
mode = copy . mode ;
min = copy . min ;
max = copy . max ;
ps = copy . ps ;
pshi = copy . pshi ;
maxipap = copy . maxipap ;
machine = copy . machine ;
per1 = copy . per1 ;
per2 = copy . per2 ;
highlight = copy . highlight ;
weighted = copy . weighted ;
prelief = copy . prelief ;
prelset = copy . prelset ;
2013-09-14 23:32:14 +00:00
}
QDate first ;
QDate last ;
int days ;
EventDataType ahi ;
EventDataType fl ;
CPAPMode mode ;
EventDataType min ;
EventDataType max ;
2013-10-16 09:36:05 +00:00
EventDataType ps ;
2013-11-19 13:06:17 +00:00
EventDataType pshi ;
EventDataType maxipap ;
2013-09-14 23:32:14 +00:00
EventDataType per1 ;
EventDataType per2 ;
EventDataType weighted ;
PRTypes prelief ;
2014-04-17 05:52:25 +00:00
Machine * machine ;
2013-09-14 23:32:14 +00:00
short prelset ;
short highlight ;
} ;
2013-11-19 13:06:17 +00:00
enum RXSortMode { RX_first , RX_last , RX_days , RX_ahi , RX_mode , RX_min , RX_max , RX_ps , RX_pshi , RX_maxipap , RX_per1 , RX_per2 , RX_weighted } ;
2014-04-17 05:52:25 +00:00
RXSortMode RXsort = RX_first ;
bool RXorder = false ;
bool operator < ( const RXChange & c1 , const RXChange & c2 )
{
const RXChange * comp1 = & c1 ;
const RXChange * comp2 = & c2 ;
2013-09-14 23:32:14 +00:00
if ( RXorder ) {
switch ( RXsort ) {
2014-04-17 05:52:25 +00:00
case RX_ahi :
return comp1 - > ahi < comp2 - > ahi ;
case RX_days :
return comp1 - > days < comp2 - > days ;
case RX_first :
return comp1 - > first < comp2 - > first ;
case RX_last :
return comp1 - > last < comp2 - > last ;
case RX_mode :
return comp1 - > mode < comp2 - > mode ;
case RX_min :
return comp1 - > min < comp2 - > min ;
case RX_max :
return comp1 - > max < comp2 - > max ;
case RX_ps :
return comp1 - > ps < comp2 - > ps ;
case RX_pshi :
return comp1 - > pshi < comp2 - > pshi ;
case RX_maxipap :
return comp1 - > maxipap < comp2 - > maxipap ;
case RX_per1 :
return comp1 - > per1 < comp2 - > per1 ;
case RX_per2 :
return comp1 - > per2 < comp2 - > per2 ;
case RX_weighted :
return comp1 - > weighted < comp2 - > weighted ;
2013-09-14 23:32:14 +00:00
} ;
} else {
switch ( RXsort ) {
2014-04-17 05:52:25 +00:00
case RX_ahi :
return comp1 - > ahi > comp2 - > ahi ;
case RX_days :
return comp1 - > days > comp2 - > days ;
case RX_first :
return comp1 - > first > comp2 - > first ;
case RX_last :
return comp1 - > last > comp2 - > last ;
case RX_mode :
return comp1 - > mode > comp2 - > mode ;
case RX_min :
return comp1 - > min > comp2 - > min ;
case RX_max :
return comp1 - > max > comp2 - > max ;
case RX_ps :
return comp1 - > ps > comp2 - > ps ;
case RX_pshi :
return comp1 - > pshi > comp2 - > pshi ;
case RX_maxipap :
return comp1 - > maxipap > comp2 - > maxipap ;
case RX_per1 :
return comp1 - > per1 > comp2 - > per1 ;
case RX_per2 :
return comp1 - > per2 > comp2 - > per2 ;
case RX_weighted :
return comp1 - > weighted > comp2 - > weighted ;
2013-09-14 23:32:14 +00:00
} ;
}
2014-04-17 05:52:25 +00:00
2013-09-14 23:32:14 +00:00
return true ;
}
2014-04-17 05:52:25 +00:00
bool RXSort ( const RXChange * comp1 , const RXChange * comp2 )
{
2013-09-14 23:32:14 +00:00
if ( RXorder ) {
switch ( RXsort ) {
2014-04-17 05:52:25 +00:00
case RX_ahi :
return comp1 - > ahi < comp2 - > ahi ;
case RX_days :
return comp1 - > days < comp2 - > days ;
case RX_first :
return comp1 - > first < comp2 - > first ;
case RX_last :
return comp1 - > last < comp2 - > last ;
case RX_mode :
return comp1 - > mode < comp2 - > mode ;
case RX_min :
return comp1 - > min < comp2 - > min ;
case RX_max :
return comp1 - > max < comp2 - > max ;
case RX_ps :
return comp1 - > ps < comp2 - > ps ;
case RX_pshi :
return comp1 - > pshi < comp2 - > pshi ;
case RX_maxipap :
return comp1 - > maxipap < comp2 - > maxipap ;
case RX_per1 :
return comp1 - > per1 < comp2 - > per1 ;
case RX_per2 :
return comp1 - > per2 < comp2 - > per2 ;
case RX_weighted :
return comp1 - > weighted < comp2 - > weighted ;
2013-09-14 23:32:14 +00:00
} ;
} else {
switch ( RXsort ) {
2014-04-17 05:52:25 +00:00
case RX_ahi :
return comp1 - > ahi > comp2 - > ahi ;
case RX_days :
return comp1 - > days > comp2 - > days ;
case RX_first :
return comp1 - > first > comp2 - > first ;
case RX_last :
return comp1 - > last > comp2 - > last ;
case RX_mode :
return comp1 - > mode > comp2 - > mode ;
case RX_min :
return comp1 - > min > comp2 - > min ;
case RX_max :
return comp1 - > max > comp2 - > max ;
case RX_ps :
return comp1 - > ps > comp2 - > ps ;
case RX_pshi :
return comp1 - > pshi > comp2 - > pshi ;
case RX_maxipap :
return comp1 - > maxipap > comp2 - > maxipap ;
case RX_per1 :
return comp1 - > per1 > comp2 - > per1 ;
case RX_per2 :
return comp1 - > per2 > comp2 - > per2 ;
case RX_weighted :
return comp1 - > weighted > comp2 - > weighted ;
2013-09-14 23:32:14 +00:00
} ;
}
2014-04-17 05:52:25 +00:00
2013-09-14 23:32:14 +00:00
return true ;
}
struct UsageData {
2014-04-17 05:52:25 +00:00
UsageData ( ) { ahi = 0 ; hours = 0 ; }
UsageData ( QDate d , EventDataType v , EventDataType h ) { date = d ; ahi = v ; hours = h ; }
UsageData ( const UsageData & copy ) { date = copy . date ; ahi = copy . ahi ; hours = copy . hours ; }
2013-09-14 23:32:14 +00:00
QDate date ;
EventDataType ahi ;
EventDataType hours ;
} ;
2014-04-17 05:52:25 +00:00
bool operator < ( const UsageData & c1 , const UsageData & c2 )
2013-09-14 23:32:14 +00:00
{
2014-04-17 05:52:25 +00:00
if ( c1 . ahi < c2 . ahi ) {
2013-09-14 23:32:14 +00:00
return true ;
2014-04-17 05:52:25 +00:00
}
if ( ( c1 . ahi = = c2 . ahi ) & & ( c1 . date > c2 . date ) ) { return true ; }
2013-09-14 23:32:14 +00:00
return false ;
//return c1.value < c2.value;
}
2014-05-06 11:54:41 +00:00
struct Period {
Period ( ) {
}
Period ( QDate start , QDate end , QString header ) {
this - > start = start ;
this - > end = end ;
this - > header = header ;
}
Period ( const Period & copy ) {
start = copy . start ;
end = copy . end ;
header = copy . header ;
}
QDate start ;
QDate end ;
QString header ;
} ;
2014-04-25 05:28:10 +00:00
QString Statistics : : GenerateHTML ( )
2013-09-14 23:32:14 +00:00
{
2014-05-06 09:11:31 +00:00
2014-05-06 09:33:06 +00:00
QString heading_color = " #ffffff " ;
2014-05-06 09:11:31 +00:00
QString subheading_color = " #efefef " ;
2014-04-17 05:52:25 +00:00
QString html = htmlHeader ( ) ;
2014-05-06 09:11:31 +00:00
// Find first and last days with valid CPAP data
2014-04-17 05:52:25 +00:00
QDate lastcpap = p_profile - > LastGoodDay ( MT_CPAP ) ;
QDate firstcpap = p_profile - > FirstGoodDay ( MT_CPAP ) ;
2014-05-06 09:11:31 +00:00
QDate cpapweek = lastcpap . addDays ( - 6 ) ;
QDate cpapmonth = lastcpap . addDays ( - 29 ) ;
2014-04-17 05:52:25 +00:00
QDate cpap6month = lastcpap . addMonths ( - 6 ) ;
2014-05-06 09:11:31 +00:00
QDate cpapyear = lastcpap . addMonths ( - 12 ) ;
2014-04-17 05:52:25 +00:00
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 ) ;
2013-09-14 23:32:14 +00:00
QList < Machine * > mach ;
mach . append ( cpap_machines ) ;
mach . append ( oximeters ) ;
2014-04-17 05:52:25 +00:00
if ( mach . size ( ) = = 0 ) {
html + = " <table cellpadding=2 cellspacing=0 border=0 width=100% height=60%> " ;
html + = " <tr><td align=center><h1> " + tr ( " Please Import Some Data " ) + " </h1><i> " +
tr ( " SleepyHead is pretty much useless without it. " ) + " </i><br/><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></td></tr></table> " ;
html + = htmlFooter ( ) ;
2013-09-14 23:32:14 +00:00
return html ;
}
2014-04-17 05:52:25 +00:00
int cpapdays = PROFILE . countDays ( MT_CPAP , firstcpap , lastcpap ) ;
CPAPMode cpapmode = ( CPAPMode ) ( int ) p_profile - > calcSettingsMax ( CPAP_Mode , MT_CPAP , firstcpap ,
lastcpap ) ;
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
float percentile = PROFILE . general - > prefCalcPercentile ( ) / 100.0 ;
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
// int mididx=PROFILE.general->prefCalcMiddle();
// SummaryType ST_mid;
// if (mididx==0) ST_mid=ST_PERC;
// if (mididx==1) ST_mid=ST_WAVG;
// if (mididx==2) ST_mid=ST_AVG;
2013-09-14 23:32:14 +00:00
QString ahitxt ;
2014-04-17 05:52:25 +00:00
2013-09-14 23:32:14 +00:00
if ( PROFILE . general - > calculateRDI ( ) ) {
2014-04-17 05:52:25 +00:00
ahitxt = STR_TR_RDI ;
2013-09-14 23:32:14 +00:00
} else {
2014-04-17 05:52:25 +00:00
ahitxt = STR_TR_AHI ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
int decimals = 2 ;
html + = " <div align=center> " ;
html + = QString ( " <table cellpadding=2 cellspacing=0 border=1 width=90%> " ) ;
2014-05-06 11:54:41 +00:00
int number_periods = 0 ;
if ( p_profile - > general - > statReportMode ( ) = = 1 ) {
number_periods = 12 ;
}
QDate last = lastcpap , first = lastcpap ;
2013-09-14 23:32:14 +00:00
2014-05-06 11:54:41 +00:00
QList < Period > periods ;
2013-09-14 23:32:14 +00:00
2014-05-06 09:11:31 +00:00
bool skipsection = false ; ;
for ( auto i = rows . begin ( ) ; i ! = rows . end ( ) ; + + i ) {
StatisticsRow & row = ( * i ) ;
QString name ;
2013-09-14 23:32:14 +00:00
2014-05-06 09:11:31 +00:00
if ( row . calc = = SC_HEADING ) { // All sections begin with a heading
last = p_profile - > LastGoodDay ( row . type ) ;
first = p_profile - > FirstGoodDay ( row . type ) ;
2013-09-14 23:32:14 +00:00
2014-05-06 11:54:41 +00:00
periods . clear ( ) ;
if ( number_periods = = 0 ) {
periods . push_back ( Period ( last , last , tr ( " Most Recent " ) ) ) ;
periods . push_back ( Period ( qMax ( last . addDays ( - 6 ) , first ) , last , tr ( " Last Week " ) ) ) ;
periods . push_back ( Period ( qMax ( last . addDays ( - 29 ) , first ) , last , tr ( " Last 30 Days " ) ) ) ;
periods . push_back ( Period ( qMax ( last . addMonths ( - 6 ) , first ) , last , tr ( " Last 6 Months " ) ) ) ;
periods . push_back ( Period ( qMax ( last . addMonths ( - 12 ) , first ) , last , tr ( " Last Year " ) ) ) ;
} else {
QDate l = last , s = last ;
periods . push_back ( Period ( last , last , tr ( " Last Session " ) ) ) ;
2013-09-14 23:32:14 +00:00
2014-05-06 11:54:41 +00:00
bool done = false ;
for ( int j = 0 ; j < number_periods ; j + + ) {
s = QDate ( l . year ( ) , l . month ( ) , 1 ) ;
if ( s < first ) {
done = true ;
s = first ;
}
if ( p_profile - > countDays ( row . type , s , l ) > 0 ) {
periods . push_back ( Period ( s , l , s . toString ( " MMMM " ) ) ) ;
}
l = s . addDays ( - 1 ) ;
if ( done | | ( l < first ) ) break ;
}
}
2014-04-17 05:52:25 +00:00
2014-05-06 09:11:31 +00:00
int days = PROFILE . countDays ( row . type , first , last ) ;
skipsection = ( days = = 0 ) ;
if ( days > 0 ) {
2014-05-06 11:54:41 +00:00
html + = QString ( " <tr bgcolor='%1'><td colspan=%2 align=center><font size=+3>%3</font></td></tr> \n " ) .
arg ( heading_color ) . arg ( periods . size ( ) + 1 ) . arg ( row . src ) ;
2014-05-06 09:11:31 +00:00
}
continue ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
2014-05-06 09:11:31 +00:00
// Bypass this entire section if no data is present
if ( skipsection ) continue ;
2013-09-14 23:32:14 +00:00
2014-05-06 09:11:31 +00:00
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 ) {
2014-05-06 11:54:41 +00:00
html + = QString ( " <tr><td><b>%1</b></td> " ) . arg ( tr ( " Details " ) ) ;
for ( int j = 0 ; j < periods . size ( ) ; j + + ) {
2014-05-06 12:37:00 +00:00
html + = QString ( " <td onmouseover='ChangeColor(this, \" #eeeeee \" );' onmouseout='ChangeColor(this, \" #ffffff \" );' onclick='alert( \" overview=%1,%2 \" );'><b>%3</b></td> " ) . arg ( periods . at ( j ) . start . toString ( Qt : : ISODate ) ) . arg ( periods . at ( j ) . end . toString ( Qt : : ISODate ) ) . arg ( periods . at ( j ) . header ) ;
2014-05-06 11:54:41 +00:00
}
html + = " </tr> \n " ;
2014-05-06 09:11:31 +00:00
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 ( value = = 0 ) {
2014-05-06 11:54:41 +00:00
html + = QString ( " <tr><td colspan=%1 align=center>%2</td></tr> \n " ) . arg ( periods . size ( ) + 1 ) .
arg ( QString ( tr ( " No %1 data available. " ) ) . arg ( machine ) ) ;
2014-05-06 09:11:31 +00:00
} else if ( value = = 1 ) {
2014-05-06 11:54:41 +00:00
html + = QString ( " <tr><td colspan=%1 align=center>%2</td></tr> \n " ) . arg ( periods . size ( ) + 1 ) .
arg ( QString ( " %1 day of %2 Data on %3 " )
2014-05-06 09:11:31 +00:00
. arg ( value )
. arg ( machine )
2014-05-06 11:54:41 +00:00
. arg ( last . toString ( ) ) ) ;
2014-05-06 09:11:31 +00:00
} else {
2014-05-06 11:54:41 +00:00
html + = QString ( " <tr><td colspan=%1 align=center>%2</td></tr> \n " ) . arg ( periods . size ( ) + 1 ) .
arg ( QString ( " %1 days of %2 Data, between %3 and %4 " )
2014-05-06 09:11:31 +00:00
. arg ( value )
. arg ( machine )
. arg ( first . toString ( ) )
2014-05-06 11:54:41 +00:00
. arg ( last . toString ( ) ) ) ;
2014-05-06 09:11:31 +00:00
}
continue ;
} else if ( row . calc = = SC_SUBHEADING ) { // subheading..
2014-05-06 11:54:41 +00:00
html + = QString ( " <tr bgcolor='%1'><td colspan=%2 align=center><b>%3</b></td></tr> \n " ) .
arg ( subheading_color ) . arg ( periods . size ( ) + 1 ) . arg ( row . src ) ;
2014-05-06 09:11:31 +00:00
continue ;
} else if ( row . calc = = SC_UNDEFINED ) {
continue ;
} else {
ChannelID id = schema : : channel [ row . src ] . id ( ) ;
if ( ( id = = NoChannel ) | | ( ! PROFILE . hasChannel ( id ) ) ) {
continue ;
}
name = calcnames [ row . calc ] . arg ( row . src ) ;
2013-09-14 23:32:14 +00:00
}
2014-05-06 11:54:41 +00:00
html + = QString ( " <tr><td>%1</td> " ) . arg ( name ) ;
for ( int j = 0 ; j < periods . size ( ) ; j + + ) {
html + = QString ( " <td>%2</td> " )
. arg ( row . value ( periods . at ( j ) . start , periods . at ( j ) . end ) ) ;
}
html + = " </tr> \n " ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
html + = " </table> " ;
html + = " </div> " ;
2013-09-14 23:32:14 +00:00
QList < UsageData > AHI ;
2014-04-17 05:52:25 +00:00
if ( cpapdays > 0 ) {
QDate first , last = lastcpap ;
CPAPMode mode = MODE_UNKNOWN , cmode = MODE_UNKNOWN ;
EventDataType cmin = 0 , cmax = 0 , cps = 0 , cpshi = 0 , cmaxipap = 0 , min = 0 , max = 0 , maxipap = 0 ,
ps = 0 , pshi = 0 ;
2014-04-23 13:19:56 +00:00
Machine * mach = nullptr , * lastmach = nullptr ;
2014-04-17 05:52:25 +00:00
PRTypes lastpr = PR_UNKNOWN , prelief = PR_UNKNOWN ;
short prelset = 0 , lastprelset = - 1 ;
QDate date = lastcpap ;
Day * day ;
bool lastchanged = false ;
2013-09-14 23:32:14 +00:00
QVector < RXChange > rxchange ;
EventDataType hours ;
2014-04-17 05:52:25 +00:00
int compliant = 0 ;
2013-09-14 23:32:14 +00:00
do {
2014-04-17 05:52:25 +00:00
day = PROFILE . GetGoodDay ( date , MT_CPAP ) ;
2013-09-14 23:32:14 +00:00
if ( day ) {
2014-04-17 05:52:25 +00:00
lastchanged = false ;
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
hours = day - > hours ( ) ;
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
if ( hours > PROFILE . cpap - > complianceHours ( ) ) {
2013-09-14 23:32:14 +00:00
compliant + + ;
2014-04-17 05:52:25 +00:00
}
EventDataType ahi = day - > count ( CPAP_Obstructive ) + day - > count ( CPAP_Hypopnea ) + day - > count (
CPAP_Apnea ) + day - > count ( CPAP_ClearAirway ) ;
if ( PROFILE . general - > calculateRDI ( ) ) { ahi + = day - > count ( CPAP_RERA ) ; }
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
ahi / = hours ;
AHI . push_back ( UsageData ( date , ahi , hours ) ) ;
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
prelief = ( PRTypes ) ( int ) round ( day - > settings_wavg ( CPAP_PresReliefType ) ) ;
prelset = round ( day - > settings_wavg ( CPAP_PresReliefSet ) ) ;
mode = ( CPAPMode ) ( int ) round ( day - > settings_wavg ( CPAP_Mode ) ) ;
mach = day - > machine ;
2013-11-19 13:06:17 +00:00
2014-04-17 05:52:25 +00:00
min = max = ps = pshi = maxipap = 0 ;
2013-11-19 13:06:17 +00:00
2014-04-17 05:52:25 +00:00
if ( mode = = MODE_CPAP ) {
min = day - > settings_min ( CPAP_Pressure ) ;
} else if ( mode < MODE_BIPAP ) {
min = day - > settings_min ( CPAP_PressureMin ) ;
max = day - > settings_max ( CPAP_PressureMax ) ;
2013-09-14 23:32:14 +00:00
} else {
2013-11-19 13:06:17 +00:00
// BIPAP or ASV machines
// min & max hold EPAP
if ( day - > settingExists ( CPAP_EPAPLo ) ) {
2014-04-17 05:52:25 +00:00
min = day - > settings_min ( CPAP_EPAPLo ) ;
2013-11-19 13:06:17 +00:00
} else if ( day - > settingExists ( CPAP_EPAP ) ) {
2014-04-17 05:52:25 +00:00
max = min = day - > settings_min ( CPAP_EPAP ) ;
2013-11-19 13:06:17 +00:00
}
2014-04-17 05:52:25 +00:00
2013-11-19 13:06:17 +00:00
if ( day - > settingExists ( CPAP_EPAPHi ) ) {
2014-04-17 05:52:25 +00:00
max = day - > settings_min ( CPAP_EPAPHi ) ;
2013-11-19 13:06:17 +00:00
}
if ( day - > settingExists ( CPAP_PSMin ) ) {
2014-04-17 05:52:25 +00:00
ps = day - > settings_min ( CPAP_PSMin ) ;
2013-11-19 13:06:17 +00:00
} else if ( day - > settingExists ( CPAP_PS ) ) {
2014-04-17 05:52:25 +00:00
pshi = ps = day - > settings_min ( CPAP_PS ) ;
2013-11-19 13:06:17 +00:00
}
2014-04-17 05:52:25 +00:00
2013-11-19 13:06:17 +00:00
if ( day - > settingExists ( CPAP_PSMax ) ) {
2014-04-17 05:52:25 +00:00
pshi = day - > settings_max ( CPAP_PSMax ) ;
2013-11-19 13:06:17 +00:00
}
2014-04-17 05:52:25 +00:00
2013-11-19 13:06:17 +00:00
if ( day - > settingExists ( CPAP_IPAPHi ) ) {
2014-04-17 05:52:25 +00:00
maxipap = day - > settings_max ( CPAP_IPAPHi ) ;
2013-11-19 13:06:17 +00:00
}
2013-09-14 23:32:14 +00:00
}
2013-11-19 13:06:17 +00:00
2014-04-17 05:52:25 +00:00
if ( ( mode ! = cmode ) | | ( min ! = cmin ) | | ( max ! = cmax ) | | ( ps ! = cps ) | | ( pshi ! = cpshi )
| | ( maxipap ! = cmaxipap ) | | ( mach ! = lastmach ) | | ( prelset ! = lastprelset ) ) {
2014-04-23 13:19:56 +00:00
if ( ( cmode ! = MODE_UNKNOWN ) & & ( lastmach ! = nullptr ) ) {
2014-04-17 05:52:25 +00:00
first = date . addDays ( 1 ) ;
int days = PROFILE . countDays ( MT_CPAP , first , last ) ;
2013-09-14 23:32:14 +00:00
RXChange rx ;
2014-04-17 05:52:25 +00:00
rx . first = first ;
rx . last = last ;
rx . days = days ;
rx . ahi = calcAHI ( first , last ) ;
rx . fl = calcFL ( first , last ) ;
rx . mode = cmode ;
rx . min = cmin ;
rx . max = cmax ;
rx . ps = cps ;
rx . pshi = cpshi ;
rx . maxipap = cmaxipap ;
rx . prelief = lastpr ;
rx . prelset = lastprelset ;
rx . machine = lastmach ;
if ( mode < MODE_BIPAP ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_Pressure , percentile , MT_CPAP , first , last ) ;
rx . per2 = 0 ;
} else if ( mode < MODE_ASV ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_IPAP , percentile , MT_CPAP , first , last ) ;
2013-09-14 23:32:14 +00:00
} else {
2014-04-17 05:52:25 +00:00
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_IPAP , percentile , MT_CPAP , first , last ) ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
rx . weighted = float ( rx . days ) / float ( cpapdays ) * rx . ahi ;
2013-09-14 23:32:14 +00:00
rxchange . push_back ( rx ) ;
}
2014-04-17 05:52:25 +00:00
cmode = mode ;
cmin = min ;
cmax = max ;
cps = ps ;
cpshi = pshi ;
cmaxipap = maxipap ;
lastpr = prelief ;
lastprelset = prelset ;
last = date ;
lastmach = mach ;
lastchanged = true ;
2013-09-14 23:32:14 +00:00
}
}
2014-04-17 05:52:25 +00:00
date = date . addDays ( - 1 ) ;
} while ( date > = firstcpap ) ;
2013-09-14 23:32:14 +00:00
// Sort list by AHI
qSort ( AHI ) ;
2014-04-17 05:52:25 +00:00
lastchanged = false ;
2014-04-23 13:19:56 +00:00
if ( ! lastchanged & & ( mach ! = nullptr ) ) {
2014-04-17 05:52:25 +00:00
// last=date.addDays(1);
first = firstcpap ;
int days = PROFILE . countDays ( MT_CPAP , first , last ) ;
2013-09-14 23:32:14 +00:00
RXChange rx ;
2014-04-17 05:52:25 +00:00
rx . first = first ;
rx . last = last ;
rx . days = days ;
rx . ahi = calcAHI ( first , last ) ;
rx . fl = calcFL ( first , last ) ;
rx . mode = mode ;
rx . min = min ;
rx . max = max ;
rx . ps = ps ;
rx . pshi = pshi ;
rx . maxipap = maxipap ;
rx . prelief = prelief ;
rx . prelset = prelset ;
rx . machine = mach ;
if ( mode < MODE_BIPAP ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_Pressure , percentile , MT_CPAP , first , last ) ;
rx . per2 = 0 ;
} else if ( mode < MODE_ASV ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_IPAP , percentile , MT_CPAP , first , last ) ;
2013-09-14 23:32:14 +00:00
} else {
2014-04-17 05:52:25 +00:00
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_IPAPHi , percentile , MT_CPAP , first , last ) ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
rx . weighted = float ( rx . days ) / float ( cpapdays ) ;
2013-09-14 23:32:14 +00:00
//rx.weighted=float(days)*rx.ahi;
rxchange . push_back ( rx ) ;
}
2014-04-17 05:52:25 +00:00
int rxthresh = 5 ;
2013-09-14 23:32:14 +00:00
QVector < RXChange * > tmpRX ;
2014-04-17 05:52:25 +00:00
for ( int i = 0 ; i < rxchange . size ( ) ; i + + ) {
RXChange & rx = rxchange [ i ] ;
if ( rx . days > rxthresh ) {
2013-09-14 23:32:14 +00:00
tmpRX . push_back ( & rx ) ;
2014-04-17 05:52:25 +00:00
}
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
QString recbox = " <html><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; } "
" a:link,a:visited { color: inherit; text-decoration: none; } " //font-weight: normal;
" a:hover { background-color: inherit; color: white; text-decoration:none; font-weight: bold; } "
" </style></head><body> " ;
recbox + = " <table width=100% cellpadding=1 cellspacing=0> " ;
int numdays = AHI . size ( ) ;
if ( numdays > 1 ) {
int z = numdays / 2 ;
if ( z > 4 ) { z = 4 ; }
recbox + = QString ( " <tr><td colspan=2 align=center><b>%1</b></td></tr> " ) . arg (
tr ( " Usage Information " ) ) ;
recbox + = QString ( " <tr><td>%1</td><td align=right>%2</td></tr> " ) . arg ( tr ( " Total Days " ) ) . arg (
numdays ) ;
2013-09-14 23:32:14 +00:00
if ( PROFILE . cpap - > showComplianceInfo ( ) ) {
2014-04-17 05:52:25 +00:00
recbox + = QString ( " <tr><td>%1</td><td align=right>%2</td></tr> " ) . arg ( tr ( " Compliant Days " ) ) . arg (
compliant ) ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
int highahi = 0 ;
for ( int i = 0 ; i < numdays ; i + + ) {
if ( AHI . at ( i ) . ahi > 5.0 ) {
2013-09-14 23:32:14 +00:00
highahi + + ;
2014-04-17 05:52:25 +00:00
}
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
recbox + = QString ( " <tr><td>%1</td><td align=right>%2</td></tr> " ) . arg ( tr ( " Days AHI >5.0 " ) ) . arg (
highahi ) ;
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
recbox + = QString ( " <tr><td colspan=2> </td></tr> " ) ;
recbox + = QString ( " <tr><td colspan=2 align=center><b>%1</b></td></tr> " ) . arg ( tr ( " Best %1 " ) . arg (
ahitxt ) ) ;
for ( int i = 0 ; i < z ; i + + ) {
const UsageData & a = AHI . at ( i ) ;
recbox + = QString ( " <tr><td><a href='daily=%1'>%2</a></td><td align=right>%3</td></tr> " )
. arg ( a . date . toString ( Qt : : ISODate ) )
. arg ( a . date . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( a . ahi , 0 , ' f ' , decimals ) ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
recbox + = QString ( " <tr><td colspan=2> </td></tr> " ) ;
recbox + = QString ( " <tr><td colspan=2 align=center><b>%1</b></td></tr> " ) . arg (
tr ( " Worst %1 " ) . arg ( ahitxt ) ) ;
for ( int i = 0 ; i < z ; i + + ) {
const UsageData & a = AHI . at ( ( numdays - 1 ) - i ) ;
recbox + = QString ( " <tr><td><a href='daily=%1'>%2</a></td><td align=right>%3</td></tr> " )
. arg ( a . date . toString ( Qt : : ISODate ) )
. arg ( a . date . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( a . ahi , 0 , ' f ' , decimals ) ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
recbox + = QString ( " <tr><td colspan=2> </td></tr> " ) ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
if ( tmpRX . size ( ) > 0 ) {
RXsort = RX_ahi ;
QString minstr , maxstr , modestr , maxhistr ;
qSort ( tmpRX . begin ( ) , tmpRX . end ( ) , RXSort ) ;
tmpRX [ 0 ] - > highlight = 4 ; // worst
int ls = tmpRX . size ( ) - 1 ;
tmpRX [ ls ] - > highlight = 1 ; //best
CPAPMode mode = ( CPAPMode ) ( int ) PROFILE . calcSettingsMax ( CPAP_Mode , MT_CPAP , tmpRX [ ls ] - > first ,
tmpRX [ ls ] - > last ) ;
if ( mode < MODE_APAP ) { // is CPAP?
minstr = STR_TR_Pressure ;
maxstr = " " ;
modestr = STR_TR_CPAP ;
} else if ( mode < MODE_BIPAP ) { // is AUTO?
minstr = STR_TR_Min ;
maxstr = STR_TR_Max ;
modestr = STR_TR_APAP ;
2013-11-19 13:06:17 +00:00
} else {
2014-04-17 05:52:25 +00:00
if ( mode < MODE_ASV ) { // BIPAP
minstr = STR_TR_EPAP ;
maxstr = STR_TR_IPAP ;
modestr = STR_TR_BiLevel ;
} else {
minstr = STR_TR_EPAP ;
maxstr = STR_TR_IPAPLo ;
maxhistr = STR_TR_IPAPHi ;
modestr = STR_TR_STASV ;
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
}
2013-11-19 13:06:17 +00:00
}
2013-09-14 23:32:14 +00:00
2014-04-17 05:52:25 +00:00
recbox + = QString ( " <tr><td colspan=2><table width=100% border=0 cellpadding=1 cellspacing=0><tr><td colspan=2 align=center><b>%3</b></td></tr> " )
. arg ( tr ( " Best RX Setting " ) ) ;
recbox + = QString ( " <tr><td valign=top> " ) + STR_TR_Start + " <br/> " + STR_TR_End +
QString ( " </td><td align=right><a href='overview=%1,%2'>%3<br/>%4</a></td></tr> " )
. arg ( tmpRX [ ls ] - > first . toString ( Qt : : ISODate ) )
. arg ( tmpRX [ ls ] - > last . toString ( Qt : : ISODate ) )
. arg ( tmpRX [ ls ] - > first . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( tmpRX [ ls ] - > last . toString ( Qt : : SystemLocaleShortDate ) ) ;
recbox + = QString ( " <tr><td><b>%1</b></td><td align=right><b>%2</b></td></tr> " ) . arg ( ahitxt ) . arg (
tmpRX [ ls ] - > ahi , 0 , ' f ' , decimals ) ;
recbox + = QString ( " <tr><td>%1</td><td align=right>%2</td></tr> " ) . arg ( STR_TR_Mode ) . arg ( modestr ) ;
recbox + = QString ( " <tr><td>%1</td><td align=right>%2%3</td></tr> " ) . arg ( minstr ) . arg ( tmpRX [ ls ] - > min ,
0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
if ( ! maxstr . isEmpty ( ) ) { recbox + = QString ( " <tr><td>%1</td><td align=right>%2%3</td></tr> " ) . arg ( maxstr ) . arg ( tmpRX [ ls ] - > max , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ; }
if ( ! maxhistr . isEmpty ( ) ) { recbox + = QString ( " <tr><td>%1</td><td align=right>%2%3</td></tr> " ) . arg ( maxhistr ) . arg ( tmpRX [ ls ] - > maxipap , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ; }
recbox + = " </table></td></tr> " ;
recbox + = QString ( " <tr><td colspan=2> </td></tr> " ) ;
mode = ( CPAPMode ) ( int ) PROFILE . calcSettingsMax ( CPAP_Mode , MT_CPAP , tmpRX [ 0 ] - > first , tmpRX [ 0 ] - > last ) ;
if ( mode < MODE_APAP ) { // is CPAP?
minstr = STR_TR_Pressure ;
maxstr = " " ;
modestr = STR_TR_CPAP ;
} else if ( mode < MODE_BIPAP ) { // is AUTO?
minstr = STR_TR_Min ;
maxstr = STR_TR_Max ;
modestr = STR_TR_APAP ;
} else if ( mode < MODE_ASV ) { // BIPAP or greater
minstr = STR_TR_EPAP ;
maxstr = STR_TR_IPAP ;
modestr = STR_TR_BiLevel ;
2013-09-14 23:32:14 +00:00
} else {
2014-04-17 05:52:25 +00:00
minstr = STR_TR_EPAP ;
maxstr = STR_TR_IPAPLo ;
maxhistr = STR_TR_IPAPHi ;
modestr = STR_TR_STASV ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
recbox + = QString ( " <tr><td colspan=2><table width=100% border=0 cellpadding=1 cellspacing=0><tr><td colspan=2 align=center><b>%3</b></td></tr> " )
. arg ( tr ( " Worst RX Setting " ) ) ;
recbox + = QString ( " <tr><td valign=top> " ) + STR_TR_Start + " <br/> " + STR_TR_End +
QString ( " </td><td align=right><a href='overview=%1,%2'>%3<br/>%4</a></td></tr> " )
. arg ( tmpRX [ 0 ] - > first . toString ( Qt : : ISODate ) )
. arg ( tmpRX [ 0 ] - > last . toString ( Qt : : ISODate ) )
. arg ( tmpRX [ 0 ] - > first . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( tmpRX [ 0 ] - > last . toString ( Qt : : SystemLocaleShortDate ) ) ;
recbox + = QString ( " <tr><td><b>%1</b></td><td align=right><b>%2</b></td></tr> " ) . arg ( ahitxt ) . arg (
tmpRX [ 0 ] - > ahi , 0 , ' f ' , decimals ) ;
recbox + = QString ( " <tr><td>%1</td><td align=right>%2</td></tr> " ) . arg ( STR_TR_Mode ) . arg ( modestr ) ;
recbox + = QString ( " <tr><td>%1</td><td align=right>%2%3</td></tr> " ) . arg ( minstr ) . arg ( tmpRX [ 0 ] - > min ,
0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
if ( ! maxstr . isEmpty ( ) ) { recbox + = QString ( " <tr><td>%1</td><td align=right>%2%3</td></tr> " ) . arg ( maxstr ) . arg ( tmpRX [ 0 ] - > max , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ; }
if ( ! maxhistr . isEmpty ( ) ) { recbox + = QString ( " <tr><td>%1</td><td align=right>%2%3</td></tr> " ) . arg ( maxhistr ) . arg ( tmpRX [ 0 ] - > maxipap , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ; }
recbox + = " </table></td></tr> " ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
recbox + = " </table> " ;
recbox + = " </body></html> " ;
2013-09-14 23:32:14 +00:00
mainwin - > setRecBoxHTML ( recbox ) ;
/*RXsort=RX_min;
RXorder = true ;
qSort ( rxchange . begin ( ) , rxchange . end ( ) ) ; */
2014-05-06 09:11:31 +00:00
html + = " <div align=center><br/> " ;
2014-04-17 05:52:25 +00:00
html + = QString ( " <table cellpadding=2 cellspacing=0 border=1 width=90%> " ) ;
2014-05-06 09:11:31 +00:00
html + = " <tr bgcolor=' " + heading_color + " '><td colspan=9 align=center><font size=+3> " + tr ( " Changes to Prescription Settings " ) + " </font></td></tr> " ;
2013-09-14 23:32:14 +00:00
QString extratxt ;
QString tooltip ;
2014-04-17 05:52:25 +00:00
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</b></td><td><b>%7</b></td><td><b>%8</b></td><td><b>%9</b></td></tr> " )
. arg ( STR_TR_First )
. arg ( STR_TR_Last )
. arg ( tr ( " Days " ) )
. arg ( ahitxt )
. arg ( tr ( " FL " ) )
. arg ( STR_TR_Machine )
. arg ( tr ( " Pr. Rel. " ) )
. arg ( STR_TR_Mode )
2014-04-25 05:28:10 +00:00
. arg ( tr ( " Pressure Settings " ) ) ;
2014-04-17 05:52:25 +00:00
for ( int i = 0 ; i < rxchange . size ( ) ; i + + ) {
RXChange rx = rxchange . at ( i ) ;
2013-09-14 23:32:14 +00:00
QString color ;
2014-04-17 05:52:25 +00:00
if ( rx . highlight = = 1 ) {
color = " #c0ffc0 " ;
} else if ( rx . highlight = = 2 ) {
color = " #e0ffe0 " ;
} else if ( rx . highlight = = 3 ) {
color = " #ffe0e0 " ;
} else if ( rx . highlight = = 4 ) {
color = " #ffc0c0 " ;
} else { color = " " ; }
2013-09-14 23:32:14 +00:00
QString machstr ;
2014-04-17 05:52:25 +00:00
if ( rx . machine - > properties . contains ( STR_PROP_Brand ) ) {
machstr + = rx . machine - > properties [ STR_PROP_Brand ] ;
}
2013-09-14 23:32:14 +00:00
if ( rx . machine - > properties . contains ( STR_PROP_Model ) ) {
2014-04-17 05:52:25 +00:00
machstr + = " " + rx . machine - > properties [ STR_PROP_Model ] ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
2013-09-14 23:32:14 +00:00
if ( rx . machine - > properties . contains ( STR_PROP_Serial ) ) {
2014-04-17 05:52:25 +00:00
machstr + = " ( " + rx . machine - > properties [ STR_PROP_Serial ] + " )<br/> " ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
mode = rx . mode ;
extratxt = " <table cellpadding=0 cellspacing=0 border=0 width=100%><tr> " ;
// tooltip=QString("%1 %2% ").arg(machstr).arg(percentile*100.0)+STR_TR_EPAP+
// QString("=%1<br/>%2% ").arg(rx.per1,0,'f',decimals).arg(percentile*100.0)+
// STR_TR_IPAP+QString("=%1").arg(rx.per2,0,'f',decimals);
if ( mode > = MODE_BIPAP ) {
if ( rx . min > 0 ) {
2014-04-25 05:28:10 +00:00
extratxt + = " <td> " + QString ( tr ( " EPAP %1 " ) )
2014-04-17 05:52:25 +00:00
. arg ( rx . min , 4 , ' f ' , 1 ) ;
2013-11-19 13:06:17 +00:00
}
2014-04-17 05:52:25 +00:00
if ( ( rx . max > 0 ) & & ( rx . min ! = rx . max ) ) {
extratxt + = QString ( " - %2 " )
. arg ( rx . max , 4 , ' f ' , 1 ) ;
2013-11-19 13:06:17 +00:00
}
2014-04-17 05:52:25 +00:00
extratxt + = " </td><td> " ;
if ( rx . ps > 0 ) {
2014-04-25 05:28:10 +00:00
extratxt + = " <td> " + QString ( tr ( " PS %1 " ) ) . arg ( rx . ps , 4 , ' f ' , 1 ) ;
2013-11-19 13:06:17 +00:00
}
2014-04-17 05:52:25 +00:00
if ( ( rx . pshi > 0 ) & & ( rx . ps ! = rx . pshi ) ) {
2014-04-25 05:28:10 +00:00
extratxt + = QString ( " - %2 " ) . arg ( rx . pshi , 4 , ' f ' , 1 ) ;
2013-11-19 13:06:17 +00:00
}
2014-04-17 05:52:25 +00:00
extratxt + = " </td> " ;
if ( rx . maxipap > 0 ) {
2014-04-25 05:28:10 +00:00
extratxt + = " <td> " + QString ( tr ( " IPAP %1 " ) + " </td> " )
2014-04-17 05:52:25 +00:00
. arg ( rx . maxipap , 4 , ' f ' , 1 ) ;
2013-11-19 13:06:17 +00:00
}
2014-04-17 05:52:25 +00:00
tooltip = QString ( " %1 %2% " ) . arg ( machstr ) . arg ( percentile * 100.0 ) +
STR_TR_EPAP +
QString ( " =%1<br/>%2% " ) . arg ( rx . per1 , 0 , ' f ' , decimals )
. arg ( percentile * 100.0 )
+ STR_TR_IPAP + QString ( " =%1 " ) . arg ( rx . per2 , 0 , ' f ' , decimals ) ;
} else if ( mode > MODE_CPAP ) {
2014-04-25 05:28:10 +00:00
extratxt + = " <td align=left> " + QString ( tr ( " APAP %1 - %2 " ) + " </td><td align=left></td> " )
2014-04-17 05:52:25 +00:00
. arg ( rx . min , 4 , ' f ' , 1 )
. arg ( rx . max , 4 , ' f ' , 1 ) ;
tooltip = QString ( " %1 %2% " ) . arg ( machstr ) . arg ( percentile * 100.0 ) + STR_TR_Pressure +
QString ( " =%2 " ) . arg ( rx . per1 , 0 , ' f ' , decimals ) ;
2013-09-14 23:32:14 +00:00
} else {
2013-11-19 13:06:17 +00:00
2014-04-17 05:52:25 +00:00
if ( cpapmode > MODE_CPAP ) {
2014-04-25 05:28:10 +00:00
extratxt + = " <td colspan=2> " + QString ( tr ( " CPAP %1 " ) + " </td> " ) . arg ( rx . min , 4 , ' f ' , 1 ) ;
2014-04-17 05:52:25 +00:00
tooltip = QString ( " %1 " ) . arg ( machstr ) ;
2013-09-14 23:32:14 +00:00
} else {
2014-04-17 05:52:25 +00:00
extratxt + = " " ;
tooltip = " " ;
2013-09-14 23:32:14 +00:00
}
}
2014-04-17 05:52:25 +00:00
extratxt + = " </tr></table> " ;
2013-09-14 23:32:14 +00:00
QString presrel ;
2014-04-17 05:52:25 +00:00
if ( rx . prelset > 0 ) {
presrel = schema : : channel [ CPAP_PresReliefType ] . option ( int ( rx . prelief ) ) ;
presrel + = QString ( " x%1 " ) . arg ( rx . prelset ) ;
} else { presrel = STR_TR_None ; }
QString tooltipshow , tooltiphide ;
2013-09-14 23:32:14 +00:00
if ( ! tooltip . isEmpty ( ) ) {
2014-04-17 05:52:25 +00:00
tooltipshow = QString ( " tooltip.show( \" %1 \" ); " ) . arg ( tooltip ) ;
tooltiphide = " tooltip.hide(); " ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
html + = QString ( " <tr bgcolor=' " + color +
" ' onmouseover='ChangeColor(this, \" #eeeeee \" ); %12' onmouseout='ChangeColor(this, \" " + color +
" \" ); %13' onclick='alert( \" overview=%1,%2 \" );'><td>%3</td><td>%4</td><td>%5</td><td>%6</td><td>%7</td><td>%8</td><td>%9</td><td>%10</td><td>%11</td></tr> " )
2013-09-14 23:32:14 +00:00
. arg ( rx . first . toString ( Qt : : ISODate ) )
. arg ( rx . last . toString ( Qt : : ISODate ) )
. arg ( rx . first . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( rx . last . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( rx . days )
2014-04-17 05:52:25 +00:00
. arg ( rx . ahi , 0 , ' f ' , decimals )
. arg ( rx . fl , 0 , ' f ' , decimals ) // Not the best way to do this.. Todo: Add an extra field for data..
2013-09-14 23:32:14 +00:00
. arg ( rx . machine - > GetClass ( ) )
. arg ( presrel )
2014-04-17 05:52:25 +00:00
. arg ( schema : : channel [ CPAP_Mode ] . option ( int ( rx . mode ) - 1 ) )
2013-09-14 23:32:14 +00:00
. arg ( extratxt )
. arg ( tooltipshow )
. arg ( tooltiphide ) ;
}
2014-04-17 05:52:25 +00:00
html + = " </table> " ;
html + = QString ( " <i> " ) +
tr ( " Efficacy highlighting ignores prescription settings with less than %1 days of recorded data. " ) . arg (
rxthresh ) + QString ( " </i><br/> " ) ;
html + = " </div> " ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
if ( mach . size ( ) > 0 ) {
2014-05-06 09:11:31 +00:00
html + = " <div align=center><br/> " ;
2014-04-17 05:52:25 +00:00
html + = QString ( " <table cellpadding=2 cellspacing=0 border=1 width=90%> " ) ;
2014-05-06 09:11:31 +00:00
html + = " <tr bgcolor=' " + heading_color + " '><td colspan=5 align=center><font size=+3> " + tr ( " Machine Information " ) + " </font></td></tr> " ;
2014-04-17 05:52:25 +00:00
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 )
. arg ( STR_TR_Serial )
. arg ( tr ( " First Use " ) )
. arg ( tr ( " Last Use " ) ) ;
2013-09-14 23:32:14 +00:00
Machine * m ;
2014-04-17 05:52:25 +00:00
for ( int i = 0 ; i < mach . size ( ) ; i + + ) {
m = mach . at ( i ) ;
if ( m - > GetType ( ) = = MT_JOURNAL ) { continue ; }
QString mn = m - > properties [ STR_PROP_ModelNumber ] ;
2013-09-14 23:32:14 +00:00
//if (mn.isEmpty())
2014-04-17 05:52:25 +00:00
html + = QString ( " <tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td></tr> " )
2013-09-14 23:32:14 +00:00
. arg ( m - > properties [ STR_PROP_Brand ] )
2014-04-17 05:52:25 +00:00
. arg ( m - > properties [ STR_PROP_Model ] + " " + m - > properties [ STR_PROP_SubModel ] +
( mn . isEmpty ( ) ? " " : QString ( " ( " ) + mn + QString ( " ) " ) ) )
2013-09-14 23:32:14 +00:00
. arg ( m - > properties [ STR_PROP_Serial ] )
. arg ( m - > FirstDay ( ) . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( m - > LastDay ( ) . toString ( Qt : : SystemLocaleShortDate ) ) ;
}
2014-04-17 05:52:25 +00:00
html + = " </table> " ;
html + = " </div> " ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
html + = " <script type='text/javascript' language='javascript' src='qrc:/docs/script.js'></script> " ;
2013-09-14 23:32:14 +00:00
//updateFavourites();
2014-04-17 05:52:25 +00:00
html + = htmlFooter ( ) ;
2013-09-14 23:32:14 +00:00
return html ;
}
2014-05-06 09:11:31 +00:00
QString StatisticsRow : : value ( QDate start , QDate end )
{
const int decimals = 2 ;
2014-05-06 11:54:41 +00:00
QString value = " " ;
2014-05-06 09:11:31 +00:00
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 ;
}