/* Welcome page Implementation * * Copyright (c) 2019-2024 The OSCAR Team * Copyright (c) 2018 Mark Watkins * * 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 // remember to turn test_macros off for release. #define TEST_MACROS_ENABLEDoff #include #include "welcome.h" #include "ui_welcome.h" #include "mainwindow.h" extern MainWindow * mainwin; Welcome::Welcome(QWidget *parent) : QWidget(parent), ui(new Ui::Welcome) { ui->setupUi(this); pixmap.load(":/icons/mask.png"); refreshPage(); } Welcome::~Welcome() { delete ui; } void Welcome::refreshPage() { bool b; const auto & mlist = p_profile->GetMachines(MT_CPAP); b = mlist.size() > 0; QList oximachines = p_profile->GetMachines(MT_OXIMETER); QList posmachines = p_profile->GetMachines(MT_POSITION); QList stgmachines = p_profile->GetMachines(MT_SLEEPSTAGE); bool noMachines = mlist.isEmpty() && posmachines.isEmpty() && oximachines.isEmpty() && stgmachines.isEmpty(); bool showCardWarning = noMachines; // 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 for (auto & mach :mlist) { if (mach->brand().contains(STR_MACH_ResMed) && mach->series().contains("S9")) showCardWarning = true; } ui->S9Warning->setVisible(showCardWarning); if (!b) { qDebug() << "No devices in Profile"; // sleep(3); ui->cpapIcon->setPixmap(pixmap); } b = !noMachines; // 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 ui->dailyButton->setEnabled(b); ui->oximetryButton->setEnabled(true); // Import features always enabled ui->overviewButton->setEnabled(b); ui->statisticsButton->setEnabled(b); ui->importButton->repaint(); ui->dailyButton->repaint(); ui->overviewButton->repaint(); ui->statisticsButton->repaint(); ui->oximetryButton->repaint(); mainwin->EnableTabs(b); 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(); } extern EventDataType calcAHI(QDate start, QDate end); extern EventDataType calcFL(QDate start, QDate end); QString Welcome::GenerateCPAPHTML() { auto cpap_machines = p_profile->GetMachines(MT_CPAP); auto oximeters = p_profile->GetMachines(MT_OXIMETER); QList mach; mach.append(cpap_machines); mach.append(oximeters); bool havecpapdata = false; bool haveoximeterdata = false; for (auto & mach : cpap_machines) { int daysize = mach->day.size(); if (daysize > 0) { havecpapdata = true; break; } } for (auto & mach : oximeters) { int daysize = mach->day.size(); if (daysize > 0) { haveoximeterdata = true; break; } } QString html = QString("")+ // "" "" "" ""; html += "" ; Machine * cpap = nullptr; if (!havecpapdata && !haveoximeterdata) { html += "

" + tr("It would be a good idea to check File->Preferences first,") + "
" + tr("as there are some options that affect import.")+"

" + "

" + tr("Note that some preferences are forced when a ResMed device is detected") + "

" + "

" + tr("First import can take a few minutes.") + "

"; } else { QDate date = p_profile->LastDay(MT_CPAP); Day *day = p_profile->GetDay(date, MT_CPAP); if (havecpapdata && day) { cpap = day->machine(MT_CPAP); } if (day && (cpap != nullptr)) { QString cpapimage = cpap->getPixmapPath(); ui->cpapIcon->setPixmap(QPixmap(cpapimage)); html+= ""+tr("The last time you used your %1...").arg(cpap->brand()+" "+cpap->model())+"
"; int daysto = date.daysTo(QDate::currentDate()); QString daystring; if (daysto == 1) daystring += tr("last night"); else if (daysto == 2) daystring += tr("1 day ago"); else if (daysto == 0) daystring += tr("today"); else daystring += tr("%2 days ago").arg(daysto-1); html += tr("was %1 (on %2)").arg(daystring).arg(date.toString(Qt::SystemLocaleLongDate)) + "
"; EventDataType hours = day->hours(MT_CPAP); html += "
"; int seconds = int(hours * 3600.0) % 60; int minutes = int(hours * 60) % 60; int hour = hours; QString timestr = tr("%1 hours, %2 minutes and %3 seconds").arg(hour).arg(minutes).arg(seconds); const EventDataType compliance_min = p_profile->cpap->m_complianceHours; // 4.0; if (hours > compliance_min) html += tr("Your device was on for %1.").arg(timestr)+"
"; else html += tr("You only had the mask on for %1.").arg(timestr)+"
"; int averagedays = 7; // how many days to look back QDate starttime = date.addDays(-averagedays); QDate endtime = date.addDays(-1); // 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; EventDataType ahidays = calcAHI(starttime, endtime); const QString under = tr("under"); const QString over = tr("over"); const QString close = tr("reasonably close to"); const QString equal = tr("equal to"); QString comp; if ((ahi < ahidays) && ((ahidays - ahi) >= 0.1)) { comp = under; } else if ((ahi > ahidays) && ((ahi - ahidays) >= 0.1)) { comp = over; } else if ((fabs(ahi - ahidays) >= 0.01) ) { comp = close; } else { comp = equal; } 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); html += "
"; CPAPMode cpapmode = (CPAPMode)(int)day->settings_max(CPAP_Mode); ChannelID pressChanID = day->getPressureChannelID(); // Get channel id for pressure that we should report double perc= p_profile->general->prefCalcPercentile(); // 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; } ChannelID epapDataChanID = CPAP_EPAP; if (day->channelHasData(CPAP_EPAPSet)) { epapDataChanID = CPAP_EPAPSet; } if (day->channelHasData(CPAP_EEPAP)) { epapDataChanID = CPAP_EEPAP; } if (pressChanID == NoChannel) { qWarning() << "Unable to find pressure channel for welcome summary!"; } if (cpapmode == MODE_CPAP) { pressSettingChanID = CPAP_Pressure; // DreamStation ventilators report EPAP/IPAP data, but the setting is Pressure EventDataType pressure = day->settings_max(pressSettingChanID); qDebug() << pressSettingChanID << pressure; html += tr("Your CPAP device used a constant %1 %2 of air") .arg(pressure) .arg(schema::channel[pressChanID].units()); } else if (cpapmode == MODE_APAP) { EventDataType pressure = day->percentile(pressChanID, perc/100.0); html += tr("Your pressure was under %1 %2 for %3% of the time.") .arg(pressure) .arg(schema::channel[pressChanID].units()) .arg(perc); } else if (cpapmode == MODE_BILEVEL_FIXED) { // pressSettingChanID = CPAP_IPAP; // EventDataType ipap = day->settings_max(pressSettingChanID); // EventDataType epap = day->settings_min(CPAP_EPAP); html += tr("Your device used a constant %1-%2 %3 of air.") .arg(day->validPressure(day->settings_min(CPAP_EPAP))) .arg(day->validPressure(day->settings_max(CPAP_IPAP))) .arg(schema::channel[CPAP_IPAP].units()); } else if (cpapmode == MODE_BILEVEL_AUTO_FIXED_PS) { EventDataType ipap = day->percentile(pressChanID, perc/100.0); EventDataType epap = day->percentile(epapDataChanID, perc/100.0); html += tr("Your device was under %1-%2 %3 for %4% of the time.") .arg(epap) .arg(ipap) .arg(schema::channel[pressChanID].units()) .arg(perc); } else if (cpapmode == MODE_ASV || cpapmode == MODE_AVAPS){ EventDataType ipap = day->percentile(pressChanID, perc/100.0); EventDataType epap = qRound(day->settings_wavg(CPAP_EPAP)); html += tr("Your EPAP pressure fixed at %1 %2.") .arg(epap) .arg(schema::channel[epapDataChanID].units())+"
"; html += tr("Your IPAP pressure was under %1 %2 for %3% of the time.") .arg(ipap) .arg(schema::channel[pressChanID].units()) .arg(perc); } else if (cpapmode == MODE_ASV_VARIABLE_EPAP || cpapmode == MODE_BILEVEL_AUTO_VARIABLE_PS){ EventDataType ipap = day->percentile(pressChanID, perc/100.0); EventDataType epap = day->percentile(epapDataChanID, perc/100.0); html += tr("Your EPAP pressure was under %1 %2 for %3% of the time.").arg(epap).arg(schema::channel[epapDataChanID].units()).arg(perc)+"
"; html += tr("Your IPAP pressure was under %1 %2 for %3% of the time.").arg(ipap).arg(schema::channel[pressChanID].units()).arg(perc); } 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)+"
"; html += tr("Your IPAP pressure was under %1 %2 for %3% of the time.").arg(ipap).arg(schema::channel[pressChanID].units()).arg(perc); } html += "
"; //EventDataType lat = day->timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline())/ 60.0; //EventDataType leaks = 1.0/hours * lat; EventDataType leak = day->wavg(CPAP_Leak); EventDataType leakdays = p_profile->calcWavg(CPAP_Leak, MT_CPAP, starttime, endtime); if ((leak < leakdays) && ((leakdays - leak) >= 0.1)) { comp = under; } else if ((leak > leakdays) && ((leak - leakdays) >= 0.1)) { comp = over; } else if ((fabs(leak - leakdays) >= 0.01) ) { comp = close; } else { comp = equal; } 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); html += "
"; } else { html += "

"+tr("No CPAP data has been imported yet.")+"

"; } } 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 += "" ; if (haveoximeterdata) { QDate oxidate=p_profile->LastDay(MT_OXIMETER); int daysto = oxidate.daysTo(QDate::currentDate()); html += "

"+QObject::tr("Most recent Oximetry data: %1 ").arg(oxidate.toString(Qt::SystemLocaleLongDate)).arg(oxidate.toString(Qt::ISODate)); if (daysto == 1) html += QObject::tr("(last night)"); else if (daysto == 2) html += QObject::tr("(1 day ago)"); else html += QObject::tr("(%2 days ago)").arg(oxidate.daysTo(QDate::currentDate())); html+="

"; ui->oxiIcon->setVisible(true); ui->oxiInfo->setVisible(true); } else { html += "

"+QObject::tr("No oximetry data has been imported yet.")+"

"; ui->oxiIcon->setVisible(false); ui->oxiInfo->setVisible(false); } html += ""; return html; }