2019-08-18 21:27:21 +00:00
/* Welcome page Implementation
2018-05-29 08:12:37 +00:00
*
2024-01-13 20:27:48 +00:00
* Copyright ( c ) 2019 - 2024 The OSCAR Team
2024-02-01 00:14:19 +00:00
* Copyright ( c ) 2018 Mark Watkins
2018-05-29 08:12:37 +00:00
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file COPYING in the main directory of Source Code . */
# include <cmath>
2023-11-26 00:00:33 +00:00
// remember to turn test_macros off for release.
# define TEST_MACROS_ENABLEDoff
# include <test_macros.h>
2018-05-29 08:12:37 +00:00
# include "welcome.h"
2018-05-07 15:11:02 +00:00
# include "ui_welcome.h"
2014-07-25 07:53:48 +00:00
2018-05-07 15:11:02 +00:00
# include "mainwindow.h"
extern MainWindow * mainwin ;
2014-07-25 07:53:48 +00:00
2018-05-07 15:11:02 +00:00
Welcome : : Welcome ( QWidget * parent ) :
QWidget ( parent ) ,
ui ( new Ui : : Welcome )
{
ui - > setupUi ( this ) ;
2018-05-07 16:54:08 +00:00
pixmap . load ( " :/icons/mask.png " ) ;
2018-05-07 15:11:02 +00:00
2018-05-07 16:54:08 +00:00
refreshPage ( ) ;
}
2018-05-07 15:11:02 +00:00
2018-05-07 16:54:08 +00:00
Welcome : : ~ Welcome ( )
{
delete ui ;
}
void Welcome : : refreshPage ( )
{
2019-03-25 21:37:26 +00:00
bool b ;
2019-08-28 22:23:13 +00:00
const auto & mlist = p_profile - > GetMachines ( MT_CPAP ) ;
b = mlist . size ( ) > 0 ;
2018-05-07 16:54:08 +00:00
2019-08-28 22:23:13 +00:00
QList < Machine * > oximachines = p_profile - > GetMachines ( MT_OXIMETER ) ;
QList < Machine * > posmachines = p_profile - > GetMachines ( MT_POSITION ) ;
QList < Machine * > stgmachines = p_profile - > GetMachines ( MT_SLEEPSTAGE ) ;
2018-05-07 16:54:08 +00:00
2019-08-28 22:23:13 +00:00
bool noMachines = mlist . isEmpty ( ) & & posmachines . isEmpty ( ) & & oximachines . isEmpty ( ) & & stgmachines . isEmpty ( ) ;
2019-09-04 04:44:02 +00:00
bool showCardWarning = noMachines ;
2018-05-07 16:54:08 +00:00
// The SDCard warning does not need to be seen anymore for people who DON'T use ResMed S9's.. show first import and only when S9 is present
2018-05-07 15:11:02 +00:00
for ( auto & mach : mlist ) {
2019-09-07 05:15:36 +00:00
if ( mach - > brand ( ) . contains ( STR_MACH_ResMed ) & & mach - > series ( ) . contains ( " S9 " ) ) showCardWarning = true ;
2018-05-07 15:11:02 +00:00
}
2019-08-28 22:23:13 +00:00
2018-05-07 15:11:02 +00:00
ui - > S9Warning - > setVisible ( showCardWarning ) ;
2018-05-07 16:54:08 +00:00
if ( ! b ) {
2022-02-27 14:18:39 +00:00
qDebug ( ) < < " No devices in Profile " ;
2019-03-10 16:03:19 +00:00
// sleep(3);
2018-05-07 16:54:08 +00:00
ui - > cpapIcon - > setPixmap ( pixmap ) ;
}
2019-07-05 01:11:12 +00:00
2019-08-28 22:23:13 +00:00
b = ! noMachines ;
2019-07-05 01:11:12 +00:00
// Copy application font to tool buttons
ui - > importButton - > setFont ( QApplication : : font ( ) ) ;
ui - > dailyButton - > setFont ( QApplication : : font ( ) ) ;
ui - > overviewButton - > setFont ( QApplication : : font ( ) ) ;
ui - > statisticsButton - > setFont ( QApplication : : font ( ) ) ;
ui - > oximetryButton - > setFont ( QApplication : : font ( ) ) ;
// Enable buttons that might be disabled
2018-05-07 16:54:08 +00:00
ui - > dailyButton - > setEnabled ( b ) ;
2019-08-28 22:23:13 +00:00
ui - > oximetryButton - > setEnabled ( true ) ; // Import features always enabled
2018-05-07 16:54:08 +00:00
ui - > overviewButton - > setEnabled ( b ) ;
ui - > statisticsButton - > setEnabled ( b ) ;
2019-07-13 02:28:57 +00:00
ui - > importButton - > repaint ( ) ;
ui - > dailyButton - > repaint ( ) ;
ui - > overviewButton - > repaint ( ) ;
ui - > statisticsButton - > repaint ( ) ;
ui - > oximetryButton - > repaint ( ) ;
2019-03-26 01:10:03 +00:00
mainwin - > EnableTabs ( b ) ;
2019-07-05 01:11:12 +00:00
2018-05-07 15:11:02 +00:00
ui - > cpapInfo - > setHtml ( GenerateCPAPHTML ( ) ) ;
ui - > oxiInfo - > setHtml ( GenerateOxiHTML ( ) ) ;
}
void Welcome : : on_dailyButton_clicked ( )
{
mainwin - > JumpDaily ( ) ;
}
void Welcome : : on_overviewButton_clicked ( )
{
mainwin - > JumpOverview ( ) ;
}
void Welcome : : on_statisticsButton_clicked ( )
{
mainwin - > JumpStatistics ( ) ;
}
void Welcome : : on_oximetryButton_clicked ( )
{
mainwin - > JumpOxiWizard ( ) ;
}
void Welcome : : on_importButton_clicked ( )
{
mainwin - > JumpImport ( ) ;
}
2014-07-25 07:53:48 +00:00
2018-04-17 04:11:28 +00:00
extern EventDataType calcAHI ( QDate start , QDate end ) ;
extern EventDataType calcFL ( QDate start , QDate end ) ;
2018-05-07 15:11:02 +00:00
QString Welcome : : GenerateCPAPHTML ( )
2014-07-25 07:53:48 +00:00
{
2018-05-07 15:11:02 +00:00
auto cpap_machines = p_profile - > GetMachines ( MT_CPAP ) ;
auto oximeters = p_profile - > GetMachines ( MT_OXIMETER ) ;
2014-07-25 07:53:48 +00:00
QList < Machine * > mach ;
mach . append ( cpap_machines ) ;
mach . append ( oximeters ) ;
bool havecpapdata = false ;
bool haveoximeterdata = false ;
2018-05-07 15:11:02 +00:00
for ( auto & mach : cpap_machines ) {
int daysize = mach - > day . size ( ) ;
2014-07-25 07:53:48 +00:00
if ( daysize > 0 ) {
havecpapdata = true ;
break ;
}
}
2018-05-07 15:11:02 +00:00
for ( auto & mach : oximeters ) {
int daysize = mach - > day . size ( ) ;
2014-07-25 07:53:48 +00:00
if ( daysize > 0 ) {
haveoximeterdata = true ;
break ;
}
}
QString html = QString ( " <html><head> " ) +
2019-07-02 04:47:09 +00:00
// "</head>"
2014-07-25 07:53:48 +00:00
" <style type='text/css'> "
" p,a,td,body { font-family: ' " + QApplication : : font ( ) . family ( ) + " '; } "
" p,a,td,body { font-size: " + QString : : number ( QApplication : : font ( ) . pointSize ( ) + 2 ) + " px; } "
" </style> "
" </head> "
2023-11-26 00:00:33 +00:00
2018-05-08 08:12:32 +00:00
" <body leftmargin=5 topmargin=10 rightmargin=5 bottommargin=5 vertical-align=center align=center> " ;
2023-11-26 00:00:33 +00:00
html + = " <font size='+0'> " ;
2018-04-17 23:47:04 +00:00
Machine * cpap = nullptr ;
2014-07-25 07:53:48 +00:00
if ( ! havecpapdata & & ! haveoximeterdata ) {
2019-02-25 19:08:17 +00:00
html + = " <p> " + tr ( " It would be a good idea to check File->Preferences first, " ) + " <br /> " +
2019-02-24 22:10:31 +00:00
tr ( " as there are some options that affect import. " ) + " </p> " +
2022-02-27 14:18:39 +00:00
" <p> " + tr ( " Note that some preferences are forced when a ResMed device is detected " ) + " </p> " +
2018-05-07 15:11:02 +00:00
" <p> " + tr ( " First import can take a few minutes. " ) + " </p> " ;
2014-07-25 07:53:48 +00:00
} else {
2014-08-20 17:17:13 +00:00
QDate date = p_profile - > LastDay ( MT_CPAP ) ;
Day * day = p_profile - > GetDay ( date , MT_CPAP ) ;
if ( havecpapdata & & day ) {
2018-03-28 07:10:52 +00:00
cpap = day - > machine ( MT_CPAP ) ;
}
if ( day & & ( cpap ! = nullptr ) ) {
2018-05-07 15:11:02 +00:00
QString cpapimage = cpap - > getPixmapPath ( ) ;
ui - > cpapIcon - > setPixmap ( QPixmap ( cpapimage ) ) ;
2018-04-17 03:02:24 +00:00
2019-08-18 21:27:21 +00:00
html + = " <b> " + tr ( " The last time you used your %1... " ) . arg ( cpap - > brand ( ) + " " + cpap - > model ( ) ) + " </b><br/> " ;
2014-07-25 07:53:48 +00:00
int daysto = date . daysTo ( QDate : : currentDate ( ) ) ;
QString daystring ;
2018-05-07 15:11:02 +00:00
if ( daysto = = 1 ) daystring + = tr ( " last night " ) ;
2020-06-03 16:36:52 +00:00
else if ( daysto = = 2 ) daystring + = tr ( " 1 day ago " ) ;
2022-05-29 00:47:55 +00:00
else if ( daysto = = 0 ) daystring + = tr ( " today " ) ;
2020-08-15 15:57:26 +00:00
else daystring + = tr ( " %2 days ago " ) . arg ( daysto - 1 ) ;
2014-07-25 07:53:48 +00:00
2018-05-07 15:11:02 +00:00
html + = tr ( " was %1 (on %2) " ) . arg ( daystring ) . arg ( date . toString ( Qt : : SystemLocaleLongDate ) ) + " <br/> " ;
2014-07-25 07:53:48 +00:00
2020-07-24 18:29:19 +00:00
EventDataType hours = day - > hours ( MT_CPAP ) ;
2018-04-17 03:02:24 +00:00
html + = " <br/> " ;
2014-07-25 07:53:48 +00:00
int seconds = int ( hours * 3600.0 ) % 60 ;
int minutes = int ( hours * 60 ) % 60 ;
int hour = hours ;
2018-05-07 15:11:02 +00:00
QString timestr = tr ( " %1 hours, %2 minutes and %3 seconds " ) . arg ( hour ) . arg ( minutes ) . arg ( seconds ) ;
2014-07-25 07:53:48 +00:00
2019-06-03 00:48:27 +00:00
const EventDataType compliance_min = p_profile - > cpap - > m_complianceHours ; // 4.0;
2022-02-27 14:18:39 +00:00
if ( hours > compliance_min ) html + = tr ( " Your device was on for %1. " ) . arg ( timestr ) + " <br/> " ;
2018-05-07 15:11:02 +00:00
else html + = tr ( " <font color = red>You only had the mask on for %1.</font> " ) . arg ( timestr ) + " <br/> " ;
2014-07-25 07:53:48 +00:00
2018-04-17 03:02:24 +00:00
2018-04-17 04:11:28 +00:00
int averagedays = 7 ; // how many days to look back
2018-04-17 03:02:24 +00:00
2019-07-31 17:31:08 +00:00
QDate starttime = date . addDays ( - averagedays ) ;
QDate endtime = date . addDays ( - 1 ) ;
2018-04-17 04:11:28 +00:00
2021-07-25 04:12:15 +00:00
// EventDataType ahi = (day->count(CPAP_AllApnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Hypopnea) + day->count(CPAP_ClearAirway) + day->count(CPAP_Apnea)) / hours;
EventDataType ahi = day - > count ( AllAhiChannels ) / hours ;
2019-07-31 17:31:08 +00:00
EventDataType ahidays = calcAHI ( starttime , endtime ) ;
2018-04-17 04:11:28 +00:00
2018-05-07 15:11:02 +00:00
const QString under = tr ( " under " ) ;
const QString over = tr ( " over " ) ;
const QString close = tr ( " reasonably close to " ) ;
const QString equal = tr ( " equal to " ) ;
2018-04-17 04:11:28 +00:00
QString comp ;
if ( ( ahi < ahidays ) & & ( ( ahidays - ahi ) > = 0.1 ) ) {
comp = under ;
} else if ( ( ahi > ahidays ) & & ( ( ahi - ahidays ) > = 0.1 ) ) {
comp = over ;
2019-08-01 19:55:21 +00:00
} else if ( ( fabs ( ahi - ahidays ) > = 0.01 ) ) {
2018-04-17 23:47:04 +00:00
comp = close ;
2018-04-17 04:11:28 +00:00
} else {
2018-04-17 23:47:04 +00:00
comp = equal ;
2018-04-17 04:11:28 +00:00
}
2018-04-17 03:02:24 +00:00
2018-05-29 03:48:59 +00:00
html + = tr ( " You had an AHI of %1, which is %2 your %3 day average of %4. " ) . arg ( ahi , 0 , ' f ' , 2 ) . arg ( comp ) . arg ( averagedays ) . arg ( ahidays , 0 , ' f ' , 2 ) ;
2018-04-17 04:11:28 +00:00
html + = " <br/> " ;
2014-07-25 07:53:48 +00:00
CPAPMode cpapmode = ( CPAPMode ) ( int ) day - > settings_max ( CPAP_Mode ) ;
2019-09-16 19:05:47 +00:00
ChannelID pressChanID = day - > getPressureChannelID ( ) ; // Get channel id for pressure that we should report
2014-07-25 07:53:48 +00:00
double perc = p_profile - > general - > prefCalcPercentile ( ) ;
2020-03-30 01:50:19 +00:00
// When CPAP_PressureSet and CPAP_IPAPSet have data (used for percentiles, etc.)
// CPAP_Pressure and CPAP_IPAP are their corresponding settings channels.
ChannelID pressSettingChanID ;
if ( pressChanID = = CPAP_PressureSet ) {
pressSettingChanID = CPAP_Pressure ;
} else if ( pressChanID = = CPAP_IPAPSet ) {
pressSettingChanID = CPAP_IPAP ;
} else {
pressSettingChanID = pressChanID ;
}
2020-03-30 16:07:18 +00:00
ChannelID epapDataChanID = CPAP_EPAP ;
if ( day - > channelHasData ( CPAP_EPAPSet ) ) {
epapDataChanID = CPAP_EPAPSet ;
}
2023-04-15 13:48:55 +00:00
if ( day - > channelHasData ( CPAP_EEPAP ) ) {
epapDataChanID = CPAP_EEPAP ;
}
2020-03-30 16:07:18 +00:00
2020-03-30 01:50:19 +00:00
if ( pressChanID = = NoChannel ) {
qWarning ( ) < < " Unable to find pressure channel for welcome summary! " ;
}
2014-07-25 07:53:48 +00:00
if ( cpapmode = = MODE_CPAP ) {
2020-03-30 16:07:18 +00:00
pressSettingChanID = CPAP_Pressure ; // DreamStation ventilators report EPAP/IPAP data, but the setting is Pressure
2020-03-30 01:50:19 +00:00
EventDataType pressure = day - > settings_max ( pressSettingChanID ) ;
2020-03-30 16:07:18 +00:00
qDebug ( ) < < pressSettingChanID < < pressure ;
2022-02-27 14:18:39 +00:00
html + = tr ( " Your CPAP device used a constant %1 %2 of air " )
2020-08-30 20:43:00 +00:00
. arg ( pressure )
. arg ( schema : : channel [ pressChanID ] . units ( ) ) ;
2014-07-25 07:53:48 +00:00
} else if ( cpapmode = = MODE_APAP ) {
2019-09-16 19:05:47 +00:00
EventDataType pressure = day - > percentile ( pressChanID , perc / 100.0 ) ;
2020-08-30 20:43:00 +00:00
html + = tr ( " Your pressure was under %1 %2 for %3% of the time. " )
. arg ( pressure )
. arg ( schema : : channel [ pressChanID ] . units ( ) )
. arg ( perc ) ;
2014-07-25 07:53:48 +00:00
} else if ( cpapmode = = MODE_BILEVEL_FIXED ) {
2020-08-30 20:43:00 +00:00
// pressSettingChanID = CPAP_IPAP;
// EventDataType ipap = day->settings_max(pressSettingChanID);
// EventDataType epap = day->settings_min(CPAP_EPAP);
2022-02-27 14:18:39 +00:00
html + = tr ( " Your device used a constant %1-%2 %3 of air. " )
2020-08-30 20:43:00 +00:00
. arg ( day - > validPressure ( day - > settings_min ( CPAP_EPAP ) ) )
. arg ( day - > validPressure ( day - > settings_max ( CPAP_IPAP ) ) )
. arg ( schema : : channel [ CPAP_IPAP ] . units ( ) ) ;
2014-07-25 07:53:48 +00:00
} else if ( cpapmode = = MODE_BILEVEL_AUTO_FIXED_PS ) {
2019-09-16 19:05:47 +00:00
EventDataType ipap = day - > percentile ( pressChanID , perc / 100.0 ) ;
2020-03-30 16:07:18 +00:00
EventDataType epap = day - > percentile ( epapDataChanID , perc / 100.0 ) ;
2022-02-27 14:18:39 +00:00
html + = tr ( " Your device was under %1-%2 %3 for %4% of the time. " )
2020-08-30 20:43:00 +00:00
. arg ( epap )
. arg ( ipap )
. arg ( schema : : channel [ pressChanID ] . units ( ) )
. arg ( perc ) ;
2020-03-25 01:33:50 +00:00
} else if ( cpapmode = = MODE_ASV | | cpapmode = = MODE_AVAPS ) {
2019-09-16 19:05:47 +00:00
EventDataType ipap = day - > percentile ( pressChanID , perc / 100.0 ) ;
2014-07-28 13:56:29 +00:00
EventDataType epap = qRound ( day - > settings_wavg ( CPAP_EPAP ) ) ;
2020-08-30 20:43:00 +00:00
html + = tr ( " Your EPAP pressure fixed at %1 %2. " )
. arg ( epap )
. arg ( schema : : channel [ epapDataChanID ] . units ( ) ) + " <br/> " ;
html + = tr ( " Your IPAP pressure was under %1 %2 for %3% of the time. " )
. arg ( ipap )
. arg ( schema : : channel [ pressChanID ] . units ( ) )
. arg ( perc ) ;
2020-03-30 16:07:18 +00:00
} else if ( cpapmode = = MODE_ASV_VARIABLE_EPAP | | cpapmode = = MODE_BILEVEL_AUTO_VARIABLE_PS ) {
2019-09-16 19:05:47 +00:00
EventDataType ipap = day - > percentile ( pressChanID , perc / 100.0 ) ;
2020-03-30 16:07:18 +00:00
EventDataType epap = day - > percentile ( epapDataChanID , perc / 100.0 ) ;
2014-07-28 13:56:29 +00:00
2020-03-30 16:07:18 +00:00
html + = tr ( " Your EPAP pressure was under %1 %2 for %3% of the time. " ) . arg ( epap ) . arg ( schema : : channel [ epapDataChanID ] . units ( ) ) . arg ( perc ) + " <br/> " ;
2019-09-16 19:05:47 +00:00
html + = tr ( " Your IPAP pressure was under %1 %2 for %3% of the time. " ) . arg ( ipap ) . arg ( schema : : channel [ pressChanID ] . units ( ) ) . arg ( perc ) ;
2023-04-15 13:48:55 +00:00
} else if ( cpapmode = = MODE_TRILEVEL_AUTO_VARIABLE_PDIFF ) {
EventDataType ipap = day - > percentile ( pressChanID , perc / 100.0 ) ;
EventDataType eepap = day - > percentile ( epapDataChanID , perc / 100.0 ) ;
html + = tr ( " Your EEPAP pressure was under %1 %2 for %3% of the time. " ) . arg ( eepap ) . arg ( schema : : channel [ epapDataChanID ] . units ( ) ) . arg ( perc ) + " <br/> " ;
html + = tr ( " Your IPAP pressure was under %1 %2 for %3% of the time. " ) . arg ( ipap ) . arg ( schema : : channel [ pressChanID ] . units ( ) ) . arg ( perc ) ;
2014-07-25 07:53:48 +00:00
}
2018-04-17 04:11:28 +00:00
html + = " <br/> " ;
//EventDataType lat = day->timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline())/ 60.0;
//EventDataType leaks = 1.0/hours * lat;
2014-07-25 07:53:48 +00:00
2020-10-31 16:49:02 +00:00
EventDataType leak = day - > wavg ( CPAP_Leak ) ;
EventDataType leakdays = p_profile - > calcWavg ( CPAP_Leak , MT_CPAP , starttime , endtime ) ;
2018-04-17 04:11:28 +00:00
if ( ( leak < leakdays ) & & ( ( leakdays - leak ) > = 0.1 ) ) {
comp = under ;
} else if ( ( leak > leakdays ) & & ( ( leak - leakdays ) > = 0.1 ) ) {
comp = over ;
2021-10-07 21:15:04 +00:00
} else if ( ( fabs ( leak - leakdays ) > = 0.01 ) ) {
2018-04-17 23:47:04 +00:00
comp = close ;
2018-04-17 04:11:28 +00:00
} else {
2018-04-17 23:47:04 +00:00
comp = equal ;
2018-04-17 04:11:28 +00:00
}
2018-05-29 03:48:59 +00:00
html + = tr ( " Your average leaks were %1 %2, which is %3 your %4 day average of %5. " ) . arg ( leak , 0 , ' f ' , 2 ) . arg ( schema : : channel [ CPAP_Leak ] . units ( ) ) . arg ( comp ) . arg ( averagedays ) . arg ( leakdays , 0 , ' f ' , 2 ) ;
2018-04-17 04:11:28 +00:00
html + = " <br/> " ;
2014-07-25 07:53:48 +00:00
} else {
2018-05-07 15:11:02 +00:00
html + = " <p> " + tr ( " No CPAP data has been imported yet. " ) + " </p> " ;
2014-07-25 07:53:48 +00:00
}
2018-04-17 23:47:04 +00:00
}
2018-05-07 15:11:02 +00:00
html + = " </body></html> " ;
return html ;
}
QString Welcome : : GenerateOxiHTML ( )
{
auto oximeters = p_profile - > GetMachines ( MT_OXIMETER ) ;
bool haveoximeterdata = false ;
for ( auto & mach : oximeters ) {
int daysize = mach - > day . size ( ) ;
if ( daysize > 0 ) {
haveoximeterdata = true ;
break ;
}
}
QString html = QString ( " <html><head> " ) +
2019-07-02 04:47:09 +00:00
// "</head>"
2018-05-07 15:11:02 +00:00
" <style type='text/css'> "
" p,a,td,body { font-family: ' " + QApplication : : font ( ) . family ( ) + " '; } "
" p,a,td,body { font-size: " + QString : : number ( QApplication : : font ( ) . pointSize ( ) + 2 ) + " px; } "
" </style> "
" </head> "
2023-11-26 00:00:33 +00:00
2018-05-07 15:11:02 +00:00
" <body leftmargin=5 topmargin=5 rightmargin=5 bottommargin=5 valign=center align=center> " ;
2023-11-26 00:00:33 +00:00
html + = " <font size='+0'> " ;
2018-05-07 15:11:02 +00:00
if ( haveoximeterdata ) {
QDate oxidate = p_profile - > LastDay ( MT_OXIMETER ) ;
int daysto = oxidate . daysTo ( QDate : : currentDate ( ) ) ;
2018-06-04 20:48:38 +00:00
html + = " <p> " + QObject : : tr ( " Most recent Oximetry data: <a onclick='alert( \" daily=%2 \" );'>%1</a> " ) . arg ( oxidate . toString ( Qt : : SystemLocaleLongDate ) ) . arg ( oxidate . toString ( Qt : : ISODate ) ) ;
2018-05-07 15:11:02 +00:00
if ( daysto = = 1 ) html + = QObject : : tr ( " (last night) " ) ;
2020-06-03 16:36:52 +00:00
else if ( daysto = = 2 ) html + = QObject : : tr ( " (1 day ago) " ) ;
else html + = QObject : : tr ( " (%2 days ago) " ) . arg ( oxidate . daysTo ( QDate : : currentDate ( ) ) ) ;
2018-05-07 15:11:02 +00:00
html + = " </p> " ;
ui - > oxiIcon - > setVisible ( true ) ;
ui - > oxiInfo - > setVisible ( true ) ;
2018-04-17 23:47:04 +00:00
} else {
2018-06-04 20:48:38 +00:00
html + = " <p> " + QObject : : tr ( " No oximetry data has been imported yet. " ) + " </p> " ;
2018-05-07 15:11:02 +00:00
ui - > oxiIcon - > setVisible ( false ) ;
ui - > oxiInfo - > setVisible ( false ) ;
2018-04-17 23:47:04 +00:00
}
2018-05-07 15:11:02 +00:00
html + = " </body></html> " ;
2014-07-25 07:53:48 +00:00
return html ;
}