2014-08-17 12:56:05 +00:00
/* Statistics Report Generator Implementation
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>
2014-09-11 14:23:08 +00:00
# include <QFile>
# include <QDataStream>
2013-09-14 23:32:14 +00:00
# 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);
}
2014-09-11 14:23:08 +00:00
QDataStream & operator > > ( QDataStream & in , RXItem & rx )
{
in > > rx . start ;
in > > rx . end ;
in > > rx . days ;
in > > rx . ahi ;
in > > rx . rdi ;
in > > rx . hours ;
QString loadername ;
in > > loadername ;
QString serial ;
in > > serial ;
MachineLoader * loader = GetLoader ( loadername ) ;
if ( loader ) {
rx . machine = loader - > lookupMachine ( serial ) ;
} else {
qDebug ( ) < < " Bad machine object " < < loadername < < serial ;
rx . machine = nullptr ;
}
in > > rx . relief ;
in > > rx . mode ;
in > > rx . pressure ;
QList < QDate > list ;
in > > list ;
rx . dates . clear ( ) ;
for ( int i = 0 ; i < list . size ( ) ; + + i ) {
QDate date = list . at ( i ) ;
rx . dates [ date ] = p_profile - > FindDay ( date , MT_CPAP ) ;
}
in > > rx . s_count ;
in > > rx . s_sum ;
return in ;
}
QDataStream & operator < < ( QDataStream & out , const RXItem & rx )
{
out < < rx . start ;
out < < rx . end ;
out < < rx . days ;
out < < rx . ahi ;
out < < rx . rdi ;
out < < rx . hours ;
out < < rx . machine - > loaderName ( ) ;
out < < rx . machine - > serial ( ) ;
2014-09-29 14:41:31 +00:00
2014-09-11 14:23:08 +00:00
out < < rx . relief ;
out < < rx . mode ;
out < < rx . pressure ;
out < < rx . dates . keys ( ) ;
out < < rx . s_count ;
out < < rx . s_sum ;
return out ;
}
void Statistics : : loadRXChanges ( )
{
QString path = p_profile - > Get ( " { " + STR_GEN_DataFolder + " }/RXChanges.cache " ) ;
QFile file ( path ) ;
if ( ! file . open ( QFile : : ReadOnly ) ) {
return ;
}
QDataStream in ( & file ) ;
in . setByteOrder ( QDataStream : : LittleEndian ) ;
quint32 mag32 ;
if ( in . version ( ) ! = QDataStream : : Qt_5_0 ) {
}
in > > mag32 ;
if ( mag32 ! = magic ) {
return ;
}
quint16 version ;
in > > version ;
in > > rxitems ;
}
void Statistics : : saveRXChanges ( )
{
QString path = p_profile - > Get ( " { " + STR_GEN_DataFolder + " }/RXChanges.cache " ) ;
QFile file ( path ) ;
if ( ! file . open ( QFile : : WriteOnly ) ) {
return ;
}
QDataStream out ( & file ) ;
out . setByteOrder ( QDataStream : : LittleEndian ) ;
out . setVersion ( QDataStream : : Qt_5_0 ) ;
out < < magic ;
out < < ( quint16 ) 0 ;
out < < rxitems ;
}
bool rxAHILessThan ( const RXItem * rx1 , const RXItem * rx2 )
{
return ( double ( rx1 - > ahi ) / rx1 - > hours ) < ( double ( rx2 - > ahi ) / rx2 - > hours ) ;
}
void Statistics : : updateRXChanges ( )
{
rxitems . clear ( ) ;
loadRXChanges ( ) ;
QMap < QDate , Day * > : : iterator di ;
QMap < QDate , Day * > : : iterator it ;
QMap < QDate , Day * > : : iterator it_end = p_profile - > daylist . end ( ) ;
QMap < QDate , RXItem > : : iterator ri ;
QMap < QDate , RXItem > : : iterator ri_end = rxitems . end ( ) ;
quint64 tmp ;
for ( it = p_profile - > daylist . begin ( ) ; it ! = it_end ; + + it ) {
const QDate & date = it . key ( ) ;
Day * day = it . value ( ) ;
Machine * mach = day - > machine ( MT_CPAP ) ;
if ( mach = = nullptr ) continue ;
bool fnd = false ;
ri_end = rxitems . end ( ) ;
for ( ri = rxitems . begin ( ) ; ri ! = ri_end ; + + ri ) {
RXItem & rx = ri . value ( ) ;
if ( ( date > = rx . start ) & & ( date < = rx . end ) ) {
// Fits in date range
if ( rx . dates . contains ( date ) ) {
fnd = true ;
break ;
}
// First up, check if fits in date range, but isn't loaded for some reason
// Need summaries for this
day - > OpenSummary ( ) ;
QList < ChannelID > flags = day - > getSortedMachineChannels ( MT_CPAP , schema : : FLAG | schema : : MINOR_FLAG | schema : : SPAN ) ;
QString relief = day - > getPressureRelief ( ) ;
QString mode = day - > getCPAPMode ( ) ;
QString pressure = day - > getPressureSettings ( ) ;
if ( ( rx . relief = = relief ) & & ( rx . mode = = mode ) & & ( rx . pressure = = pressure ) & & ( rx . machine = = mach ) ) {
for ( int i = 0 ; i < flags . size ( ) ; i + + ) {
ChannelID code = flags . at ( i ) ;
rx . s_count [ code ] + = day - > count ( code ) ;
rx . s_sum [ code ] + = day - > sum ( code ) ;
}
tmp = day - > count ( CPAP_Hypopnea ) + day - > count ( CPAP_Obstructive ) + day - > count ( CPAP_Apnea ) + day - > count ( CPAP_ClearAirway ) ;
rx . ahi + = tmp ;
rx . rdi + = tmp + day - > count ( CPAP_RERA ) ;
rx . hours + = day - > hours ( MT_CPAP ) ;
rx . dates [ date ] = day ;
rx . days = rx . dates . size ( ) ;
fnd = true ;
break ;
} else {
// Bleh.... split the day record!
RXItem rx1 , rx2 ;
// First create the new day..
rx1 . start = date ;
rx1 . end = date ;
rx1 . days = 1 ;
tmp = day - > count ( CPAP_Hypopnea ) + day - > count ( CPAP_Obstructive ) + day - > count ( CPAP_Apnea ) + day - > count ( CPAP_ClearAirway ) ;
rx1 . ahi = tmp ;
rx1 . rdi = tmp + day - > count ( CPAP_RERA ) ;
for ( int i = 0 ; i < flags . size ( ) ; i + + ) {
ChannelID code = flags . at ( i ) ;
rx1 . s_count [ code ] = day - > count ( code ) ;
rx1 . s_sum [ code ] = day - > sum ( code ) ;
}
rx1 . hours = day - > hours ( MT_CPAP ) ;
rx1 . relief = relief ;
rx1 . mode = mode ;
rx1 . pressure = pressure ;
rx1 . machine = day - > machine ( MT_CPAP ) ;
rx1 . dates [ date ] = day ;
rxitems . insert ( date , rx1 ) ;
QMap < QDate , Day * > datecopy = rx . dates ;
rx . dates . clear ( ) ;
rx . end = rx . start ;
rx2 . start = rx . end ;
rx2 . end = rx . start ;
rx2 . ahi = 0 ;
rx2 . rdi = 0 ;
rx2 . hours = 0 ;
rx . ahi = 0 ;
rx . rdi = 0 ;
rx . hours = 0 ;
rx . s_count . clear ( ) ;
rx2 . s_count . clear ( ) ;
rx . s_sum . clear ( ) ;
rx2 . s_sum . clear ( ) ;
for ( di = datecopy . begin ( ) ; di ! = datecopy . end ( ) ; + + di ) {
// Split everything before
if ( di . key ( ) < date ) {
Day * dy = rx . dates [ di . key ( ) ] = p_profile - > GetDay ( di . key ( ) , MT_CPAP ) ;
tmp = dy - > count ( CPAP_Hypopnea ) + dy - > count ( CPAP_Obstructive ) + dy - > count ( CPAP_Apnea ) + dy - > count ( CPAP_ClearAirway ) ; ;
rx . ahi + = tmp ;
rx . rdi + = tmp + dy - > count ( CPAP_RERA ) ;
QList < ChannelID > flags2 = dy - > getSortedMachineChannels ( MT_CPAP , schema : : FLAG | schema : : MINOR_FLAG | schema : : SPAN ) ;
for ( int i = 0 ; i < flags2 . size ( ) ; i + + ) {
ChannelID code = flags2 . at ( i ) ;
rx . s_count [ code ] + = dy - > count ( code ) ;
rx . s_sum [ code ] + = dy - > sum ( code ) ;
}
rx . hours + = dy - > hours ( MT_CPAP ) ;
//rx.days++;
rx . end = qMax ( di . key ( ) , rx . end ) ;
}
// Split everything after
if ( di . key ( ) > date ) {
Day * dy = rx2 . dates [ di . key ( ) ] = p_profile - > GetDay ( di . key ( ) , MT_CPAP ) ;
tmp = dy - > count ( CPAP_Hypopnea ) + dy - > count ( CPAP_Obstructive ) + dy - > count ( CPAP_Apnea ) + dy - > count ( CPAP_ClearAirway ) ; ;
rx2 . ahi + = tmp ;
rx2 . rdi + = tmp + dy - > count ( CPAP_RERA ) ;
QList < ChannelID > flags2 = dy - > getSortedMachineChannels ( MT_CPAP , schema : : FLAG | schema : : MINOR_FLAG | schema : : SPAN ) ;
for ( int i = 0 ; i < flags2 . size ( ) ; i + + ) {
ChannelID code = flags2 . at ( i ) ;
rx2 . s_count [ code ] + = dy - > count ( code ) ;
rx2 . s_sum [ code ] + = dy - > sum ( code ) ;
}
rx2 . hours + = dy - > hours ( MT_CPAP ) ;
rx2 . end = qMax ( di . key ( ) , rx2 . end ) ;
rx2 . start = qMin ( di . key ( ) , rx2 . start ) ;
}
}
rx . days = rx . dates . size ( ) ;
rx2 . pressure = rx . pressure ;
rx2 . mode = rx . mode ;
rx2 . relief = rx . relief ;
rx2 . machine = rx . machine ;
rx2 . days = rx2 . dates . size ( ) ;
rxitems . insert ( rx2 . end , rx2 ) ;
fnd = true ;
break ;
}
}
}
if ( fnd ) continue ;
day - > OpenSummary ( ) ;
QList < ChannelID > flags3 = day - > getSortedMachineChannels ( MT_CPAP , schema : : FLAG | schema : : MINOR_FLAG | schema : : SPAN ) ;
QString relief = day - > getPressureRelief ( ) ;
QString mode = day - > getCPAPMode ( ) ;
QString pressure = day - > getPressureSettings ( ) ;
for ( ri = rxitems . begin ( ) ; ri ! = ri_end ; + + ri ) {
RXItem & rx = ri . value ( ) ;
if ( rx . end . daysTo ( date ) = = 1 ) {
if ( ( rx . relief = = relief ) & & ( rx . mode = = mode ) & & ( rx . pressure = = pressure ) & & ( rx . machine = = day - > machine ( MT_CPAP ) ) ) {
tmp = day - > count ( CPAP_Hypopnea ) + day - > count ( CPAP_Obstructive ) + day - > count ( CPAP_Apnea ) + day - > count ( CPAP_ClearAirway ) ;
rx . ahi + = tmp ;
rx . rdi + = tmp + day - > count ( CPAP_RERA ) ;
for ( int i = 0 ; i < flags3 . size ( ) ; i + + ) {
ChannelID code = flags3 . at ( i ) ;
rx . s_count [ code ] + = day - > count ( code ) ;
rx . s_sum [ code ] + = day - > sum ( code ) ;
}
rx . hours + = day - > hours ( MT_CPAP ) ;
rx . dates [ date ] = day ;
rx . days = rx . dates . size ( ) ;
rx . end = date ;
fnd = true ;
break ;
}
}
}
if ( ! fnd ) {
RXItem rx ;
rx . start = date ;
rx . end = date ;
rx . days = 1 ;
tmp = day - > count ( CPAP_Hypopnea ) + day - > count ( CPAP_Obstructive ) + day - > count ( CPAP_Apnea ) + day - > count ( CPAP_ClearAirway ) ;
rx . ahi = tmp ;
rx . rdi = tmp + day - > count ( CPAP_RERA ) ;
for ( int i = 0 ; i < flags3 . size ( ) ; i + + ) {
ChannelID code = flags3 . at ( i ) ;
rx . s_count [ code ] = day - > count ( code ) ;
rx . s_sum [ code ] = day - > sum ( code ) ;
}
rx . hours = day - > hours ( ) ;
rx . relief = relief ;
rx . mode = mode ;
rx . pressure = pressure ;
rx . machine = day - > machine ( MT_CPAP ) ;
rx . dates . insert ( date , day ) ;
rxitems . insert ( date , rx ) ;
}
}
saveRXChanges ( ) ;
QList < RXItem * > list ;
ri_end = rxitems . end ( ) ;
for ( ri = rxitems . begin ( ) ; ri ! = ri_end ; + + ri ) {
list . append ( & ri . value ( ) ) ;
ri . value ( ) . highlight = 0 ;
}
qSort ( list . begin ( ) , list . end ( ) , rxAHILessThan ) ;
if ( list . size ( ) > = 4 ) {
list [ 0 ] - > highlight = 1 ; // best
list [ 1 ] - > highlight = 2 ; // best
int ls = list . size ( ) - 1 ;
list [ ls - 1 ] - > highlight = 3 ; // best
list [ ls ] - > highlight = 4 ;
} else if ( list . size ( ) > = 2 ) {
list [ 0 ] - > highlight = 1 ; // best
int ls = list . size ( ) - 1 ;
list [ ls ] - > highlight = 4 ;
} else if ( list . size ( ) > 0 ) {
list [ 0 ] - > highlight = 1 ; // best
}
// update record box info..
}
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 ) ) ;
2014-05-17 05:12:57 +00:00
rows . push_back ( StatisticsRow ( tr ( " Compliance " ) , SC_COMPLIANCE , MT_CPAP ) ) ;
2014-05-06 17:39:05 +00:00
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 ) ) ;
2014-05-15 04:26:48 +00:00
rows . push_back ( StatisticsRow ( " SensAwake " , SC_CPH , MT_CPAP ) ) ;
2014-05-06 17:39:05 +00:00
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 ) ) ;
2014-05-15 21:48:53 +00:00
rows . push_back ( StatisticsRow ( " Leak " , SC_ABOVE , MT_CPAP ) ) ;
2014-05-06 17:39:05 +00:00
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 " ) ;
2014-05-15 21:48:53 +00:00
calcnames [ SC_ABOVE ] = tr ( " % of time above %1 threshold " ) ;
calcnames [ SC_BELOW ] = tr ( " % of time below %1 threshold " ) ;
2014-05-06 17:39:05 +00:00
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
}
2014-07-25 07:53:48 +00:00
const QString table_width = " width=99% " ;
QString htmlHeader ( bool showheader )
2013-09-14 23:32:14 +00:00
{
2014-07-11 12:09:38 +00:00
QString address = p_profile - > user - > address ( ) ;
2014-04-17 05:52:25 +00:00
address . replace ( " \n " , " <br/> " ) ;
2014-05-06 09:33:06 +00:00
QString userinfo ;
2014-07-11 12:09:38 +00:00
if ( ! p_profile - > user - > firstName ( ) . isEmpty ( ) ) {
userinfo = QString ( QObject : : tr ( " Name: %1, %2 " ) ) . arg ( p_profile - > user - > lastName ( ) ) . arg ( p_profile - > user - > firstName ( ) ) + " <br/> " ;
if ( ! p_profile - > user - > DOB ( ) . isNull ( ) ) {
userinfo + = QString ( QObject : : tr ( " DOB: %1 " ) ) . arg ( p_profile - > user - > DOB ( ) . toString ( ) ) + " <br/> " ;
2014-05-06 09:33:06 +00:00
}
2014-07-11 12:09:38 +00:00
if ( ! p_profile - > user - > phone ( ) . isEmpty ( ) ) {
userinfo + = QString ( QObject : : tr ( " Phone: %1 " ) ) . arg ( p_profile - > user - > phone ( ) ) + " <br/> " ;
2014-05-06 12:37:00 +00:00
}
2014-07-11 12:09:38 +00:00
if ( ! p_profile - > user - > email ( ) . isEmpty ( ) ) {
userinfo + = QString ( QObject : : tr ( " Email: %1 " ) ) . arg ( p_profile - > user - > email ( ) ) + " <br/><br/> " ;
2014-05-06 12:37:00 +00:00
}
2014-07-11 12:09:38 +00:00
if ( ! p_profile - > user - > address ( ) . isEmpty ( ) ) {
2014-05-06 12:37:00 +00:00
userinfo + = QObject : : tr ( " Address: " ) + " <br/> " + address ;
}
2014-05-06 09:33:06 +00:00
}
2014-07-25 07:53:48 +00:00
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; } "
2014-05-28 13:09:03 +00:00
2014-07-25 07:53:48 +00:00
" table.curved { "
2014-05-26 19:29:01 +00:00
" border: 1px solid gray; "
" border-radius:10px; "
" -moz-border-radius:10px; "
" -webkit-border-radius:10px; "
2014-05-28 09:35:21 +00:00
" page-break-after:auto; "
" -fs-table-paginate: paginate; "
2014-07-25 07:53:48 +00:00
" } "
" tr.datarow:nth-child(even) { "
2014-05-26 19:29:01 +00:00
" background-color: #f8f8f8; "
2014-07-25 07:53:48 +00:00
" } "
" 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> " ;
if ( showheader ) {
html + = " <div align=center><table class=curved " + table_width + " > "
" <td> " + userinfo + " </td> "
" <td align='right'> "
" <font size='+2'> " + STR_TR_SleepyHead + " </font><br/> "
" <font size='+1'> " + 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><br/> " ;
}
return html ;
2013-09-14 23:32:14 +00:00
}
2014-07-25 07:53:48 +00:00
QString htmlFooter ( bool showinfo = true )
2013-09-14 23:32:14 +00:00
{
2014-07-25 07:53:48 +00:00
QString html ;
if ( showinfo ) {
html + = " <hr/><div align=center><font size='-1'><i> " ;
html + = 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 " ) ;
html + = " </i></font></div> " ;
}
html + = " </body></html> " ;
return 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 ) ) ;
2014-07-11 12:09:38 +00:00
if ( p_profile - > general - > calculateRDI ( ) ) {
2014-04-17 05:52:25 +00:00
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-05-15 03:51:32 +00:00
EventDataType calcSA ( QDate start , QDate end )
{
EventDataType val = ( p_profile - > calcCount ( CPAP_SensAwake , MT_CPAP , start , end ) ) ;
EventDataType hours = p_profile - > calcHours ( MT_CPAP , start , end ) ;
if ( hours > 0 ) {
val / = hours ;
} else {
val = 0 ;
}
return val ;
}
2013-09-14 23:32:14 +00:00
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 ;
2014-08-03 13:00:13 +00:00
pressure_string = copy . pressure_string ;
pr_relief_string = copy . pr_relief_string ;
2013-09-14 23:32:14 +00:00
}
QDate first ;
QDate last ;
int days ;
EventDataType ahi ;
EventDataType fl ;
CPAPMode mode ;
2014-08-03 13:00:13 +00:00
QString pressure_string ;
QString pr_relief_string ;
2013-09-14 23:32:14 +00:00
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 ;
2014-04-17 05:52:25 +00:00
Machine * machine ;
2013-09-14 23:32:14 +00:00
short highlight ;
} ;
2014-09-30 11:02:35 +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 };
//RXSortMode RXsort = RX_first;
//bool RXorder = false;
2014-04-17 05:52:25 +00:00
2014-09-30 11:02:35 +00:00
/*bool operator<(const RXChange &c1, const RXChange &c2)
2014-04-17 05:52:25 +00:00
{
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 ;
2014-09-30 11:02:35 +00:00
} */
2013-09-14 23:32:14 +00:00
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-09-11 14:23:08 +00:00
const QString heading_color = " #ffffff " ;
const QString subheading_color = " #e0e0e0 " ;
2014-09-30 11:02:35 +00:00
//const int rxthresh = 5;
2014-09-11 14:23:08 +00:00
QString Statistics : : GenerateMachineList ( )
2013-09-14 23:32:14 +00:00
{
2014-09-11 14:23:08 +00:00
QList < Machine * > cpap_machines = p_profile - > GetMachines ( MT_CPAP ) ;
QList < Machine * > oximeters = p_profile - > GetMachines ( MT_OXIMETER ) ;
QList < Machine * > mach ;
2014-05-06 09:11:31 +00:00
2014-09-11 14:23:08 +00:00
mach . append ( cpap_machines ) ;
mach . append ( oximeters ) ;
2014-05-06 09:11:31 +00:00
2014-09-11 14:23:08 +00:00
QString html ;
if ( mach . size ( ) > 0 ) {
html + = " <div align=center><br/> " ;
2014-04-17 05:52:25 +00:00
2014-09-11 14:23:08 +00:00
html + = QString ( " <table class=curved style= \" page-break-before:auto; \" " + table_width + " > " ) ;
2014-05-06 09:11:31 +00:00
2014-09-11 14:23:08 +00:00
html + = " <thead> " ;
html + = " <tr bgcolor=' " + heading_color + " '><th colspan=7 align=center><font size=+2> " + tr ( " Machine Information " ) + " </font></th></tr> " ;
2014-05-06 09:11:31 +00:00
2014-09-11 14:23:08 +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></tr> " )
. arg ( STR_TR_Brand )
. arg ( STR_TR_Series )
. arg ( STR_TR_Model )
. arg ( STR_TR_Serial )
. arg ( tr ( " First Use " ) )
. arg ( tr ( " Last Use " ) ) ;
2014-04-17 05:52:25 +00:00
2014-09-11 14:23:08 +00:00
html + = " </thead> " ;
Machine * m ;
for ( int i = 0 ; i < mach . size ( ) ; i + + ) {
m = mach . at ( i ) ;
if ( m - > type ( ) = = MT_JOURNAL ) { continue ; }
QDate d1 = m - > FirstDay ( ) ;
QDate d2 = m - > LastDay ( ) ;
QString mn = m - > modelnumber ( ) ;
html + = QString ( " <tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr> " )
. arg ( m - > brand ( ) )
. arg ( m - > series ( ) )
. arg ( m - > model ( ) +
( mn . isEmpty ( ) ? " " : QString ( " ( " ) + mn + QString ( " ) " ) ) )
. arg ( m - > serial ( ) )
. arg ( d1 . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( d2 . toString ( Qt : : SystemLocaleShortDate ) ) ;
}
2014-04-17 05:52:25 +00:00
2014-09-11 14:23:08 +00:00
html + = " </table> " ;
html + = " </div> " ;
}
return html ;
}
QString Statistics : : GenerateRXChanges ( )
{
updateRXChanges ( ) ;
QString ahitxt ;
bool rdi = p_profile - > general - > calculateRDI ( ) ;
if ( rdi ) {
ahitxt = STR_TR_RDI ;
} else {
ahitxt = STR_TR_AHI ;
}
QString html = " <div align=center><br/> " ;
html + = QString ( " <table class=curved style= \" page-break-before:always; \" " + table_width + " > " ) ;
html + = " <thead> " ;
html + = " <tr bgcolor=' " + heading_color + " '><th colspan=10 align=center><font size=+2> " + tr ( " Changes to Prescription Settings " ) + " </font></th></tr> " ;
QString extratxt ;
QString tooltip ;
QStringList hdrlist ;
hdrlist . push_back ( STR_TR_First ) ;
hdrlist . push_back ( STR_TR_Last ) ;
hdrlist . push_back ( tr ( " Days " ) ) ;
hdrlist . push_back ( ahitxt ) ;
hdrlist . push_back ( STR_TR_FL ) ;
hdrlist . push_back ( STR_TR_Machine ) ;
hdrlist . push_back ( tr ( " Pressure Relief " ) ) ;
hdrlist . push_back ( STR_TR_Mode ) ;
hdrlist . push_back ( tr ( " Pressure Settings " ) ) ;
html + = " <tr> \n " ;
for ( int i = 0 ; i < hdrlist . size ( ) ; + + i ) {
html + = QString ( " <th align=left><b>%1</b></th> \n " ) . arg ( hdrlist . at ( i ) ) ;
}
html + = " </tr> \n " ;
html + = " </thead> " ;
// html += "<tfoot>";
// html += "<tr><td colspan=10 align=center>";
// html += QString("<i>") +
// tr("Efficacy highlighting ignores prescription settings with less than %1 days of recorded data.").
// arg(rxthresh) + QString("</i><br/>");
// html += "</td></tr>";
// html += "</tfoot>";
QMapIterator < QDate , RXItem > it ( rxitems ) ;
it . toBack ( ) ;
while ( it . hasPrevious ( ) ) {
it . previous ( ) ;
const RXItem & rx = it . value ( ) ;
QString color ;
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 = " " ; }
QString datarowclass ;
if ( rx . highlight = = 0 ) datarowclass = " class=datarow " ;
html + = QString ( " <tr %4 bgcolor='%1' onmouseover='ChangeColor(this, \" #dddddd \" );' onmouseout='ChangeColor(this, \" %1 \" );' onclick='alert( \" overview=%2,%3 \" );'> " )
. arg ( color )
. arg ( rx . start . toString ( Qt : : ISODate ) )
. arg ( rx . end . toString ( Qt : : ISODate ) )
. arg ( datarowclass ) ;
double ahi = rdi ? ( double ( rx . rdi ) / rx . hours ) : ( double ( rx . ahi ) / rx . hours ) ;
double fli = double ( rx . count ( CPAP_FlowLimit ) ) / rx . hours ;
html + = QString ( " <td>%1</td> " ) . arg ( rx . start . toString ( ) ) +
QString ( " <td>%1</td> " ) . arg ( rx . end . toString ( ) ) +
QString ( " <td>%1</td> " ) . arg ( rx . days ) +
QString ( " <td>%1</td> " ) . arg ( ahi , 0 , ' f ' , 2 ) +
QString ( " <td>%1</td> " ) . arg ( fli , 0 , ' f ' , 2 ) +
2014-09-30 16:28:14 +00:00
QString ( " <td>%1 (%2)</td> " ) . arg ( rx . machine - > model ( ) ) . arg ( rx . machine - > modelnumber ( ) ) +
2014-09-11 14:23:08 +00:00
QString ( " <td>%1</td> " ) . arg ( rx . relief ) +
QString ( " <td>%1</td> " ) . arg ( rx . mode ) +
QString ( " <td>%1</td> " ) . arg ( rx . pressure ) +
" </tr> " ;
}
html + = " </table></div> " ;
return html ;
}
QString Statistics : : GenerateHTML ( )
{
2014-07-11 12:09:38 +00:00
QList < Machine * > cpap_machines = p_profile - > GetMachines ( MT_CPAP ) ;
QList < Machine * > oximeters = p_profile - > GetMachines ( MT_OXIMETER ) ;
2013-09-14 23:32:14 +00:00
QList < Machine * > mach ;
2014-07-25 07:53:48 +00:00
2013-09-14 23:32:14 +00:00
mach . append ( cpap_machines ) ;
mach . append ( oximeters ) ;
2014-07-25 07:53:48 +00:00
bool havedata = false ;
for ( int i = 0 ; i < mach . size ( ) ; + + i ) {
int daysize = mach [ i ] - > day . size ( ) ;
if ( daysize > 0 ) {
havedata = true ;
break ;
}
}
2014-09-11 14:23:08 +00:00
2014-07-25 07:53:48 +00:00
QString html = htmlHeader ( havedata ) ;
2013-09-14 23:32:14 +00:00
2014-09-11 14:23:08 +00:00
// return "";
// Find first and last days with valid CPAP data
QDate lastcpap = p_profile - > LastGoodDay ( MT_CPAP ) ;
QDate firstcpap = p_profile - > FirstGoodDay ( MT_CPAP ) ;
QDate cpapweek = lastcpap . addDays ( - 6 ) ;
QDate cpapmonth = lastcpap . addDays ( - 29 ) ;
QDate cpap6month = lastcpap . addMonths ( - 6 ) ;
QDate cpapyear = lastcpap . addMonths ( - 12 ) ;
if ( cpapweek < firstcpap ) { cpapweek = firstcpap ; }
if ( cpapmonth < firstcpap ) { cpapmonth = firstcpap ; }
if ( cpap6month < firstcpap ) { cpap6month = firstcpap ; }
if ( cpapyear < firstcpap ) { cpapyear = firstcpap ; }
2014-07-25 07:53:48 +00:00
if ( ! havedata ) {
html + = " <div align=center><table class=curved height=100% " + table_width + " > " ;
html + = QString ( " <tr><td colspan=2 align=center> " ) +
" <img src='qrc:/icons/bob-v3.0.png' height=100px> "
" <p><font size=+3> " + tr ( " I can haz data?!? " ) + " </font></p> " +
" <p><i> " + tr ( " This lonely sleepy sheep has no data :( " ) + " </i></p> "
" </table></div> " ;
html + = htmlFooter ( havedata ) ;
2013-09-14 23:32:14 +00:00
return html ;
}
2014-09-11 14:23:08 +00:00
2014-09-30 11:02:35 +00:00
// int cpapdays = p_profile->countDays(MT_CPAP, firstcpap, lastcpap);
2014-04-17 05:52:25 +00:00
2014-07-25 07:53:48 +00:00
// CPAPMode cpapmode = (CPAPMode)(int)p_profile->calcSettingsMax(CPAP_Mode, MT_CPAP, firstcpap, lastcpap);
2013-09-14 23:32:14 +00:00
2014-09-30 11:02:35 +00:00
// float percentile = p_profile->general->prefCalcPercentile() / 100.0;
2013-09-14 23:32:14 +00:00
2014-07-11 12:09:38 +00:00
// int mididx=p_profile->general->prefCalcMiddle();
2014-04-17 05:52:25 +00:00
// 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
2014-07-11 12:09:38 +00:00
if ( p_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-09-30 11:02:35 +00:00
// int decimals = 2;
2014-04-17 05:52:25 +00:00
html + = " <div align=center> " ;
2014-07-25 07:53:48 +00:00
html + = QString ( " <table class=curved " + table_width + " > " ) ;
2014-04-17 05:52:25 +00:00
2014-05-06 11:54:41 +00:00
int number_periods = 0 ;
if ( p_profile - > general - > statReportMode ( ) = = 1 ) {
2014-07-11 12:09:38 +00:00
number_periods = p_profile - > FirstDay ( ) . daysTo ( p_profile - > LastDay ( ) ) / 30 ;
2014-05-28 19:26:05 +00:00
if ( number_periods > 12 ) {
number_periods = 12 ;
}
2014-05-06 11:54:41 +00:00
}
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-28 19:26:05 +00:00
2014-05-06 09:11:31 +00:00
bool skipsection = false ; ;
2014-05-06 18:03:13 +00:00
for ( QList < StatisticsRow > : : iterator i = rows . begin ( ) ; i ! = rows . end ( ) ; + + i ) {
2014-05-06 09:11:31 +00:00
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 ) ;
2014-05-28 19:26:05 +00:00
first = p_profile - > FirstGoodDay ( row . type ) ;
2013-09-14 23:32:14 +00:00
2014-05-06 11:54:41 +00:00
periods . clear ( ) ;
2014-05-28 19:26:05 +00:00
if ( p_profile - > general - > statReportMode ( ) = = 0 ) {
2014-05-06 11:54:41 +00:00
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 ;
2014-05-28 19:26:05 +00:00
int j = 0 ;
do {
2014-05-06 11:54:41 +00:00
s = QDate ( l . year ( ) , l . month ( ) , 1 ) ;
if ( s < first ) {
done = true ;
s = first ;
}
2014-05-28 19:26:05 +00:00
if ( p_profile - > countDays ( row . type , s , l ) > 0 ) {
2014-05-06 11:54:41 +00:00
periods . push_back ( Period ( s , l , s . toString ( " MMMM " ) ) ) ;
2014-05-28 19:26:05 +00:00
j + + ;
2014-05-06 11:54:41 +00:00
}
2014-05-28 19:26:05 +00:00
l = s . addDays ( - 1 ) ;
} while ( ( l > first ) & & ( j < number_periods ) ) ;
// for (; 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")));
// } else {
// }
// l = s.addDays(-1);
// if (done || (l < first)) break;
// }
for ( ; j < number_periods ; + + j ) {
periods . push_back ( Period ( last , last , " " ) ) ;
2014-05-06 11:54:41 +00:00
}
}
2014-04-17 05:52:25 +00:00
2014-07-11 12:09:38 +00:00
int days = p_profile - > countDays ( row . type , first , last ) ;
2014-05-06 09:11:31 +00:00
skipsection = ( days = = 0 ) ;
if ( days > 0 ) {
2014-05-28 13:09:03 +00:00
html + = QString ( " <tr bgcolor='%1'><th colspan=%2 align=center><font size=+2>%3</font></th></tr> \n " ) .
2014-05-06 11:54:41 +00:00
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-15 04:26:48 +00:00
html + = QString ( " <tr><td><b>%1</b></td> \n " ) . arg ( tr ( " Details " ) ) ;
2014-05-06 11:54:41 +00:00
for ( int j = 0 ; j < periods . size ( ) ; j + + ) {
2014-05-15 04:26:48 +00:00
html + = QString ( " <td onmouseover='ChangeColor(this, \" #eeeeee \" );' onmouseout='ChangeColor(this, \" #ffffff \" );' onclick='alert( \" overview=%1,%2 \" );'><b>%3</b></td> \n " ) . 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 ) .
2014-05-14 13:35:55 +00:00
arg ( QString ( tr ( " %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 ) .
2014-05-14 13:35:55 +00:00
arg ( QString ( tr ( " %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 ( ) ;
2014-09-11 14:23:08 +00:00
if ( ( id = = NoChannel ) | | ( ! p_profile - > channelAvailable ( id ) ) ) {
2014-05-06 09:11:31 +00:00
continue ;
}
2014-05-14 13:35:55 +00:00
name = calcnames [ row . calc ] . arg ( schema : : channel [ id ] . fullname ( ) ) ;
2013-09-14 23:32:14 +00:00
}
2014-05-15 22:48:51 +00:00
QString line ;
2014-05-28 19:26:05 +00:00
line + = QString ( " <tr class=datarow><td width=25%>%1</td> " ) . arg ( name ) ;
int np = periods . size ( ) ;
int width ;
for ( int j = 0 ; j < np ; j + + ) {
if ( p_profile - > general - > statReportMode ( ) = = 1 ) {
width = j < np - 1 ? 6 : 100 - ( 25 + 6 * ( np - 1 ) ) ;
} else {
width = 75 / np ;
}
line + = QString ( " <td width=%1%> " ) . arg ( width ) ;
if ( ! periods . at ( j ) . header . isEmpty ( ) ) {
line + = row . value ( periods . at ( j ) . start , periods . at ( j ) . end ) ;
} else {
line + = " " ;
}
line + = " </td> " ;
2014-05-06 11:54:41 +00:00
}
2014-05-15 22:48:51 +00:00
html + = line ;
2014-05-06 11:54:41 +00:00
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
2014-09-11 14:23:08 +00:00
/*
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-08-03 13:00:13 +00:00
QString last_prel_str , last_pressure_str , prel_str , pressure_str ;
2014-04-17 05:52:25 +00:00
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-07-11 12:09:38 +00:00
day = p_profile - > GetGoodDay ( date , MT_CPAP ) ;
2013-09-14 23:32:14 +00:00
2014-08-03 13:00:13 +00:00
CPAPLoader * loader = nullptr ;
2014-08-20 17:17:13 +00:00
if ( day ) loader = dynamic_cast < CPAPLoader * > ( day - > machine ( MT_CPAP ) - > loader ( ) ) ;
2014-08-03 13:00:13 +00:00
if ( day & & loader ) {
2014-04-17 05:52:25 +00:00
lastchanged = false ;
2013-09-14 23:32:14 +00:00
2014-08-20 17:17:13 +00:00
hours = day - > hours ( MT_CPAP ) ;
2013-09-14 23:32:14 +00:00
2014-07-11 12:09:38 +00:00
if ( hours > p_profile - > cpap - > complianceHours ( ) ) {
2013-09-14 23:32:14 +00:00
compliant + + ;
2014-04-17 05:52:25 +00:00
}
2014-08-03 13:00:13 +00:00
EventDataType ahi = day - > count ( CPAP_Obstructive ) + day - > count ( CPAP_Hypopnea ) + day - > count ( CPAP_Apnea ) + day - > count ( CPAP_ClearAirway ) ;
2014-04-17 05:52:25 +00:00
2014-07-11 12:09:38 +00:00
if ( p_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-08-03 13:00:13 +00:00
prel_str = day - > getPressureRelief ( ) ;
pressure_str = day - > getPressureSettings ( ) ;
2014-04-17 05:52:25 +00:00
mode = ( CPAPMode ) ( int ) round ( day - > settings_wavg ( CPAP_Mode ) ) ;
2014-08-03 13:00:13 +00:00
if ( mode = = 0 ) {
mode = ( CPAPMode ) ( int ) round ( day - > settings_wavg ( CPAP_Mode ) ) ;
}
2014-08-20 17:17:13 +00:00
mach = day - > machine ( MT_CPAP ) ;
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 ) ;
2014-07-25 07:53:48 +00:00
} else if ( mode = = MODE_APAP ) {
2014-04-17 05:52:25 +00:00
min = day - > settings_min ( CPAP_PressureMin ) ;
max = day - > settings_max ( CPAP_PressureMax ) ;
2014-07-25 07:53:48 +00:00
} else if ( mode = = MODE_BILEVEL_FIXED ) {
min = day - > settings_min ( CPAP_EPAP ) ;
max = day - > settings_max ( CPAP_IPAP ) ;
ps = max - min ;
} else if ( mode = = MODE_BILEVEL_AUTO_FIXED_PS ) {
min = day - > settings_min ( CPAP_EPAPLo ) ;
maxipap = max = day - > settings_max ( CPAP_IPAPHi ) ;
ps = day - > settings_min ( CPAP_PS ) ;
2014-08-03 13:00:13 +00:00
} else if ( mode = = MODE_BILEVEL_AUTO_VARIABLE_PS ) { // Similar pressure control as ASV Variable EPAP
min = day - > settings_min ( CPAP_EPAPLo ) ;
max = day - > settings_min ( CPAP_EPAPHi ) ;
ps = day - > settings_min ( CPAP_PSMin ) ;
pshi = day - > settings_max ( CPAP_PSMax ) ;
maxipap = max = day - > settings_max ( CPAP_IPAPHi ) ;
2014-07-25 07:53:48 +00:00
} else if ( mode = = MODE_ASV ) {
2014-08-21 06:21:37 +00:00
min = day - > settings_min ( CPAP_EPAP ) ;
2014-07-25 07:53:48 +00:00
ps = day - > settings_min ( CPAP_PSMin ) ;
pshi = day - > settings_max ( CPAP_PSMax ) ;
maxipap = min + pshi ;
} else if ( mode = = MODE_ASV_VARIABLE_EPAP ) {
min = day - > settings_min ( CPAP_EPAPLo ) ;
max = day - > settings_max ( CPAP_EPAPHi ) ;
ps = day - > settings_min ( CPAP_PSMin ) ;
pshi = day - > settings_max ( CPAP_PSMax ) ;
maxipap = max + pshi ;
2014-08-03 13:00:13 +00:00
2013-09-14 23:32:14 +00:00
}
2013-11-19 13:06:17 +00:00
2014-08-03 13:00:13 +00:00
if ( lastmach = = nullptr ) {
lastmach = mach ;
cmode = mode ;
last_pressure_str = pressure_str ;
last_prel_str = prel_str ;
}
2014-04-17 05:52:25 +00:00
2014-08-03 13:00:13 +00:00
if ( ( mode ! = cmode ) | | ( pressure_str ! = last_pressure_str ) | | ( prel_str ! = last_prel_str ) | | ( mach ! = lastmach ) ) {
first = date . addDays ( 1 ) ;
int days = p_profile - > countDays ( MT_CPAP , first , last ) ;
RXChange rx ;
rx . first = first ;
rx . last = last ;
rx . days = days ;
rx . ahi = calcAHI ( first , last ) ;
rx . fl = calcFL ( first , last ) ;
rx . mode = cmode ;
rx . pressure_string = last_pressure_str ;
rx . pr_relief_string = last_prel_str ;
rx . min = cmin ;
rx . max = cmax ;
rx . ps = cps ;
rx . pshi = cpshi ;
rx . maxipap = cmaxipap ;
rx . machine = lastmach ;
rx . per1 = 0 ;
rx . per2 = 0 ;
if ( mode = = MODE_APAP ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_Pressure , percentile , MT_CPAP , first , last ) ;
rx . per2 = 0 ;
} else if ( mode = = MODE_BILEVEL_AUTO_FIXED_PS ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_IPAP , percentile , MT_CPAP , first , last ) ;
} else if ( mode = = MODE_BILEVEL_AUTO_VARIABLE_PS ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_PS , percentile , MT_CPAP , first , last ) ;
} else if ( mode = = MODE_ASV ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_PS , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_PS , percentile , MT_CPAP , first , last ) ;
} else if ( mode = = MODE_ASV_VARIABLE_EPAP ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_PS , percentile , MT_CPAP , first , last ) ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
2014-08-03 13:00:13 +00:00
rx . weighted = float ( rx . days ) / float ( cpapdays ) * rx . ahi ;
rxchange . push_back ( rx ) ;
2014-04-17 05:52:25 +00:00
cmode = mode ;
cmin = min ;
cmax = max ;
cps = ps ;
cpshi = pshi ;
cmaxipap = maxipap ;
2014-08-03 13:00:13 +00:00
last_prel_str = prel_str ;
last_pressure_str = pressure_str ;
2014-04-17 05:52:25 +00:00
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-08-03 13:00:13 +00:00
lastchanged = false ;
2014-04-17 05:52:25 +00:00
2014-08-03 13:00:13 +00:00
// Add the final entry
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 ;
2014-07-11 12:09:38 +00:00
int days = p_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 . machine = mach ;
2014-08-03 13:00:13 +00:00
rx . pressure_string = pressure_str ;
rx . pr_relief_string = prel_str ;
2014-07-25 07:53:48 +00:00
2014-08-03 13:00:13 +00:00
// Todo: Clean up by Calculating this crap later..
2014-07-25 07:53:48 +00:00
if ( mode = = MODE_APAP ) {
2014-04-17 05:52:25 +00:00
rx . per1 = p_profile - > calcPercentile ( CPAP_Pressure , percentile , MT_CPAP , first , last ) ;
rx . per2 = 0 ;
2014-07-25 07:53:48 +00:00
} else if ( mode = = MODE_BILEVEL_AUTO_FIXED_PS ) {
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 ) ;
2014-08-03 13:00:13 +00:00
} else if ( mode = = MODE_BILEVEL_AUTO_VARIABLE_PS ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_PS , percentile , MT_CPAP , first , last ) ;
2014-07-25 07:53:48 +00:00
} else if ( mode = = MODE_ASV ) {
rx . per1 = p_profile - > calcPercentile ( CPAP_PS , percentile , MT_CPAP , first , last ) ;
rx . per2 = p_profile - > calcPercentile ( CPAP_PS , percentile , MT_CPAP , first , last ) ;
} else if ( mode = = MODE_ASV_VARIABLE_EPAP ) {
2014-04-17 05:52:25 +00:00
rx . per1 = p_profile - > calcPercentile ( CPAP_EPAP , percentile , MT_CPAP , first , last ) ;
2014-07-25 07:53:48 +00:00
rx . per2 = p_profile - > calcPercentile ( CPAP_PS , 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 ] ;
2014-05-28 16:23:40 +00:00
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 ) ;
2014-07-11 12:09:38 +00:00
if ( p_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 ;
qSort ( tmpRX . begin ( ) , tmpRX . end ( ) , RXSort ) ;
tmpRX [ 0 ] - > highlight = 4 ; // worst
int ls = tmpRX . size ( ) - 1 ;
tmpRX [ ls ] - > highlight = 1 ; //best
2014-07-25 07:53:48 +00:00
QString modestr ;
2013-11-19 13:06:17 +00:00
2014-07-25 07:53:48 +00:00
QString idxstr [ 2 ] ;
int idx [ 2 ] ;
idx [ 0 ] = ls ;
idx [ 1 ] = 0 ;
idxstr [ 0 ] = tr ( " Best RX Setting " ) ;
idxstr [ 1 ] = tr ( " Worst RX Setting " ) ;
2013-09-14 23:32:14 +00:00
2014-07-25 07:53:48 +00:00
for ( int i = 0 ; i < 2 ; + + i ) {
int i2 = idx [ i ] ;
if ( ( i = = 1 ) & & ( ls = = 0 ) ) break ;
CPAPMode mode = ( CPAPMode ) ( int ) p_profile - > calcSettingsMax ( CPAP_Mode , MT_CPAP , tmpRX [ i2 ] - > first , tmpRX [ i2 ] - > last ) ;
2014-04-17 05:52:25 +00:00
2014-07-25 07:53:48 +00:00
if ( mode = = MODE_CPAP ) {
modestr = STR_TR_CPAP ;
} else if ( mode = = MODE_APAP ) { // is AUTO?
modestr = STR_TR_APAP ;
} else if ( mode = = MODE_BILEVEL_FIXED ) {
modestr = STR_TR_BiLevel ;
} else if ( mode = = MODE_BILEVEL_AUTO_FIXED_PS ) {
modestr = QObject : : tr ( " Auto Bi-Level " ) ;
2014-08-03 13:00:13 +00:00
} else if ( mode = = MODE_BILEVEL_AUTO_VARIABLE_PS ) {
modestr = QObject : : tr ( " Auto Bi-Level " ) ;
2014-07-25 07:53:48 +00:00
} else if ( mode = = MODE_ASV ) {
2014-08-03 13:00:13 +00:00
modestr = QObject : : tr ( " ASV Fixed EPAP " ) ;
2014-07-25 07:53:48 +00:00
} else if ( mode = = MODE_ASV_VARIABLE_EPAP ) {
2014-08-03 13:00:13 +00:00
modestr = QObject : : tr ( " ASV Auto EPAP " ) ;
2014-07-25 07:53:48 +00:00
} else modestr = STR_TR_Unknown ;
2014-08-03 13:00:13 +00:00
2014-07-25 07:53:48 +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 ( idxstr [ i ] ) ;
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 [ i2 ] - > first . toString ( Qt : : ISODate ) )
. arg ( tmpRX [ i2 ] - > last . toString ( Qt : : ISODate ) )
. arg ( tmpRX [ i2 ] - > first . toString ( Qt : : SystemLocaleShortDate ) )
. arg ( tmpRX [ i2 ] - > last . toString ( Qt : : SystemLocaleShortDate ) ) ;
recbox + = QString ( " <tr><td><b>%1</b></td><td align=right><b>%2</b></td></tr> " ) . arg ( ahitxt ) . arg ( tmpRX [ i2 ] - > ahi , 0 , ' f ' , decimals ) ;
recbox + = QString ( " <tr><td>%1</td><td align=right>%2</td></tr> " ) . arg ( STR_TR_Mode ) . arg ( modestr ) ;
if ( mode = = MODE_CPAP ) {
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " Fixed " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > min , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
} else if ( mode = = MODE_APAP ) {
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " Min " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > min , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " Max " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > max , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
} else if ( mode = = MODE_BILEVEL_FIXED ) {
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " EPAP " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > min , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " IPAP " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > max , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
} else if ( mode = = MODE_BILEVEL_AUTO_FIXED_PS ) {
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " Min EPAP " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > min , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " Max IPAP " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > max , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " PS " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > ps , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
} else if ( mode = = MODE_ASV ) {
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " EPAP " ) + QString ( " </td><td align=right>%1%2</td></tr> " ) . arg ( tmpRX [ i2 ] - > min , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " PS " ) + QString ( " </td><td align=right>%1-%2%3</td></tr> " ) . arg ( tmpRX [ i2 ] - > ps , 0 , ' f ' , 1 ) . arg ( tmpRX [ i2 ] - > pshi , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
// recbox += QString("<tr><td>")+QObject::tr("Max IPAP")+QString("</td><td align=right>%1%2</td></tr>").arg(tmpRX[i2]->maxipap, 0, 'f', 1).arg(STR_UNIT_CMH2O);
} else if ( mode = = MODE_ASV_VARIABLE_EPAP ) {
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " EPAP " ) + QString ( " </td><td align=right>%1-%2%3</td></tr> " ) . arg ( tmpRX [ i2 ] - > min , 0 , ' f ' , 1 ) . arg ( tmpRX [ i2 ] - > max , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
recbox + = QString ( " <tr><td> " ) + QObject : : tr ( " PS " ) + QString ( " </td><td align=right>%1-%2%3</td></tr> " ) . arg ( tmpRX [ i2 ] - > ps , 0 , ' f ' , 1 ) . arg ( tmpRX [ i2 ] - > pshi , 0 , ' f ' , 1 ) . arg ( STR_UNIT_CMH2O ) ;
// recbox += QString("<tr><td>")+QObject::tr("Max IPAP")+QString("</td><td align=right>%1%2</td></tr>").arg(tmpRX[i2]->maxipap, 0, 'f', 1).arg(STR_UNIT_CMH2O);
}
recbox + = " </table><br/></td></tr> " ;
2013-09-14 23:32:14 +00:00
}
}
2014-04-17 05:52:25 +00:00
recbox + = " </table> " ;
recbox + = " </body></html> " ;
2014-09-11 14:23:08 +00:00
mainwin - > setRecBoxHTML ( recbox ) ; */
2013-09-14 23:32:14 +00:00
/*RXsort=RX_min;
RXorder = true ;
qSort ( rxchange . begin ( ) , rxchange . end ( ) ) ; */
2014-05-28 09:35:21 +00:00
2014-09-11 14:23:08 +00:00
/* html += "<div align=center><br/>";
2014-07-25 07:53:48 +00:00
html + = QString ( " <table class=curved style= \" page-break-before:always; \" " + table_width + " > " ) ;
2014-05-28 09:35:21 +00:00
html + = " <thead> " ;
2014-05-28 13:09:03 +00:00
html + = " <tr bgcolor=' " + heading_color + " '><th colspan=10 align=center><font size=+2> " + tr ( " Changes to Prescription Settings " ) + " </font></th></tr> " ;
2014-05-28 09:35:21 +00:00
2013-09-14 23:32:14 +00:00
QString extratxt ;
QString tooltip ;
2014-05-15 04:26:48 +00:00
QStringList hdrlist ;
hdrlist . push_back ( STR_TR_First ) ;
hdrlist . push_back ( STR_TR_Last ) ;
hdrlist . push_back ( tr ( " Days " ) ) ;
hdrlist . push_back ( ahitxt ) ;
hdrlist . push_back ( STR_TR_FL ) ;
2014-07-11 12:09:38 +00:00
if ( p_profile - > hasChannel ( CPAP_SensAwake ) ) {
2014-05-15 04:26:48 +00:00
hdrlist . push_back ( STR_TR_SA ) ;
}
hdrlist . push_back ( STR_TR_Machine ) ;
2014-08-03 13:00:13 +00:00
hdrlist . push_back ( tr ( " Pressure Relief " ) ) ;
2014-05-15 04:26:48 +00:00
hdrlist . push_back ( STR_TR_Mode ) ;
hdrlist . push_back ( tr ( " Pressure Settings " ) ) ;
html + = " <tr> \n " ;
for ( int i = 0 ; i < hdrlist . size ( ) ; + + i ) {
2014-05-28 09:35:21 +00:00
html + = QString ( " <th align=left><b>%1</b></th> \n " ) . arg ( hdrlist . at ( i ) ) ;
2014-05-15 04:26:48 +00:00
}
html + = " </tr> \n " ;
2014-05-28 09:35:21 +00:00
html + = " </thead> " ;
html + = " <tfoot> " ;
html + = " <tr><td colspan=10 align=center> " ;
html + = QString ( " <i> " ) +
tr ( " Efficacy highlighting ignores prescription settings with less than %1 days of recorded data. " ) .
arg ( rxthresh ) + QString ( " </i><br/> " ) ;
html + = " </td></tr> " ;
html + = " </tfoot> " ;
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
2014-07-28 13:56:29 +00:00
if ( ! rx . machine - > brand ( ) . isEmpty ( ) ) {
machstr + = rx . machine - > brand ( ) ;
2014-04-17 05:52:25 +00:00
}
2014-07-28 13:56:29 +00:00
if ( ! rx . machine - > model ( ) . isEmpty ( ) ) {
machstr + = " " + rx . machine - > model ( ) ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
2014-07-28 13:56:29 +00:00
if ( ! rx . machine - > serial ( ) . isEmpty ( ) ) {
machstr + = " ( " + rx . machine - > serial ( ) + " )<br/> " ;
2013-09-14 23:32:14 +00:00
}
2014-04-17 05:52:25 +00:00
mode = rx . mode ;
2014-05-26 19:29:01 +00:00
extratxt = " <table border=0 width=100%><tr> " ; //cellpadding=0 cellspacing=0
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);
2014-07-25 07:53:48 +00:00
tooltip = QString ( " %1 " ) . arg ( machstr ) ;
2014-04-17 05:52:25 +00:00
2014-08-03 13:00:13 +00:00
extratxt + = " <td colspan=2> " + rx . pressure_string + " </td> " ;
// if (mode == MODE_CPAP) {
// extratxt += "<td colspan=2>"+QString(tr("Fixed %1 %2")+"</td>").arg(rx.min, 4, 'f', 1).arg(schema::channel[CPAP_Pressure].units());
// } else if (mode == MODE_APAP) {
// extratxt += "<td colspan=2>"+QString(tr("%1 - %2 %3")+"</td>").arg(rx.min, 4, 'f', 1).arg(rx.max, 4, 'f', 1).arg(schema::channel[CPAP_Pressure].units());
// } else if (mode == MODE_BILEVEL_FIXED) {
// extratxt += "<td colspan=2>"+QString(tr("EPAP %1 %3 IPAP %2 %3")+"</td>").arg(rx.min, 4, 'f', 1).arg(rx.max, 4, 'f', 1).arg(schema::channel[CPAP_Pressure].units());
// } else if (mode == MODE_BILEVEL_AUTO_FIXED_PS) {
// extratxt += "<td colspan=2>"+QString(tr("PS %4 over %1 - %2 %3")+"</td>").arg(rx.min, 4, 'f', 1).arg(rx.max, 4, 'f', 1).arg(schema::channel[CPAP_Pressure].units()).arg(rx.ps, 4, 'f', 1);
// } else if (mode == MODE_BILEVEL_AUTO_VARIABLE_PS) {
// extratxt += "<td colspan=2>"+QString(tr("EPAP %1-%2, PS %3-%4 %5")+"</td>").arg(rx.min, 4, 'f', 1).arg(rx.max, 4, 'f', 1).arg(rx.ps, 4, 'f', 1).arg(rx.pshi, 4, 'f', 1).arg(schema::channel[CPAP_Pressure].units());
// } else if (mode == MODE_ASV) {
// extratxt += "<td colspan=2>"+QString(tr("EPAP %1, PS %2-%3 %4")+"</td>").arg(rx.min, 4, 'f', 1).arg(rx.ps, 4, 'f', 1).arg(rx.pshi, 4, 'f', 1).arg(schema::channel[CPAP_Pressure].units());
// 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_ASV_VARIABLE_EPAP) {
// extratxt += "<td colspan=2>"+QString(tr("EPAP %1-%2, PS %3-%4 %5")+"</td>").arg(rx.min, 4, 'f', 1).arg(rx.max, 4, 'f', 1).arg(rx.ps, 4, 'f', 1).arg(rx.pshi, 4, 'f', 1).arg(schema::channel[CPAP_Pressure].units());
// 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 {
// extratxt += "";
// tooltip = "";
// }
2014-04-17 05:52:25 +00:00
2014-08-03 13:00:13 +00:00
extratxt + = " </tr></table> " ;
2014-04-17 05:52:25 +00:00
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
2014-05-31 21:32:11 +00:00
QString datarowclass ;
if ( rx . highlight = = 0 ) datarowclass = " class=datarow " ;
html + = QString ( " <tr %6 bgcolor='%1' onmouseover='ChangeColor(this, \" #eeeeee \" ); %2' onmouseout='ChangeColor(this, \" %1 \" ); %3' onclick='alert( \" overview=%4,%5 \" );'> " )
2014-05-15 04:26:48 +00:00
. arg ( color )
2013-09-14 23:32:14 +00:00
. arg ( tooltipshow )
2014-05-15 04:26:48 +00:00
. arg ( tooltiphide )
. arg ( rx . first . toString ( Qt : : ISODate ) )
2014-05-31 21:32:11 +00:00
. arg ( rx . last . toString ( Qt : : ISODate ) )
. arg ( datarowclass ) ;
2014-05-15 04:26:48 +00:00
html + = QString ( " <td>%1</td> " ) . arg ( rx . first . toString ( Qt : : SystemLocaleShortDate ) ) ;
html + = QString ( " <td>%1</td> " ) . arg ( rx . last . toString ( Qt : : SystemLocaleShortDate ) ) ;
html + = QString ( " <td>%1</td> " ) . arg ( rx . days ) ;
html + = QString ( " <td>%1</td> " ) . arg ( rx . ahi , 0 , ' f ' , decimals ) ;
html + = QString ( " <td>%1</td> " ) . arg ( rx . fl , 0 , ' f ' , decimals ) ; // Not the best way to do this.. Todo: Add an extra field for data..
2014-07-11 12:09:38 +00:00
if ( p_profile - > hasChannel ( CPAP_SensAwake ) ) {
2014-05-18 17:53:09 +00:00
html + = QString ( " <td>%1</td> " ) . arg ( calcSA ( rx . first , rx . last ) , 0 , ' f ' , decimals ) ;
2014-05-15 04:26:48 +00:00
}
2014-07-28 13:56:29 +00:00
html + = QString ( " <td>%1</td> " ) . arg ( rx . machine - > loaderName ( ) ) ;
2014-08-03 13:00:13 +00:00
html + = QString ( " <td>%1</td> " ) . arg ( rx . pr_relief_string ) ;
html + = QString ( " <td>%1</td> " ) . arg ( schema : : channel [ CPAP_Mode ] . option ( int ( rx . mode ) ) ) ;
2014-05-15 04:26:48 +00:00
html + = QString ( " <td>%1</td> " ) . arg ( extratxt ) ;
html + = " </tr> " ;
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
2014-09-11 14:23:08 +00:00
} */
html + = GenerateRXChanges ( ) ;
html + = GenerateMachineList ( ) ;
2014-05-06 09:11:31 +00:00
2014-09-30 08:40:12 +00:00
UpdateRecordsBox ( ) ;
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
2014-09-30 08:40:12 +00:00
void Statistics : : UpdateRecordsBox ( )
{
QString html = " <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> " ;
Machine * cpap = p_profile - > GetMachine ( MT_CPAP ) ;
if ( cpap ) {
QDate first = p_profile - > FirstGoodDay ( MT_CPAP ) ;
QDate last = p_profile - > LastGoodDay ( MT_CPAP ) ;
/////////////////////////////////////////////////////////////////////////////////////
/// Compliance and usage information
/////////////////////////////////////////////////////////////////////////////////////
int totaldays = p_profile - > countDays ( MT_CPAP , first , last ) ;
int compliant = p_profile - > countCompliantDays ( MT_CPAP , first , last ) ;
float comperc = ( 100.0 / float ( totaldays ) ) * float ( compliant ) ;
html + = " <b> " + QObject : : tr ( " CPAP Usage " ) + " </b><br/> " ;
html + = QObject : : tr ( " Days Used: %1 " ) . arg ( totaldays ) + " <br/> " ;
html + = QObject : : tr ( " Low Use Days: %1 " ) . arg ( totaldays - compliant ) + " <br/> " ;
html + = QObject : : tr ( " Compliance: %1% " ) . arg ( comperc , 0 , ' f ' , 1 ) + " <br/> " ;
/////////////////////////////////////////////////////////////////////////////////////
/// AHI Records
/////////////////////////////////////////////////////////////////////////////////////
if ( p_profile - > session - > preloadSummaries ( ) ) {
const int show_records = 5 ;
QMultiMap < float , QDate > : : iterator it ;
QMultiMap < float , QDate > : : iterator it_end ;
QMultiMap < float , QDate > ahilist ;
int baddays = 0 ;
for ( QDate date = first ; date < = last ; date = date . addDays ( 1 ) ) {
Day * day = p_profile - > GetGoodDay ( date , MT_CPAP ) ;
if ( ! day ) continue ;
float ahi = day - > calcAHI ( ) ;
if ( ahi > = 5 ) {
baddays + + ;
}
ahilist . insert ( ahi , date ) ;
}
html + = QObject : : tr ( " Days AHI of 5 or greater: %1 " ) . arg ( baddays ) + " <br/><br/> " ;
if ( ahilist . size ( ) > ( show_records * 2 ) ) {
it = ahilist . begin ( ) ;
it_end = ahilist . end ( ) ;
html + = " <b> " + QObject : : tr ( " Best AHI " ) + " </b><br/> " ;
for ( int i = 0 ; ( i < show_records ) & & ( it ! = it_end ) ; + + i , + + it ) {
html + = QString ( " <a href='daily=%1'> " ) . arg ( it . value ( ) . toString ( Qt : : ISODate ) )
+ QObject : : tr ( " Date: %1 AHI: %2 " ) . arg ( it . value ( ) . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( it . key ( ) , 0 , ' f ' , 2 ) + " </a><br/> " ;
}
html + = " <br/> " ;
html + = " <b> " + QObject : : tr ( " Worst AHI " ) + " </b><br/> " ;
it = ahilist . end ( ) - 1 ;
it_end = ahilist . begin ( ) ;
for ( int i = 0 ; ( i < show_records ) & & ( it ! = it_end ) ; + + i , - - it ) {
html + = QString ( " <a href='daily=%1'> " ) . arg ( it . value ( ) . toString ( Qt : : ISODate ) )
+ QObject : : tr ( " Date: %1 AHI: %2 " ) . arg ( it . value ( ) . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( it . key ( ) , 0 , ' f ' , 2 ) + " </a><br/> " ;
}
html + = " <br/> " ;
}
/////////////////////////////////////////////////////////////////////////////////////
/// Flow Limitation Records
/////////////////////////////////////////////////////////////////////////////////////
ahilist . clear ( ) ;
for ( QDate date = first ; date < = last ; date = date . addDays ( 1 ) ) {
Day * day = p_profile - > GetGoodDay ( date , MT_CPAP ) ;
if ( ! day ) continue ;
float val = 0 ;
if ( day - > channelHasData ( CPAP_FlowLimit ) ) {
val = day - > calcIdx ( CPAP_FlowLimit ) ;
} else if ( day - > channelHasData ( CPAP_FLG ) ) {
// Use 90th percentile
val = day - > calcPercentile ( CPAP_FLG ) ;
}
ahilist . insert ( val , date ) ;
}
int cnt = 0 ;
if ( ahilist . size ( ) > ( show_records * 2 ) ) {
it = ahilist . begin ( ) ;
it_end = ahilist . end ( ) ;
html + = " <b> " + QObject : : tr ( " Best Flow Limitation " ) + " </b><br/> " ;
for ( int i = 0 ; ( i < show_records ) & & ( it ! = it_end ) ; + + i , + + it ) {
html + = QString ( " <a href='daily=%1'> " ) . arg ( it . value ( ) . toString ( Qt : : ISODate ) )
+ QObject : : tr ( " Date: %1 FL: %2 " ) . arg ( it . value ( ) . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( it . key ( ) , 0 , ' f ' , 2 ) + " </a><br/> " ;
}
html + = " <br/> " ;
html + = " <b> " + QObject : : tr ( " Worst Flow Limtation " ) + " </b><br/> " ;
it = ahilist . end ( ) - 1 ;
it_end = ahilist . begin ( ) ;
for ( int i = 0 ; ( i < show_records ) & & ( it ! = it_end ) ; + + i , - - it ) {
if ( it . key ( ) > 0 ) {
html + = QString ( " <a href='daily=%1'> " ) . arg ( it . value ( ) . toString ( Qt : : ISODate ) )
+ QObject : : tr ( " Date: %1 FL: %2 " ) . arg ( it . value ( ) . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( it . key ( ) , 0 , ' f ' , 2 ) + " </a><br/> " ;
cnt + + ;
}
}
if ( cnt = = 0 ) {
html + = " <i> " + QObject : : tr ( " No Flow Limitation on record " ) + " </i><br/> " ;
}
html + = " <br/> " ;
}
/////////////////////////////////////////////////////////////////////////////////////
/// Large Leak Records
/////////////////////////////////////////////////////////////////////////////////////
ahilist . clear ( ) ;
for ( QDate date = first ; date < = last ; date = date . addDays ( 1 ) ) {
Day * day = p_profile - > GetGoodDay ( date , MT_CPAP ) ;
if ( ! day ) continue ;
float leak = day - > calcPON ( CPAP_LargeLeak ) ;
ahilist . insert ( leak , date ) ;
}
cnt = 0 ;
if ( ahilist . size ( ) > ( show_records * 2 ) ) {
html + = " <b> " + QObject : : tr ( " Worst Large Leaks " ) + " </b><br/> " ;
it = ahilist . end ( ) - 1 ;
it_end = ahilist . begin ( ) ;
for ( int i = 0 ; ( i < show_records ) & & ( it ! = it_end ) ; + + i , - - it ) {
if ( it . key ( ) > 0 ) {
html + = QString ( " <a href='daily=%1'> " ) . arg ( it . value ( ) . toString ( Qt : : ISODate ) )
+ QObject : : tr ( " Date: %1 Leak: %2% " ) . arg ( it . value ( ) . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( it . key ( ) , 0 , ' f ' , 2 ) + " </a><br/> " ;
cnt + + ;
}
}
if ( cnt = = 0 ) {
html + = " <i> " + QObject : : tr ( " No Large Leaks on record " ) + " </i><br/> " ;
}
html + = " <br/> " ;
}
/////////////////////////////////////////////////////////////////////////////////////
/// ÇSR Records
/////////////////////////////////////////////////////////////////////////////////////
cnt = 0 ;
if ( p_profile - > hasChannel ( CPAP_CSR ) ) {
ahilist . clear ( ) ;
for ( QDate date = first ; date < = last ; date = date . addDays ( 1 ) ) {
Day * day = p_profile - > GetGoodDay ( date , MT_CPAP ) ;
if ( ! day ) continue ;
float leak = day - > calcPON ( CPAP_CSR ) ;
ahilist . insert ( leak , date ) ;
}
if ( ahilist . size ( ) > ( show_records * 2 ) ) {
html + = " <b> " + QObject : : tr ( " Worst CSR " ) + " </b><br/> " ;
it = ahilist . end ( ) - 1 ;
it_end = ahilist . begin ( ) ;
for ( int i = 0 ; ( i < show_records ) & & ( it ! = it_end ) ; + + i , - - it ) {
if ( it . key ( ) > 0 ) {
html + = QString ( " <a href='daily=%1'> " ) . arg ( it . value ( ) . toString ( Qt : : ISODate ) )
+ QObject : : tr ( " Date: %1 CSR: %2% " ) . arg ( it . value ( ) . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( it . key ( ) , 0 , ' f ' , 2 ) + " </a><br/> " ;
cnt + + ;
}
}
if ( cnt = = 0 ) {
html + = " <i> " + QObject : : tr ( " No CSR on record " ) + " </i><br/> " ;
}
html + = " <br/> " ;
}
}
} else {
html + = " <br/><b> " + QObject : : tr ( " Want more information? " ) + " </b><br/> " ;
html + = " <i> " + QObject : : tr ( " SleepyHead needs all summary data loaded to calculate best/worst data for individual days. " ) + " </i><br/><br/> " ;
html + = " <i> " + QObject : : tr ( " Please enable Pre-Load Summaries checkbox in preferences to make sure this data is available. " ) + " </i><br/><br/> " ;
}
/////////////////////////////////////////////////////////////////////////////////////
/// Sort the RX list to get best and worst settings.
/////////////////////////////////////////////////////////////////////////////////////
QList < RXItem * > list ;
QMap < QDate , RXItem > : : iterator ri_end = rxitems . end ( ) ;
QMap < QDate , RXItem > : : iterator ri ;
for ( ri = rxitems . begin ( ) ; ri ! = ri_end ; + + ri ) {
list . append ( & ri . value ( ) ) ;
ri . value ( ) . highlight = 0 ;
}
qSort ( list . begin ( ) , list . end ( ) , rxAHILessThan ) ;
if ( list . size ( ) > = 2 ) {
html + = " <b> " + QObject : : tr ( " Best RX Setting " ) + " </b><br/> " ;
const RXItem & rxbest = * list . at ( 0 ) ;
html + = QString ( " <a href='overview=%1,%2'> " ) . arg ( rxbest . start . toString ( Qt : : ISODate ) ) . arg ( rxbest . end . toString ( Qt : : ISODate ) ) +
QObject : : tr ( " Date: %1 - %2 " ) . arg ( rxbest . start . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( rxbest . end . toString ( Qt : : SystemLocaleShortDate ) ) + " </a><br/> " ;
html + = QString ( " %1 " ) . arg ( rxbest . machine - > model ( ) ) + " <br/> " ;
html + = QString ( " Serial: %1 " ) . arg ( rxbest . machine - > serial ( ) ) + " <br/> " ;
html + = QObject : : tr ( " Culminative AHI: %1 " ) . arg ( double ( rxbest . ahi ) / rxbest . hours , 0 , ' f ' , 2 ) + " <br/> " ;
html + = QObject : : tr ( " Culminative Hours: %1 " ) . arg ( rxbest . hours , 0 , ' f ' , 2 ) + " <br/> " ;
html + = QString ( " %1 " ) . arg ( rxbest . pressure ) + " <br/> " ;
html + = QString ( " %1 " ) . arg ( rxbest . relief ) + " <br/> " ;
html + = " <br/> " ;
html + = " <b> " + QObject : : tr ( " Worst RX Setting " ) + " </b><br/> " ;
const RXItem & rxworst = * list . at ( list . size ( ) - 1 ) ;
html + = QString ( " <a href='overview=%1,%2'> " ) . arg ( rxworst . start . toString ( Qt : : ISODate ) ) . arg ( rxworst . end . toString ( Qt : : ISODate ) ) +
QObject : : tr ( " Date: %1 - %2 " ) . arg ( rxworst . start . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( rxworst . end . toString ( Qt : : SystemLocaleShortDate ) ) + " </a><br/> " ;
html + = QString ( " %1 " ) . arg ( rxworst . machine - > model ( ) ) + " <br/> " ;
html + = QString ( " Serial: %1 " ) . arg ( rxworst . machine - > serial ( ) ) + " <br/> " ;
html + = QObject : : tr ( " Culminative AHI: %1 " ) . arg ( double ( rxworst . ahi ) / rxworst . hours , 0 , ' f ' , 2 ) + " <br/> " ;
html + = QObject : : tr ( " Culminative Hours: %1 " ) . arg ( rxworst . hours , 0 , ' f ' , 2 ) + " <br/> " ;
html + = QString ( " %1 " ) . arg ( rxworst . pressure ) + " <br/> " ;
html + = QString ( " %1 " ) . arg ( rxworst . relief ) + " <br/> " ;
}
}
html + = " </body></html> " ;
mainwin - > setRecBoxHTML ( html ) ;
}
2014-05-06 09:11:31 +00:00
QString StatisticsRow : : value ( QDate start , QDate end )
{
const int decimals = 2 ;
2014-05-15 22:48:51 +00:00
QString value ;
2014-07-11 12:09:38 +00:00
float days = p_profile - > countDays ( type , start , end ) ;
2014-05-06 09:11:31 +00:00
// 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 ( ) ;
2014-07-13 04:39:26 +00:00
EventDataType val = 0 ;
QString fmt = " %1 " ;
2014-05-06 09:11:31 +00:00
if ( code ! = NoChannel ) {
switch ( calc ) {
case SC_AVG :
2014-07-13 04:39:26 +00:00
val = p_profile - > calcAvg ( code , type , start , end ) ;
2014-05-06 09:11:31 +00:00
break ;
case SC_WAVG :
2014-07-13 04:39:26 +00:00
val = p_profile - > calcWavg ( code , type , start , end ) ;
2014-05-15 22:48:51 +00:00
break ;
2014-05-06 09:11:31 +00:00
case SC_MEDIAN :
2014-07-13 04:39:26 +00:00
val = p_profile - > calcPercentile ( code , 0.5F , type , start , end ) ;
2014-05-06 09:11:31 +00:00
break ;
case SC_90P :
2014-07-13 04:39:26 +00:00
val = p_profile - > calcPercentile ( code , 0.9F , type , start , end ) ;
2014-05-06 09:11:31 +00:00
break ;
case SC_MIN :
2014-07-13 04:39:26 +00:00
val = p_profile - > calcMin ( code , type , start , end ) ;
2014-05-06 09:11:31 +00:00
break ;
case SC_MAX :
2014-07-13 04:39:26 +00:00
val = p_profile - > calcMax ( code , type , start , end ) ;
2014-05-06 09:11:31 +00:00
break ;
case SC_CPH :
2014-07-13 04:39:26 +00:00
val = p_profile - > calcCount ( code , type , start , end ) / p_profile - > calcHours ( type , start , end ) ;
2014-05-06 09:11:31 +00:00
break ;
case SC_SPH :
2014-07-13 04:39:26 +00:00
fmt + = " % " ;
val = 100.0 / p_profile - > calcHours ( type , start , end ) * p_profile - > calcSum ( code , type , start , end ) / 3600.0 ;
2014-05-15 21:48:53 +00:00
break ;
case SC_ABOVE :
2014-07-13 04:39:26 +00:00
fmt + = " % " ;
val = 100.0 / p_profile - > calcHours ( type , start , end ) * ( p_profile - > calcAboveThreshold ( code , schema : : channel [ code ] . upperThreshold ( ) , type , start , end ) / 60.0 ) ;
2014-05-15 21:48:53 +00:00
break ;
case SC_BELOW :
2014-07-13 04:39:26 +00:00
fmt + = " % " ;
val = 100.0 / p_profile - > calcHours ( type , start , end ) * ( p_profile - > calcBelowThreshold ( code , schema : : channel [ code ] . lowerThreshold ( ) , type , start , end ) / 60.0 ) ;
2014-05-15 21:48:53 +00:00
break ;
2014-05-06 09:11:31 +00:00
default :
break ;
} ;
}
2014-07-13 04:39:26 +00:00
if ( ( val = = std : : numeric_limits < EventDataType > : : min ( ) ) | | ( val = = std : : numeric_limits < EventDataType > : : max ( ) ) ) {
value = " Err " ;
} else {
value = fmt . arg ( val , 0 , ' f ' , decimals ) ;
}
2014-05-06 09:11:31 +00:00
}
return value ;
}