/* Statistics Report Generator Implementation * * Copyright (c) 2019-2022 The OSCAR Team * Copyright (c) 2011-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 the source code * for more details. */ #define TEST_MACROS_ENABLEDoff #include #include #include #include #include #include #include #include #include #include #include "mainwindow.h" #include "statistics.h" #include "cprogressbar.h" #include "SleepLib/common.h" #include "version.h" extern MainWindow *mainwin; // HTML components that make up Statistics page and printed report QString htmlReportHeader = ""; // Page header QString htmlReportHeaderPrint = ""; // Page header QString htmlUsage = ""; // CPAP and Oximetry QString htmlMachineSettings = ""; // Device (formerly Rx) changes QString htmlMachines = ""; // Devices used in this profile QString htmlReportFooter = ""; // Page footer 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("logo"); } QString formatTime(float time) { int hours = time; int seconds = time * 3600.0; int minutes = (seconds / 60) % 60; //seconds %= 60; return QString::asprintf("%02i:%02i", hours, minutes); //,seconds); } QDataStream & operator>>(QDataStream & in, RXItem & rx) { in >> rx.start; in >> rx.end; in >> rx.days; in >> rx.ahi; in >> rx.rdi; in >> rx.hours; QString loadername; in >> loadername; QString serial; in >> serial; MachineLoader * loader = GetLoader(loadername); if (loader) { rx.machine = p_profile->lookupMachine(serial, loadername); } else { qDebug() << "Bad machine object" << loadername << serial; rx.machine = nullptr; } in >> rx.relief; in >> rx.mode; in >> rx.pressure; QList list; in >> list; rx.dates.clear(); for (int i=0; iFindDay(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(); 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)) { qDebug() << "Could not open" << path << "for reading, error code" << file.error() << file.errorString(); 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)) { qWarning() << "Could not open" << path << "for writing, error code" << file.error() << file.errorString(); 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::updateDisabledInfo() { QDate lastcpap = p_profile->LastGoodDay(MT_CPAP); QDate firstcpap = p_profile->FirstGoodDay(MT_CPAP); if (lastcpap > p_profile->general->statReportDate() ) { lastcpap = p_profile->general->statReportDate(); } disabledInfo.update( lastcpap, firstcpap ); } void DisabledInfo::update(QDate latestDate, QDate earliestDate) { clear(); if ( (!latestDate.isValid()) || (!earliestDate.isValid()) || (p_profile->cpap->clinicalMode()) ) return; qint64 complianceHours = 3600000.0 * p_profile->cpap->complianceHours(); // conbvert to ms totalDays = 1+earliestDate.daysTo(latestDate); for (QDate date = latestDate ; date >= earliestDate ; date=date.addDays(-1) ) { Day* day = p_profile->GetDay(date); if (!day) { daysNoData++; continue;}; // find basic statistics for a day int numDisabled=0; qint64 sessLength = 0; qint64 dayLength = 0; qint64 enabledLength = 0; QList sessions = day->getSessions(MT_CPAP,true); for (auto & sess : sessions) { sessLength = sess->length(); //if (sessLength<0) sessLength=0; // some sessions have negative length. Sould solve this issue dayLength += sessLength; if (sess->enabled(true)) { enabledLength += sessLength; } else { numDisabled ++; totalDurationOfDisabledSessions += sessLength; if (maxDurationOfaDisabledsession < sessLength) maxDurationOfaDisabledsession = sessLength; } } // calculate stats for all days // calculate if compliance for a day changed. if ( complianceHours <= enabledLength ) { daysInCompliance ++; } else { if (complianceHours < dayLength ) { numDaysDisabledSessionChangedCompliance++; } else { daysOutOfCompliance ++; } } // update disabled info for all days if ( numDisabled > 0 ) { numDisabledsessions += numDisabled; numDaysWithDisabledsessions++; }; } // convect ms to minutes maxDurationOfaDisabledsession/=60000 ; totalDurationOfDisabledSessions/=60000 ; }; QString DisabledInfo::display(int type) { /* Permissive mode: some sessions are excluded from this report, as follows: Total disabled sessions: xx, found in yy days. Duration of longest disabled session: aa minutes, Total duration of all disabled sessions: bb minutes. +tr("Date: %1 AHI: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
"; */ switch (type) { default : case 0: //return QString(QObject::tr("Permissive mode is set (Preferences/Clinical), disabled sessions are excluded from this report")); //return QString(QObject::tr("Permissive mode allows disabled sessions")); return QString(QObject::tr("Permissive Mode")); case 1: if (numDisabledsessions>0) { return QString(QObject::tr("Total disabled sessions: %1, found in %2 days") .arg(numDisabledsessions) .arg(numDaysWithDisabledsessions)); } else { return QString(QObject::tr("Total disabled sessions: %1") .arg(numDisabledsessions) ); } case 2: return QString(QObject::tr( "Duration of longest disabled session: %1 minutes, Total duration of all disabled sessions: %2 minutes.") .arg(maxDurationOfaDisabledsession, 0, 'f', 1) .arg(totalDurationOfDisabledSessions, 0, 'f', 1)); } } void Statistics::updateRXChanges() { // Set conditional progress bar. CProgressBar * progress = new CProgressBar (QObject::tr("Updating Statistics cache"), mainwin, p_profile->daylist.count()); // Clear loaded rx cache rxitems.clear(); // Read the cache from disk loadRXChanges(); QMap::iterator di; QMap::iterator it; QMap::iterator it_end = p_profile->daylist.end(); QMap::iterator ri; QMap::iterator ri_end = rxitems.end(); quint64 tmp; // Scan through each daylist in ascending date order for (it = p_profile->daylist.begin(); it != it_end; ++it) { const QDate & date = it.key(); Day * day = it.value(); progress->add (1); // Increment progress bar Machine * mach = day->machine(MT_CPAP); if (mach == nullptr) continue; if (day->first() == 0) { // Ignore invalid dates //qDebug() << "Statistics::updateRXChanges ignoring day with first=0"; continue; } bool fnd = false; // Scan through pre-existing rxitems list and see if this day is already there. ri_end = rxitems.end(); for (ri = rxitems.begin(); ri != ri_end; ++ri) { RXItem & rx = ri.value(); // Is it date between this rxitems entry date range? if ((date >= rx.start) && (date <= rx.end)) { if (rx.dates.contains(date)) { // Already there, abort. fnd = true; break; } // First up, check if fits in date range, but isn't loaded for some reason // Need summaries for this, so load them if not present. day->OpenSummary(); // Get list of Event Flags used in this day QList flags = day->getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN); // Generate the pressure/mode/relief strings QString relief = day->getPressureRelief(); QString mode = day->getCPAPModeStr(); QString pressure = day->getPressureSettings(); // Do this days settings match this rx cache entry? if ((rx.relief == relief) && (rx.mode == mode) && (rx.pressure == pressure) && (rx.machine == mach)) { // Update rx cache summaries for each event flag 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); } // Update AHI/RDI/Time counts tmp = day->count(AllAhiChannels); rx.ahi += tmp; rx.rdi += tmp + day->count(CPAP_RERA); rx.hours += day->hours(MT_CPAP); // Add this date to RX cache rx.dates[date] = day; rx.days = rx.dates.size(); // and we are done fnd = true; break; } else { // 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. RXItem rx1, rx2; // So first create the new cache entry for current day we are looking at. rx1.start = date; rx1.end = date; rx1.days = 1; // Only this days AHI/RDI counts tmp = day->count(AllAhiChannels); rx1.ahi = tmp; rx1.rdi = tmp + day->count(CPAP_RERA); // Sum and count event flags for this day 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); } //The rest of this cache record for this day rx1.hours = day->hours(MT_CPAP); rx1.relief = relief; rx1.mode = mode; rx1.pressure = pressure; rx1.machine = mach; rx1.dates[date] = day; // Insert new entry into rx cache rxitems.insert(date, rx1); // 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 QMap datecopy = rx.dates; // now zap it so we can start fresh rx.dates.clear(); rx2.end = rx2.start = rx.end; rx.end = rx.start; // Zonk the summary data, as it needs redoing 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(); // Now go through day list and recalculate according to split for (di = datecopy.begin(); di != datecopy.end(); ++di) { // Split everything before date if (di.key() < date) { // Get the day record for this date Day * dy = rx.dates[di.key()] = p_profile->GetDay(di.key(), MT_CPAP); // Update AHI/RDI counts tmp = dy->count(AllAhiChannels); rx.ahi += tmp; rx.rdi += tmp + dy->count(CPAP_RERA); // Get Event Flags list QList flags2 = dy->getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN); // Update flags counts and sums 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); } // Update time sum rx.hours += dy->hours(MT_CPAP); // 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); } // Split everything after date if (di.key() > date) { // Get the day record for this date Day * dy = rx2.dates[di.key()] = p_profile->GetDay(di.key(), MT_CPAP); // Update AHI/RDI counts tmp = dy->count(AllAhiChannels); rx2.ahi += tmp; rx2.rdi += tmp + dy->count(CPAP_RERA); // Get Event Flags list QList flags2 = dy->getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN); // Update flags counts and sums 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); } // Update time sum rx2.hours += dy->hours(MT_CPAP); // 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?? rx2.start = qMin(di.key(), rx2.start); } } // Set rx records day counts rx.days = rx.dates.size(); rx2.days = rx2.dates.size(); // Copy the pressure/mode/etc settings, because they haven't changed. rx2.pressure = rx.pressure; rx2.mode = rx.mode; rx2.relief = rx.relief; rx2.machine = rx.machine; // 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. fnd = true; break; } } } 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 day->OpenSummary(); // Get Event flags list QList flags3 = day->getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN); // Generate pressure/mode/`strings QString relief = day->getPressureRelief(); QString mode = day->getCPAPModeStr(); QString pressure = day->getPressureSettings(); // Now scan the rxcache to find the most previous entry, and the right place to insert QMap::iterator lastri = rxitems.end(); for (ri = rxitems.begin(); ri != ri_end; ++ri) { // RXItem & rx = ri.value(); // break after any date newer if (ri.key() > date) break; // Keep this.. we need the last one. lastri = ri; } // lastri should no be the last entry before this date, or the end if (lastri != rxitems.end()) { RXItem & rx = lastri.value(); // Does it match here? if ((rx.relief == relief) && (rx.mode == mode) && (rx.pressure == pressure) && (rx.machine == mach) ) { // Update AHI/RDI tmp = day->count(AllAhiChannels); 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); } // 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; } } if (!fnd) { // Okay, couldn't find a match, create a new rx cache record for this day. RXItem rx; rx.start = date; rx.end = date; rx.days = 1; // Set AHI/RDI for just this day tmp = day->count(AllAhiChannels); rx.ahi = tmp; rx.rdi = tmp + day->count(CPAP_RERA); // Set counts and sums for this day for (int i=0; i < flags3.size(); i++) { ChannelID code = flags3.at(i); rx.s_count[code] = day->count(code); rx.s_sum[code] = day->sum(code); } rx.hours = day->hours(); // Store settings, etc.. rx.relief = relief; rx.mode = mode; rx.pressure = pressure; rx.machine = mach; // add this day to this rx record rx.dates.insert(date, day); // And insert into rx record into the rx cache rxitems.insert(date, rx); } } // Store RX cache to disk saveRXChanges(); // Now do the setup for the best worst highlighting QList list; ri_end = rxitems.end(); for (ri = rxitems.begin(); ri != ri_end; ++ri) { list.append(&ri.value()); ri.value().highlight = 0; } std::sort(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 } // Close the progress bar progress->close(); delete progress; } // Statistics constructor is responsible for creating list of rows that will on the Statistics page // and skeletons of column 1 text that correspond to each calculation type. // Actual column 1 text is combination of skeleton for the row's calculation time and the text of the row. // Also creates "device" names for device types. Statistics::Statistics(QObject *parent) : QObject(parent) { rows.push_back(StatisticsRow(tr("CPAP Statistics"), SC_HEADING, MT_CPAP)); if (!p_profile->cpap->clinicalMode()) { updateDisabledInfo(); rows.push_back(StatisticsRow(disabledInfo.display(0),SC_WARNING ,MT_CPAP)); rows.push_back(StatisticsRow(disabledInfo.display(1),SC_WARNING2,MT_CPAP)); if (disabledInfo.size()>0) { rows.push_back(StatisticsRow(disabledInfo.display(2),SC_WARNING2,MT_CPAP)); } } rows.push_back(StatisticsRow("", SC_DAYS, MT_CPAP)); rows.push_back(StatisticsRow("", SC_COLUMNHEADERS, MT_CPAP)); rows.push_back(StatisticsRow(tr("CPAP Usage"), SC_SUBHEADING, MT_CPAP)); rows.push_back(StatisticsRow(tr("Average Hours per Night"), SC_HOURS, MT_CPAP)); rows.push_back(StatisticsRow(tr("Compliance (%1 hrs/day)"), SC_COMPLIANCE, MT_CPAP)); rows.push_back(StatisticsRow(tr("Therapy Efficacy"), SC_SUBHEADING, MT_CPAP)); rows.push_back(StatisticsRow("AHI", SC_AHI, MT_CPAP)); rows.push_back(StatisticsRow("AllApnea", SC_CPH, MT_CPAP)); rows.push_back(StatisticsRow("Obstructive", SC_CPH, MT_CPAP)); rows.push_back(StatisticsRow("Hypopnea", SC_CPH, MT_CPAP)); rows.push_back(StatisticsRow("Apnea", 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("FLG", SC_90P, MT_CPAP)); rows.push_back(StatisticsRow("RERA", SC_CPH, MT_CPAP)); rows.push_back(StatisticsRow("SensAwake", SC_CPH, MT_CPAP)); rows.push_back(StatisticsRow("CSR", SC_SPH, MT_CPAP)); rows.push_back(StatisticsRow(tr("Leak Statistics"), SC_SUBHEADING, MT_CPAP)); rows.push_back(StatisticsRow("Leak", SC_WAVG, MT_CPAP)); rows.push_back(StatisticsRow("Leak", SC_90P, MT_CPAP)); rows.push_back(StatisticsRow("Leak", SC_ABOVE, MT_CPAP)); rows.push_back(StatisticsRow(tr("Pressure Statistics"), SC_SUBHEADING, MT_CPAP)); rows.push_back(StatisticsRow("Pressure", SC_WAVG, MT_CPAP)); rows.push_back(StatisticsRow("Pressure", SC_MIN, MT_CPAP)); rows.push_back(StatisticsRow("Pressure", SC_MAX, MT_CPAP)); rows.push_back(StatisticsRow("Pressure", SC_90P, MT_CPAP)); rows.push_back(StatisticsRow("PressureSet", SC_WAVG, MT_CPAP)); rows.push_back(StatisticsRow("PressureSet", SC_MIN, MT_CPAP)); rows.push_back(StatisticsRow("PressureSet", SC_MAX, MT_CPAP)); rows.push_back(StatisticsRow("PressureSet", 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("EPAPSet", SC_WAVG, MT_CPAP)); rows.push_back(StatisticsRow("EPAPSet", SC_MIN, MT_CPAP)); rows.push_back(StatisticsRow("EPAPSet", 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("IPAPSet", SC_WAVG, MT_CPAP)); rows.push_back(StatisticsRow("IPAPSet", SC_90P, MT_CPAP)); rows.push_back(StatisticsRow("IPAPSet", SC_MIN, MT_CPAP)); rows.push_back(StatisticsRow("IPAPSet", SC_MAX, MT_CPAP)); rows.push_back(StatisticsRow("", SC_HEADING, MT_OXIMETER)); // Just adds some space 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)); // These are for formatting the headers for the first column int percentile=trunc(p_profile->general->prefCalcPercentile()); // Pholynyk, 10Mar2016 char perCentStr[20]; snprintf(perCentStr, 20, "%d%% %%1", percentile); // calcnames[SC_UNDEFINED] = ""; calcnames[SC_MEDIAN] = tr("%1 Median"); calcnames[SC_AVG] = tr("Average %1"); calcnames[SC_WAVG] = tr("Average %1"); calcnames[SC_90P] = tr(perCentStr); // this gets converted to whatever the upper percentile is set to calcnames[SC_MIN] = tr("Min %1"); calcnames[SC_MAX] = tr("Max %1"); calcnames[SC_CPH] = tr("%1 Index"); calcnames[SC_SPH] = tr("% of time in %1"); calcnames[SC_ABOVE] = tr("% of time above %1 threshold"); calcnames[SC_BELOW] = tr("% of time below %1 threshold"); machinenames[MT_UNKNOWN] = STR_TR_Unknown; machinenames[MT_CPAP] = STR_TR_CPAP; machinenames[MT_OXIMETER] = STR_TR_Oximetry; machinenames[MT_SLEEPSTAGE] = STR_TR_SleepStage; // { MT_JOURNAL, STR_TR_Journal }, // { MT_POSITION, STR_TR_Position }, } // Get the user information block for displaying at top of page QString Statistics::getUserInfo () { if (!AppSetting->showPersonalData()) return ""; QString address = p_profile->user->address(); address.replace("\n", "
"); QString userinfo = ""; if (!p_profile->user->firstName().isEmpty()) { userinfo = tr("Name: %1, %2").arg(p_profile->user->lastName()).arg(p_profile->user->firstName()) + "
"; if (!p_profile->user->DOB().isNull()) { userinfo += tr("DOB: %1").arg(p_profile->user->DOB().toString(MedDateFormat)) + "
"; } if (!p_profile->user->phone().isEmpty()) { userinfo += tr("Phone: %1").arg(p_profile->user->phone()) + "
"; } if (!p_profile->user->email().isEmpty()) { userinfo += tr("Email: %1").arg(p_profile->user->email()) + "

