2018-05-28 22:35:48 +00:00
/* Statistics Report Generator Implementation
2014-04-09 21:01:57 +00:00
*
2018-03-28 07:10:52 +00:00
* Copyright ( c ) 2011 - 2018 Mark Watkins < mark @ jedimark . net >
2014-04-09 21:01:57 +00:00
*
* This file is subject to the terms and conditions of the GNU General Public
2018-06-04 20:48:38 +00:00
* License . See the file COPYING in the main directory of the source code
* 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>
2018-05-28 22:35:48 +00:00
# include <QBuffer>
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 ;
2018-05-28 22:35:48 +00:00
QString resizeHTMLPixmap ( QPixmap & pixmap , int width , int height ) {
QByteArray byteArray ;
QBuffer buffer ( & byteArray ) ; // use buffer to store pixmap into byteArray
buffer . open ( QIODevice : : WriteOnly ) ;
pixmap . scaled ( width , height , Qt : : KeepAspectRatio , Qt : : SmoothTransformation ) . save ( & buffer , " PNG " ) ;
return QString ( " <img src= \" data:image/png;base64, " + byteArray . toBase64 ( ) + " \" > " ) ;
}
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 ;
2018-03-28 06:22:42 +00:00
//seconds %= 60;
2014-05-06 09:11:31 +00:00
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 ) {
2018-04-22 12:06:48 +00:00
rx . machine = p_profile - > lookupMachine ( serial , loadername ) ;
2014-09-11 14:23:08 +00:00
} 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 ( ) ;
2016-03-02 23:27:23 +00:00
// Read the cache from disk
2014-09-11 14:23:08 +00:00
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 ;
2016-03-01 11:13:52 +00:00
2016-03-02 23:27:23 +00:00
// Scan through each daylist in ascending date order
2014-09-11 14:23:08 +00:00
for ( it = p_profile - > daylist . begin ( ) ; it ! = it_end ; + + it ) {
const QDate & date = it . key ( ) ;
Day * day = it . value ( ) ;
2016-03-01 11:13:52 +00:00
2014-09-11 14:23:08 +00:00
Machine * mach = day - > machine ( MT_CPAP ) ;
2016-03-01 11:13:52 +00:00
if ( mach = = nullptr )
continue ;
2014-09-11 14:23:08 +00:00
bool fnd = false ;
2016-03-01 11:13:52 +00:00
2016-03-02 23:27:23 +00:00
// Scan through pre-existing rxitems list and see if this day is already there.
2014-09-11 14:23:08 +00:00
ri_end = rxitems . end ( ) ;
for ( ri = rxitems . begin ( ) ; ri ! = ri_end ; + + ri ) {
RXItem & rx = ri . value ( ) ;
2016-03-01 11:13:52 +00:00
// Is it date between this rxitems entry date range?
2014-09-11 14:23:08 +00:00
if ( ( date > = rx . start ) & & ( date < = rx . end ) ) {
2016-03-01 11:13:52 +00:00
2014-09-11 14:23:08 +00:00
if ( rx . dates . contains ( date ) ) {
2016-03-01 11:13:52 +00:00
// Already there, abort.
2014-09-11 14:23:08 +00:00
fnd = true ;
break ;
}
// First up, check if fits in date range, but isn't loaded for some reason
2016-03-01 11:13:52 +00:00
// Need summaries for this, so load them if not present.
2014-09-11 14:23:08 +00:00
day - > OpenSummary ( ) ;
2016-03-01 11:13:52 +00:00
2016-03-02 23:27:23 +00:00
// Get list of Event Flags used in this day
2014-09-11 14:23:08 +00:00
QList < ChannelID > flags = day - > getSortedMachineChannels ( MT_CPAP , schema : : FLAG | schema : : MINOR_FLAG | schema : : SPAN ) ;
2016-03-02 23:27:23 +00:00
// Generate the pressure/mode/relief strings
2014-09-11 14:23:08 +00:00
QString relief = day - > getPressureRelief ( ) ;
QString mode = day - > getCPAPMode ( ) ;
QString pressure = day - > getPressureSettings ( ) ;
2016-03-02 23:27:23 +00:00
// Do this days settings match this rx cache entry?
2014-09-11 14:23:08 +00:00
if ( ( rx . relief = = relief ) & & ( rx . mode = = mode ) & & ( rx . pressure = = pressure ) & & ( rx . machine = = mach ) ) {
2016-03-02 23:27:23 +00:00
// Update rx cache summaries for each event flag
2014-09-11 14:23:08 +00:00
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 ) ;
}
2016-03-02 23:27:23 +00:00
// Update AHI/RDI/Time counts
2014-09-11 14:23:08 +00:00
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 ) ;
2016-03-02 23:27:23 +00:00
// Add this date to RX cache
2014-09-11 14:23:08 +00:00
rx . dates [ date ] = day ;
rx . days = rx . dates . size ( ) ;
2016-03-02 23:27:23 +00:00
// and we are done
2014-09-11 14:23:08 +00:00
fnd = true ;
break ;
} else {
2016-03-02 23:27:23 +00:00
// In this case, the day is within the rx date range, but settings doesn't match the others
// So we need to split the rx cache record and insert the new record as it's own.
2014-09-11 14:23:08 +00:00
RXItem rx1 , rx2 ;
2016-03-02 23:27:23 +00:00
// So first create the new cache entry for current day we are looking at.
2014-09-11 14:23:08 +00:00
rx1 . start = date ;
rx1 . end = date ;
rx1 . days = 1 ;
2016-03-02 23:27:23 +00:00
// Only this days AHI/RDI counts
2014-09-11 14:23:08 +00:00
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 ) ;
2016-03-02 23:27:23 +00:00
// Sum and count event flags for this day
2014-09-11 14:23:08 +00:00
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 ) ;
}
2016-03-02 23:27:23 +00:00
//The rest of this cache record for this day
2014-09-11 14:23:08 +00:00
rx1 . hours = day - > hours ( MT_CPAP ) ;
rx1 . relief = relief ;
rx1 . mode = mode ;
rx1 . pressure = pressure ;
2016-03-04 03:34:05 +00:00
rx1 . machine = mach ;
2014-09-11 14:23:08 +00:00
rx1 . dates [ date ] = day ;
2016-03-02 23:27:23 +00:00
// Insert new entry into rx cache
2014-09-11 14:23:08 +00:00
rxitems . insert ( date , rx1 ) ;
2016-03-02 23:27:23 +00:00
// now zonk it so we can reuse the variable later
//rx1 = RXItem();
// Now that's out of the way, we need to splitting the old rx into two,
// and recalculate everything before and after today
// Copy the old rx.dates, which contains the list of Day records
2014-09-11 14:23:08 +00:00
QMap < QDate , Day * > datecopy = rx . dates ;
2016-03-02 23:27:23 +00:00
// now zap it so we can start fresh
2014-09-11 14:23:08 +00:00
rx . dates . clear ( ) ;
2016-03-02 23:27:23 +00:00
rx2 . end = rx2 . start = rx . end ;
2014-09-11 14:23:08 +00:00
rx . end = rx . start ;
2016-03-02 23:27:23 +00:00
// Zonk the summary data, as it needs redoing
2014-09-11 14:23:08 +00:00
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 ( ) ;
2016-03-02 23:27:23 +00:00
// Now go through day list and recalculate according to split
2014-09-11 14:23:08 +00:00
for ( di = datecopy . begin ( ) ; di ! = datecopy . end ( ) ; + + di ) {
2016-03-02 23:27:23 +00:00
// Split everything before date
2014-09-11 14:23:08 +00:00
if ( di . key ( ) < date ) {
2016-03-02 23:27:23 +00:00
// Get the day record for this date
2014-09-11 14:23:08 +00:00
Day * dy = rx . dates [ di . key ( ) ] = p_profile - > GetDay ( di . key ( ) , MT_CPAP ) ;
2016-03-02 23:27:23 +00:00
// Update AHI/RDI counts
2014-09-11 14:23:08 +00:00
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 ) ;
2016-03-02 23:27:23 +00:00
// Get Event Flags list
2014-09-11 14:23:08 +00:00
QList < ChannelID > flags2 = dy - > getSortedMachineChannels ( MT_CPAP , schema : : FLAG | schema : : MINOR_FLAG | schema : : SPAN ) ;
2016-03-02 23:27:23 +00:00
// Update flags counts and sums
2014-09-11 14:23:08 +00:00
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 ) ;
}
2016-03-02 23:27:23 +00:00
// Update time sum
2014-09-11 14:23:08 +00:00
rx . hours + = dy - > hours ( MT_CPAP ) ;
2016-03-02 23:27:23 +00:00
// Update the last date of this cache entry
// (Max here should be unnessary, this should be sequential because we are processing a QMap.)
rx . end = di . key ( ) ; //qMax(di.key(), rx.end);
2014-09-11 14:23:08 +00:00
}
2016-03-02 23:27:23 +00:00
// Split everything after date
2014-09-11 14:23:08 +00:00
if ( di . key ( ) > date ) {
2016-03-02 23:27:23 +00:00
// Get the day record for this date
2014-09-11 14:23:08 +00:00
Day * dy = rx2 . dates [ di . key ( ) ] = p_profile - > GetDay ( di . key ( ) , MT_CPAP ) ;
2016-03-02 23:27:23 +00:00
// Update AHI/RDI counts
2014-09-11 14:23:08 +00:00
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 ) ;
2016-03-02 23:27:23 +00:00
// Get Event Flags list
2014-09-11 14:23:08 +00:00
QList < ChannelID > flags2 = dy - > getSortedMachineChannels ( MT_CPAP , schema : : FLAG | schema : : MINOR_FLAG | schema : : SPAN ) ;
2016-03-02 23:27:23 +00:00
// Update flags counts and sums
2014-09-11 14:23:08 +00:00
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 ) ;
}
2016-03-02 23:27:23 +00:00
// Update time sum
2014-09-11 14:23:08 +00:00
rx2 . hours + = dy - > hours ( MT_CPAP ) ;
2016-03-02 23:27:23 +00:00
// Update start and end
//rx2.end = qMax(di.key(), rx2.end); // don't need to do this, the end won't change from what the old one was.
// technically only need to capture the first??
2014-09-11 14:23:08 +00:00
rx2 . start = qMin ( di . key ( ) , rx2 . start ) ;
}
}
2016-03-02 23:27:23 +00:00
// Set rx records day counts
2014-09-11 14:23:08 +00:00
rx . days = rx . dates . size ( ) ;
2016-03-02 23:27:23 +00:00
rx2 . days = rx2 . dates . size ( ) ;
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// Copy the pressure/mode/etc settings, because they haven't changed.
2014-09-11 14:23:08 +00:00
rx2 . pressure = rx . pressure ;
rx2 . mode = rx . mode ;
rx2 . relief = rx . relief ;
rx2 . machine = rx . machine ;
2016-03-02 23:27:23 +00:00
// Insert the newly split rx record
rxitems . insert ( rx2 . start , rx2 ) ; // hmmm. this was previously set to the end date.. that was a silly plan.
2014-09-11 14:23:08 +00:00
fnd = true ;
break ;
}
}
}
2016-03-02 23:27:23 +00:00
if ( fnd ) continue ; // already in rx list, move onto the next daylist entry
// So in this condition, daylist isn't in rx cache, and doesn't match date range of any previous rx cache entry.
// Need to bring in summaries for this
2014-09-11 14:23:08 +00:00
day - > OpenSummary ( ) ;
2016-03-02 23:27:23 +00:00
// Get Event flags list
QList < ChannelID > flags3 = day - > getSortedMachineChannels ( MT_CPAP , schema : : FLAG | schema : : MINOR_FLAG | schema : : SPAN ) ;
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// Generate pressure/mode/relief strings
2014-09-11 14:23:08 +00:00
QString relief = day - > getPressureRelief ( ) ;
QString mode = day - > getCPAPMode ( ) ;
QString pressure = day - > getPressureSettings ( ) ;
2016-03-02 23:27:23 +00:00
// Now scan the rxcache to find the most previous entry, and the right place to insert
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
QMap < QDate , RXItem > : : iterator lastri = rxitems . end ( ) ;
2016-03-01 11:13:52 +00:00
2016-03-02 23:27:23 +00:00
for ( ri = rxitems . begin ( ) ; ri ! = ri_end ; + + ri ) {
// RXItem & rx = ri.value();
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// break after any date newer
if ( ri . key ( ) > date )
break ;
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// Keep this.. we need the last one.
lastri = ri ;
}
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// lastri should no be the last entry before this date, or the end
if ( lastri ! = rxitems . end ( ) ) {
RXItem & rx = lastri . value ( ) ;
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// Does it match here?
2016-03-04 03:34:05 +00:00
if ( ( rx . relief = = relief ) & & ( rx . mode = = mode ) & & ( rx . pressure = = pressure ) & & ( rx . machine = = mach ) ) {
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// Update AHI/RDI
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 ) ;
// Update event flags
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 ) ;
2014-09-11 14:23:08 +00:00
}
2016-03-02 23:27:23 +00:00
// Update hours
rx . hours + = day - > hours ( MT_CPAP ) ;
// Add day to this RX Cache
rx . dates [ date ] = day ;
rx . end = date ;
rx . days = rx . dates . size ( ) ;
fnd = true ;
}
2014-09-11 14:23:08 +00:00
}
if ( ! fnd ) {
2016-03-02 23:27:23 +00:00
// Okay, couldn't find a match, create a new rx cache record for this day.
2014-09-11 14:23:08 +00:00
RXItem rx ;
rx . start = date ;
rx . end = date ;
rx . days = 1 ;
2016-03-02 23:27:23 +00:00
// Set AHI/RDI for just this day
2014-09-11 14:23:08 +00:00
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 ) ;
2016-03-02 23:27:23 +00:00
// Set counts and sums for this day
2014-09-11 14:23:08 +00:00
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 ) ;
}
2016-03-02 23:27:23 +00:00
2014-09-11 14:23:08 +00:00
rx . hours = day - > hours ( ) ;
2016-03-02 23:27:23 +00:00
// Store settings, etc..
2014-09-11 14:23:08 +00:00
rx . relief = relief ;
rx . mode = mode ;
rx . pressure = pressure ;
2016-03-04 03:34:05 +00:00
rx . machine = mach ;
2016-03-02 23:27:23 +00:00
// add this day to this rx record
2014-09-11 14:23:08 +00:00
rx . dates . insert ( date , day ) ;
2016-03-02 23:27:23 +00:00
// And insert into rx record into the rx cache
2014-09-11 14:23:08 +00:00
rxitems . insert ( date , rx ) ;
}
}
2016-03-02 23:27:23 +00:00
// Store RX cache to disk
2014-09-11 14:23:08 +00:00
saveRXChanges ( ) ;
2016-03-02 23:27:23 +00:00
// Now do the setup for the best worst highlighting
2014-09-11 14:23:08 +00:00
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
2014-10-13 04:03:18 +00:00
rows . push_back ( StatisticsRow ( tr ( " Therapy Efficacy " ) , SC_SUBHEADING , MT_CPAP ) ) ;
2014-05-06 17:39:05 +00:00
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
2017-08-02 01:31:59 +00:00
int percentile = trunc ( p_profile - > general - > prefCalcPercentile ( ) ) ; // Pholynyk, 10Mar2016
char perCentStr [ 20 ] ;
snprintf ( perCentStr , 20 , " %d%% %%1 " , percentile ) ; //
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 " ) ;
2017-08-02 01:31:59 +00:00
calcnames [ SC_90P ] = tr ( perCentStr ) ; // this gets converted to whatever the upper percentile is set to
2014-05-06 17:39:05 +00:00
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> " ;
2018-05-28 22:35:48 +00:00
QPixmap bobPixmap ( " :/icons/bob-v3.0.png " ) ;
2014-07-25 07:53:48 +00:00
if ( showheader ) {
2018-05-29 01:30:52 +00:00
html + = " <div align=center> "
+ resizeHTMLPixmap ( bobPixmap , 64 , 64 ) + " <br/> "
" <font size='+3'> " + STR_TR_SleepyHead + " </font><br/> "
" <font size='+2'> " + QObject : : tr ( " Usage Statistics " ) + " </font><br/> "
2018-05-29 03:48:59 +00:00
" <font size='+1' title= \" " + QObject : : tr ( " This is for legal reasons. Sorry, not sorry. Without manufacturer support and documentation, SleepyHead is unsuitable as a compliance/medical reporting tool. " ) + " \" > " + QObject : : tr ( " (NOT approved for compliance or medical reporting purposes) " ) + " </font><br/> " ;
2018-05-29 01:30:52 +00:00
if ( ! userinfo . isEmpty ( ) ) html + = " <br/> " + userinfo + " <br/> " ;
html + = " </div><br/> " ;
2014-07-25 07:53:48 +00:00
}
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> " ;
2018-05-29 01:30:52 +00:00
html + = QString ( QObject : : tr ( " This report was generated by SleepyHead v%1 " ) ) . arg ( ShortVersionString ) + " <br/> "
+ QObject : : tr ( " SleepyHead is free open-source CPAP research software available from http://sleepyhead.jedimark.net " ) ;
2014-07-25 07:53:48 +00:00
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 ;
} ;
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 ;
}
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 ( )
{
2016-03-02 23:27:23 +00:00
// do the actual data sorting...
2014-09-11 14:23:08 +00:00
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/> " ;
2018-05-29 03:48:59 +00:00
html + = QString ( " <table class=curved style= \" page-break-before:always; \" " + table_width + " width=100%> " ) ;
2014-09-11 14:23:08 +00:00
html + = " <thead> " ;
2018-05-29 03:48:59 +00:00
html + = " <tr bgcolor=' " + heading_color + " '><th colspan=9 align=center><font size=+2> " + tr ( " Changes to Prescription Settings " ) + " </font></th></tr> " ;
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// QString extratxt;
2014-09-11 14:23:08 +00:00
2016-03-02 23:27:23 +00:00
// QString tooltip;
2014-09-11 14:23:08 +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 ) ;
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-10-02 11:22:30 +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 ) {
2014-10-02 11:22:30 +00:00
//done = true;
2014-05-06 11:54:41 +00:00
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
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/> " ;
}
}
2016-02-27 03:37:56 +00:00
if ( p_profile - > hasChannel ( CPAP_PB ) ) {
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_PB ) ;
ahilist . insert ( leak , date ) ;
}
if ( ahilist . size ( ) > ( show_records * 2 ) ) {
html + = " <b> " + QObject : : tr ( " Worst PB " ) + " </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 PB: %2% " ) . arg ( it . value ( ) . toString ( Qt : : SystemLocaleShortDate ) ) . arg ( it . key ( ) , 0 , ' f ' , 2 ) + " </a><br/> " ;
cnt + + ;
}
}
if ( cnt = = 0 ) {
html + = " <i> " + QObject : : tr ( " No PB on record " ) + " </i><br/> " ;
}
html + = " <br/> " ;
}
}
2014-09-30 08:40:12 +00:00
} 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
2017-08-02 01:31:59 +00:00
float percentile = p_profile - > general - > prefCalcPercentile ( ) / 100.0 ; // Pholynyk, 10Mar2016
EventDataType percent = percentile ; // was 0.90F
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 :
2017-08-02 01:31:59 +00:00
val = p_profile - > calcPercentile ( code , percent , 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 ;
}