"; } if (!p_profile->user->address().isEmpty()) { userinfo += tr("Address:")+"
"+address; } } while (userinfo.length() > 0 && userinfo.endsWith("
")) // Strip trailing newlines userinfo = userinfo.mid(0, userinfo.length()-4); return userinfo; } const QString table_width = "width='100%'"; // Create the page header in HTML. Includes everything from through QString Statistics::generateHeader(bool onScreen) { QString html = QString(""); html += "Oscar Statistics Report"; html += "" "" "" "" ""; //leftmargin=0 topmargin=5 rightmargin=0>"; QPixmap logoPixmap(":/icons/logo-lg.png"); // html += "
" html += "
" "" "" "" "" "" "
" + getUserInfo() + "" "" + STR_TR_OSCAR + "   
" "" + QObject::tr("Usage Statistics") + "   " "
" + resizeHTMLPixmap(logoPixmap,80,80)+"   
" "
" "
"; return html; } // HTML for page footer QString Statistics::generateFooter(bool showinfo) { QString html; if (showinfo) { html += "
"; QDateTime timestamp = QDateTime::currentDateTime(); html += tr("This report was prepared on %1 by OSCAR %2").arg(timestamp.toString(MedDateFormat + " hh:mm")) .arg(getVersion()) + "
" + tr("OSCAR is free open-source CPAP report software"); html += "
"; } html += ""; return html; } // Calculate AHI for a period as total # of events / total hours used // Add RERA if calculating RDI instead of just AHI EventDataType calcAHI(QDate start, QDate end) { EventDataType val = 0; for (int i = 0; i < ahiChannels.size(); i++) val += p_profile->calcCount(ahiChannels.at(i), MT_CPAP, start, end); // (p_profile->calcCount(CPAP_Obstructive, MT_CPAP, start, end) // + p_profile->calcCount(CPAP_AllApnea, MT_CPAP, start, end) // + p_profile->calcCount(CPAP_Hypopnea, MT_CPAP, start, end) // + p_profile->calcCount(CPAP_ClearAirway, MT_CPAP, start, end) // + p_profile->calcCount(CPAP_Apnea, MT_CPAP, start, end)); if (p_profile->general->calculateRDI()) { val += p_profile->calcCount(CPAP_RERA, MT_CPAP, start, end); } EventDataType hours = p_profile->calcHours(MT_CPAP, start, end); if (hours > 0) { val /= hours; } else { val = 0; } return val; } // Calculate flow limits per hour EventDataType calcFL(QDate start, QDate end) { EventDataType val = (p_profile->calcCount(CPAP_FlowLimit, MT_CPAP, start, end)); EventDataType hours = p_profile->calcHours(MT_CPAP, start, end); if (hours > 0) { val /= hours; } else { val = 0; } return val; } // Calculate ...(what are these?) 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; } // Structure for recording Prescription Changes (now called Device Settings Changes) struct RXChange { RXChange() { highlight = 0; machine = nullptr; } RXChange(const RXChange ©) { 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; pressure_string = copy.pressure_string; pr_relief_string = copy.pr_relief_string; } QDate first; QDate last; int days; EventDataType ahi; EventDataType fl; CPAPMode mode; QString pressure_string; QString pr_relief_string; EventDataType min; EventDataType max; EventDataType ps; EventDataType pshi; EventDataType maxipap; EventDataType per1; EventDataType per2; EventDataType weighted; Machine *machine; short highlight; }; struct UsageData { UsageData() { ahi = 0; hours = 0; } UsageData(QDate d, EventDataType v, EventDataType h) { date = d; ahi = v; hours = h; } UsageData(const UsageData ©) { date = copy.date; ahi = copy.ahi; hours = copy.hours; } QDate date; EventDataType ahi; EventDataType hours; }; bool operator <(const UsageData &c1, const UsageData &c2) { if (c1.ahi < c2.ahi) { return true; } if ((c1.ahi == c2.ahi) && (c1.date > c2.date)) { return true; } return false; } 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; } Period(QDate first,QDate last,bool& finished, int advance , bool month,QString name) { if (finished) return; // adds date range to header. // replaces the following // periods.push_back(Period(qMax(last.addDays(-6), first), last, tr("Last Week"))); QDate next; if (month) { // note add days or addmonths returns the start of the next day or the next month. // must shorten one day for Month. next = last.addMonths(advance).addDays(+1); } else { next = last.addDays(advance); } if (next<=first) { finished = true; next = first; } name = name + "
" + next.toString(Qt::SystemLocaleShortDate) ; if (advance!=0) { name = name + " - " + last.toString(Qt::SystemLocaleShortDate); } this->header = name; this->start = next ; this->end = last ; } Period& operator=(const Period&) = default; ~Period() {}; QDate start; QDate end; QString header; }; const QString warning_color="#ffffff"; const QString heading_color="#ffffff"; const QString subheading_color="#e0e0e0"; //const int rxthresh = 5; // Sort devices by first day of use bool machineCompareFirstDay(Machine* left, Machine *right) { return left->FirstDay() > right->FirstDay(); } QString Statistics::GenerateMachineList() { QList cpap_machines = p_profile->GetMachines(MT_CPAP); QList oximeters = p_profile->GetMachines(MT_OXIMETER); QList mach; std::sort(cpap_machines.begin(), cpap_machines.end(), machineCompareFirstDay); std::sort(oximeters.begin(), oximeters.end(), machineCompareFirstDay); mach.append(cpap_machines); mach.append(oximeters); QString html; if (mach.size() > 0) { html += "

"; html += QString(""); html += ""; html += ""; html += QString("") .arg(STR_TR_Brand) .arg(STR_TR_Model) .arg(STR_TR_Serial) .arg(tr("First Use")) .arg(tr("Last Use")); html += ""; Machine *m; for (int i = 0; i < mach.size(); i++) { m = mach.at(i); if (m->type() == MT_JOURNAL) { continue; } //qDebug() << "Device" << m->brand() << "series" << m->series() << "model" << m->model() << "model number" << m->modelnumber(); QDate d1 = m->FirstDay(); QDate d2 = m->LastDay(); if (d2 > p_profile->general->statReportDate() ) { d2 = p_profile->general->statReportDate(); } QString mn = m->modelnumber(); html += QString("") .arg(m->brand()) .arg(m->model() + (mn.isEmpty() ? "" : QString(" (") + mn + QString(")"))) .arg(m->serial()) .arg(d1.toString(MedDateFormat)) .arg(d2.toString(MedDateFormat)); } html += "
" + tr("Device Information") + "
%1%2%3%4%5
%1%2%3%4%5
"; html += "
"; } return html; } QString Statistics::GenerateRXChanges() { // Generate list only if there are CPAP devices QList cpap_machines = p_profile->GetMachines(MT_CPAP); if (cpap_machines.isEmpty()) return ""; // do the actual data sorting... updateRXChanges(); QString ahitxt; bool rdi = p_profile->general->calculateRDI(); if (rdi) { ahitxt = STR_TR_RDI; } else { ahitxt = STR_TR_AHI; } QString html = "

"; html += QString(""); html += ""; html += ""; // QString extratxt; // QString tooltip; QStringList hdrlist; hdrlist.push_back(STR_TR_First); hdrlist.push_back(STR_TR_Last); hdrlist.push_back(tr("Days")); hdrlist.push_back(ahitxt); hdrlist.push_back(STR_TR_FL); hdrlist.push_back(STR_TR_Machine); hdrlist.push_back(tr("Pressure Relief")); hdrlist.push_back(STR_TR_Mode); hdrlist.push_back(tr("Pressure Settings")); html+=""; for (int i=0; i < hdrlist.size(); ++i) { html+=QString(" ").arg(hdrlist.at(i)); } html+=""; html += ""; // html += ""; // html += ""; // html += ""; QMapIterator it(rxitems); it.toBack(); while (it.hasPrevious()) { it.previous(); const RXItem & rx = it.value(); if (rx.start > p_profile->general->statReportDate() ) continue; QDate rxend=rx.end; if (rxend > p_profile->general->statReportDate() ) rxend = p_profile->general->statReportDate(); 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("") .arg(color) .arg(rx.start.toString(Qt::ISODate)) .arg(rxend.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; QString machid = QString("").arg(rx.machine->model()) .arg(rx.machine->modelnumber()); if (AppSetting->includeSerial()) machid = QString("").arg(rx.machine->model()) .arg(rx.machine->modelnumber()) .arg(rx.machine->serial()); html += QString("").arg(rx.start.toString(MedDateFormat))+ QString("").arg(rxend.toString(MedDateFormat))+ QString("").arg(rx.days)+ QString("").arg(ahi, 0, 'f', 2)+ QString("").arg(fli, 0, 'f', 2)+ machid + QString("").arg(formatRelief(rx.relief))+ QString("").arg(rx.mode)+ QString("").arg(rx.pressure)+ ""; } html+="
" + tr("Changes to Device Settings") + "
%1
"; // html += QString("") + // tr("Efficacy highlighting ignores prescription settings with less than %1 days of recorded data."). // arg(rxthresh) + QString("
"); // html += "
%1 (%2)%1 (%2) [%3]%1%1%1%1%1%1%1%1
"; return html; } // Report no data available QString Statistics::htmlNoData() { QString html = "
"; html += QString( "


" + tr("No data found?!?") + "

"+ "

logo

" "

"+tr("Oscar has no data to report :(")+"

"); return html; } // Get RDI or AHI text depending on user preferences QString Statistics::getRDIorAHIText() { if (p_profile->general->calculateRDI()) { return STR_TR_RDI; } return STR_TR_AHI; } // Create the HTML for CPAP and Oximetry usage QString Statistics::GenerateCPAPUsage() { QList cpap_machines = p_profile->GetMachines(MT_CPAP); QList oximeters = p_profile->GetMachines(MT_OXIMETER); QList mach; mach.append(cpap_machines); mach.append(oximeters); // Go through all CPAP and Oximeter devices and see if any data is present bool havedata = false; for (int i=0; i < mach.size(); ++i) { int daysize = mach[i]->day.size(); if (daysize > 0) { havedata = true; break; } } QString html = ""; // If we don't have any data, return HTML that says that and we are done if (!havedata) { return ""; } // Find first and last days with valid CPAP data QDate lastcpap = p_profile->LastGoodDay(MT_CPAP); QDate firstcpap = p_profile->FirstGoodDay(MT_CPAP); if (lastcpap > p_profile->general->statReportDate() ) { lastcpap = p_profile->general->statReportDate(); } QString ahitxt = getRDIorAHIText(); // Prepare top of table html += "
"; html += ""; // Compute number of monthly periods for a monthly rather than standard time distribution int number_periods = 0; if (p_profile->general->statReportMode() == STAT_MODE_MONTHLY) { int firstMonth = firstcpap.month(); int lastMonth = lastcpap.month(); int years = lastcpap.year() - firstcpap.year(); lastMonth += (12 * years); // handle time extending to next year number_periods = lastMonth - firstMonth + 1; if (number_periods < 1) { qDebug() << "*** Begin" << firstcpap << "beginMonth" << firstMonth << "lastMonth" << lastMonth << "periods" << number_periods; number_periods = 1; } qDebug() << "Number of months for stats (trim to 12 max)" << number_periods; // But not more than one year if (number_periods > 12) { number_periods = 12; } // } else if (p_profile->general->statReportMode() == STAT_MODE_RANGE) { } QDate last = lastcpap, first = lastcpap; QList periods; bool skipsection = false;; // Loop through all rows of the Statistics report for (QList::iterator i = rows.begin(); i != rows.end(); ++i) { StatisticsRow &row = (*i); QString name; if (row.calc == SC_HEADING) { // All sections begin with a heading last = p_profile->LastGoodDay(row.type); first = p_profile->FirstGoodDay(row.type); if (last > p_profile->general->statReportDate() ) { last = p_profile->general->statReportDate(); } // Clear the periods (columns) periods.clear(); if (p_profile->general->statReportMode() == STAT_MODE_STANDARD) { // note add days or addmonths returns the start of the next day or the next month. // must shorten one day for each. Month executed in Period method bool finished = false; // used to detect end of data - when less than a year of data. periods.push_back(Period(first,last,finished, 0, false ,tr("Most Recent"))); periods.push_back(Period(first,last,finished, -6, false ,tr("Last Week"))); periods.push_back(Period(first,last,finished, -29,false, tr("Last 30 Days"))); periods.push_back(Period(first,last,finished, -6,true, tr("Last 6 Months"))); periods.push_back(Period(first,last,finished, -12,true,tr("Last Year"))); } else if (p_profile->general->statReportMode() == STAT_MODE_MONTHLY) { QDate l=last,s=last; periods.push_back(Period(last,last,tr("Last Session"))); //bool done=false; int j=0; do { 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
yyyy"))); j++; } l = s.addDays(-1); } while ((l > first) && (j < number_periods)); for (; j < number_periods; ++j) { periods.push_back(Period(last,last, "")); } } else { // STAT_MODE_RANGE first = p_profile->general->statReportRangeStart(); last = p_profile->general->statReportRangeEnd(); if (first > last) { first = last; } periods.push_back(Period(first,last,first.toString(MedDateFormat)+" - "+last.toString(MedDateFormat))); } int days = p_profile->countDays(row.type, first, last); skipsection = (days == 0); if (days > 0) { html+=QString("
"). arg(heading_color).arg(periods.size()+1).arg(row.src); } continue; } // Bypass this entire section if no data is present if (skipsection) continue; if (row.calc == SC_AHI) { name = ahitxt; } else if (row.calc == SC_HOURS) { name = row.src; } else if (row.calc == SC_COMPLIANCE) { name = QString(row.src).arg(p_profile->cpap->m_complianceHours); } else if (row.calc == SC_COLUMNHEADERS) { html += QString("").arg(tr("Details")); for (int j=0; j < periods.size(); j++) { html += QString("").arg(periods.at(j).start.toString(Qt::ISODate)).arg(periods.at(j).end.toString(Qt::ISODate)).arg(periods.at(j).header); } html += ""; continue; } else if (row.calc == SC_DAYS) { QDate first=p_profile->FirstGoodDay(row.type); QDate last=p_profile->LastGoodDay(row.type); if (last > p_profile->general->statReportDate() ) { last = p_profile->general->statReportDate(); } QString & machine = machinenames[row.type]; int value=p_profile->countDays(row.type, first, last); if (value == 0) { html+=QString("").arg(periods.size()+1). arg(tr("Database has No %1 data available.").arg(machine)); } else if (value == 1) { html+=QString("").arg(periods.size()+1). arg(tr("Database has %1 day of %2 Data on %3") .arg(value) .arg(machine) .arg(last.toString(MedDateFormat))); } else { html+=QString("").arg(periods.size()+1). arg(tr("Database has %1 days of %2 Data, between %3 and %4") .arg(value) .arg(machine) .arg(first.toString(MedDateFormat)) .arg(last.toString(MedDateFormat))); } continue; } else if (row.calc == SC_SUBHEADING) { // subheading.. html+=QString(""). arg(subheading_color).arg(periods.size()+1).arg(row.src); continue; } else if (row.calc == SC_UNDEFINED) { continue; } else if (row.calc == SC_WARNING) { //html+=QString(""). // arg(warning_color).arg(periods.size()+1).arg(row.src); html+=QString(""). arg(warning_color).arg(periods.size()+1).arg(row.src); continue; } else if (row.calc == SC_WARNING2) { html+=QString(""). arg(warning_color).arg(periods.size()+1).arg(row.src); continue; } else { ChannelID id = schema::channel[row.src].id(); if ((id == NoChannel) || (!p_profile->channelAvailable(id))) { continue; } name = calcnames[row.calc].arg(schema::channel[id].fullname()); } // Defined percentages for columns for diffent modes. QString line; int np = periods.size(); int width; // both create header column and 5 data columns for a total of 100 int dataWidth = 14; int headerWidth = 30; if (p_profile->general->statReportMode() == STAT_MODE_MONTHLY) { // both create header column and 13 data columns for a total of 100 dataWidth = 6; headerWidth = 22; } line += QString("").arg(headerWidth).arg(name); for (int j=0; j < np; j++) { width = j < np-1 ? dataWidth : 100 - (headerWidth + dataWidth*(np-1)); line += QString(""; } html += line; html += ""; } html += "
%3
%1%3
%2
%2
%2
%3
%3
%3
%3
%2").arg(width); if (!periods.at(j).header.isEmpty()) { line += row.value(periods.at(j).start, periods.at(j).end); } else { line +=" "; } line += "
"; html += "
"; return html; } // Create the HTML that will be the Statistics page. QString Statistics::GenerateHTML() { htmlReportHeader = generateHeader(true); htmlReportHeaderPrint = generateHeader(false); htmlReportFooter = generateFooter(true); htmlUsage = GenerateCPAPUsage(); if (htmlUsage == "") { return htmlReportHeader + htmlNoData() + htmlReportFooter; } htmlMachineSettings = GenerateRXChanges(); htmlMachines = GenerateMachineList(); QString htmlScript = ""; return htmlReportHeader + htmlUsage + htmlMachineSettings + htmlMachines + htmlScript + htmlReportFooter; } // Print the Statistics page on printer void Statistics::printReport(QWidget * parent) { QPrinter printer(QPrinter::ScreenResolution); // ScreenResolution required for graphics sizing #ifdef Q_OS_LINUX printer.setPrinterName("Print to File (PDF)"); printer.setOutputFormat(QPrinter::PdfFormat); QString name = "Statistics"; QString datestr = QDate::currentDate().toString(Qt::ISODate); QString filename = p_pref->Get("{home}/") + name + "_" + p_profile->user->userName() + "_" + datestr + ".pdf"; printer.setOutputFileName(filename); #endif printer.setPrintRange(QPrinter::AllPages); printer.setPageOrientation(QPageLayout::Portrait); printer.setFullPage(false); // Print only on printable area of page and not in non-printable margins printer.setCopyCount(1); QMarginsF minMargins = printer.pageLayout().margins(QPageLayout::Millimeter); printer.setPageMargins( QMarginsF( fmax(10,minMargins.left()), fmax(10,minMargins.top()), fmax(10,minMargins.right()), fmax(12,minMargins.bottom())), QPageLayout::Millimeter); QMarginsF setMargins = printer.pageLayout().margins(QPageLayout::Millimeter); qDebug () << "Min margins" << minMargins << "Set margins" << setMargins << "millimeters"; // Show print dialog to user and allow them to change settings as desired QPrintDialog pdlg(&printer, parent); if (pdlg.exec() == QPrintDialog::Accepted) { QTextDocument doc; QSizeF printArea = printer.pageRect(QPrinter::Point).size(); QSizeF originalPrintArea = printArea; printArea.setWidth(printArea.width()*2); // scale up for better font appearance printArea.setHeight(printArea.height()*2); doc.setPageSize(printArea); // Set document to print area, in pixels, removing default 2cm margins qDebug() << "print area (points)" << originalPrintArea << "Enlarged print area" << printArea << "paper size" << printer.paperRect(QPrinter::Point).size(); // Determine appropriate font and font size QFont font = QFont("Helvetica"); float fontScalar = 12; float printWidth = printArea.width(); if (printWidth > 1000) printWidth = 1000 + (printWidth - 1000) * 0.90; // Increase font for wide paper (landscape), but not linearly float pointSize = (printWidth / fontScalar) / 10.0; font.setPointSize(round(pointSize)); // Scale the font doc.setDefaultFont(font); qDebug() << "Enlarged printer font" << font << "printer default font set" << doc.defaultFont(); doc.setHtml(htmlReportHeaderPrint + htmlUsage + htmlReportFooter + htmlMachineSettings + htmlMachines + htmlReportFooter); // Dump HTML for use with HTML4 validator // QString html = htmlReportHeaderPrint + htmlUsage + htmlMachineSettings + htmlMachines + htmlReportFooter; // qDebug() << "Html:" << html; doc.print(&printer); } } QString Statistics::UpdateRecordsBox() { QString html = "" "Device Statistics Panel" ""; Machine * cpap = p_profile->GetMachine(MT_CPAP); if (cpap) { QDate first = p_profile->FirstGoodDay(MT_CPAP); QDate last = p_profile->LastGoodDay(MT_CPAP); if (last > p_profile->general->statReportDate() ) { last = p_profile->general->statReportDate(); } ///////////////////////////////////////////////////////////////////////////////////// /// Compliance and usage information ///////////////////////////////////////////////////////////////////////////////////// int realTotal = 1+first.daysTo(last); int totaldays = p_profile->countDays(MT_CPAP, first, last); int compliant = p_profile->countCompliantDays(MT_CPAP, first, last); int lowUsed = totaldays - compliant; int daysSkipped = realTotal -(compliant+lowUsed); //float comperc = (100.0 / float(totaldays)) * float(compliant); float comperc = (100.0 / float(realTotal)) * float(compliant); html += ""+tr("Period:")+" "; html += first.toString(Qt::SystemLocaleShortDate) + " - " + last.toString(Qt::SystemLocaleShortDate) + "

"; html += ""+tr("CPAP Usage")+"
"; if (realTotal != totaldays) { html += tr("Total Days: %1").arg(realTotal) + "
"; html += tr("No Data Days: %1").arg(daysSkipped) + "
"; } html += tr("Days Used: %1").arg(totaldays) + "
"; html += tr("Low Use Days: %1").arg(lowUsed) + "
"; //html += tr("compliant Days: %1").arg(compliant) + "
"; html += tr("Compliance: %1%").arg(comperc, 0, 'f', 1) + "
"; ///////////////////////////////////////////////////////////////////////////////////// /// AHI Records ///////////////////////////////////////////////////////////////////////////////////// if (p_profile->session->preloadSummaries()) { const int show_records = 5; QMultiMap::iterator it; QMultiMap::iterator it_end; QMultiMap ahilist; int baddays = 0; for (QDate date = first; date <= last; date = date.addDays(1)) { Day * day = p_profile->GetDay(date, MT_CPAP); if (!day) continue; float ahi = day->calcAHI(); if (ahi >= 5) { baddays++; } ahilist.insert(ahi, date); } html += tr("Days AHI of 5 or greater: %1").arg(baddays) + "

"; if (ahilist.size() > (show_records * 2)) { it = ahilist.begin(); it_end = ahilist.end(); html += ""+tr("Best AHI")+"
"; for (int i=0; (i").arg(it.value().toString(Qt::ISODate)) +tr("Date: %1 AHI: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
"; } html += "
"; html += ""+tr("Worst AHI")+"
"; it = ahilist.end() - 1; it_end = ahilist.begin(); for (int i=0; (i").arg(it.value().toString(Qt::ISODate)) +tr("Date: %1 AHI: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
"; } html += "
"; } ///////////////////////////////////////////////////////////////////////////////////// /// Flow Limitation Records ///////////////////////////////////////////////////////////////////////////////////// ahilist.clear(); for (QDate date = first; date <= last; date = date.addDays(1)) { Day * day = p_profile->GetDay(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 += ""+tr("Best Flow Limitation")+"
"; for (int i=0; (i").arg(it.value().toString(Qt::ISODate)) +tr("Date: %1 FL: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
"; } html += "
"; html += ""+tr("Worst Flow Limtation")+"
"; it = ahilist.end() - 1; it_end = ahilist.begin(); for (int i=0; (i 0) { html += QString("").arg(it.value().toString(Qt::ISODate)) +tr("Date: %1 FL: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
"; cnt++; } } if (cnt == 0) { html+= ""+tr("No Flow Limitation on record")+"
"; } html += "
"; } ///////////////////////////////////////////////////////////////////////////////////// /// Large Leak Records ///////////////////////////////////////////////////////////////////////////////////// ahilist.clear(); for (QDate date = first; date <= last; date = date.addDays(1)) { Day * day = p_profile->GetDay(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 += ""+tr("Worst Large Leaks")+"
"; it = ahilist.end() - 1; it_end = ahilist.begin(); for (int i=0; (i 0) { html += QString("").arg(it.value().toString(Qt::ISODate)) +tr("Date: %1 Leak: %2%").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
"; cnt++; } } if (cnt == 0) { html+= ""+tr("No Large Leaks on record")+"
"; } html += "
"; } ///////////////////////////////////////////////////////////////////////////////////// /// CSR 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->GetDay(date, MT_CPAP); if (!day) continue; float leak = day->calcPON(CPAP_CSR); ahilist.insert(leak, date); } if (ahilist.size() > (show_records * 2)) { html += ""+tr("Worst CSR")+"
"; it = ahilist.end() - 1; it_end = ahilist.begin(); for (int i=0; (i 0) { html += QString("").arg(it.value().toString(Qt::ISODate)) +tr("Date: %1 CSR: %2%").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
"; cnt++; } } if (cnt == 0) { html+= ""+tr("No CSR on record")+"
"; } html += "
"; } } if (p_profile->hasChannel(CPAP_PB)) { ahilist.clear(); for (QDate date = first; date <= last; date = date.addDays(1)) { Day * day = p_profile->GetDay(date, MT_CPAP); if (!day) continue; float leak = day->calcPON(CPAP_PB); ahilist.insert(leak, date); } if (ahilist.size() > (show_records * 2)) { html += ""+tr("Worst PB")+"
"; 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("").arg(it.value().toString(Qt::ISODate)) +tr("Date: %1 PB: %2%").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
"; cnt++; } } if (cnt == 0) { html+= ""+tr("No PB on record")+"
"; } html += "
"; } } } else { html += "
"+tr("Want more information?")+"
"; html += ""+tr("OSCAR needs all summary data loaded to calculate best/worst data for individual days.")+"

"; html += ""+tr("Please enable Pre-Load Summaries checkbox in preferences to make sure this data is available.")+"

"; } ///////////////////////////////////////////////////////////////////////////////////// /// Sort the RX list to get best and worst settings. ///////////////////////////////////////////////////////////////////////////////////// QList list; QMap::iterator ri_end = rxitems.end(); QMap::iterator ri; for (ri = rxitems.begin(); ri != ri_end; ++ri) { list.append(&ri.value()); ri.value().highlight = 0; } std::sort(list.begin(), list.end(), rxAHILessThan); if (list.size() >= 2) { html += ""+tr("Best Device Setting")+"
"; const RXItem & rxbest = *list.at(0); html += QString("").arg(rxbest.start.toString(Qt::ISODate)).arg(rxbest.end.toString(Qt::ISODate)) + tr("Date: %1 - %2").arg(rxbest.start.toString(Qt::SystemLocaleShortDate)).arg(rxbest.end.toString(Qt::SystemLocaleShortDate)) + "
"; html += QString("%1").arg(rxbest.machine->model()) + "
"; html += QString("Serial: %1").arg(rxbest.machine->serial()) + "
"; html += tr("AHI: %1").arg(double(rxbest.ahi) / rxbest.hours, 0, 'f', 2) + "
"; html += tr("Total Hours: %1").arg(rxbest.hours, 0, 'f', 2) + "
"; html += QString("%1").arg(rxbest.pressure) + "
"; html += QString("%1").arg(formatRelief(rxbest.relief)) + "
"; html += "
"; html += ""+tr("Worst Device Setting")+"
"; const RXItem & rxworst = *list.at(list.size() -1); html += QString("").arg(rxworst.start.toString(Qt::ISODate)).arg(rxworst.end.toString(Qt::ISODate)) + tr("Date: %1 - %2").arg(rxworst.start.toString(Qt::SystemLocaleShortDate)).arg(rxworst.end.toString(Qt::SystemLocaleShortDate)) + "
"; html += QString("%1").arg(rxworst.machine->model()) + "
"; html += QString("Serial: %1").arg(rxworst.machine->serial()) + "
"; html += tr("AHI: %1").arg(double(rxworst.ahi) / rxworst.hours, 0, 'f', 2) + "
"; html += tr("Total Hours: %1").arg(rxworst.hours, 0, 'f', 2) + "
"; html += QString("%1").arg(rxworst.pressure) + "
"; html += QString("%1").arg(formatRelief(rxworst.relief)) + "
"; } } html += ""; return html; } QString StatisticsRow::value(QDate start, QDate end) { const int decimals=2; QString value; float days = p_profile->countDays(type, start, end); float percentile=p_profile->general->prefCalcPercentile()/100.0; // Pholynyk, 10Mar2016 EventDataType percent = percentile; // was 0.90F // 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; float realDays = qAbs(start.daysTo(end)) + 1; float p = (100.0 / realDays) * 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(); EventDataType val = 0; QString fmt = "%1"; if (code != NoChannel) { switch(calc) { case SC_AVG: val = p_profile->calcAvg(code, type, start, end); break; case SC_WAVG: val = p_profile->calcWavg(code, type, start, end); break; case SC_MEDIAN: val = p_profile->calcPercentile(code, 0.5F, type, start, end); break; case SC_90P: val = p_profile->calcPercentile(code, percent, type, start, end); break; case SC_MIN: val = p_profile->calcMin(code, type, start, end); break; case SC_MAX: val = p_profile->calcMax(code, type, start, end); break; case SC_CPH: val = p_profile->calcCount(code, type, start, end) / p_profile->calcHours(type, start, end); break; case SC_SPH: fmt += "%"; val = 100.0 / p_profile->calcHours(type, start, end) * p_profile->calcSum(code, type, start, end) / 3600.0; break; case SC_ABOVE: fmt += "%"; val = 100.0 / p_profile->calcHours(type, start, end) * (p_profile->calcAboveThreshold(code, schema::channel[code].upperThreshold(), type, start, end) / 60.0); break; case SC_BELOW: fmt += "%"; val = 100.0 / p_profile->calcHours(type, start, end) * (p_profile->calcBelowThreshold(code, schema::channel[code].lowerThreshold(), type, start, end) / 60.0); break; default: break; }; } if ((val == std::numeric_limits::min()) || (val == std::numeric_limits::max())) { value = "Err"; } else { value = fmt.arg(val, 0, 'f', decimals); } } return value; }