2018-06-14 07:25:54 +00:00
|
|
|
/* Statistics Report Generator Implementation
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
2021-11-02 20:34:12 +00:00
|
|
|
* Copyright (c) 2019-2022 The OSCAR Team
|
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
|
|
|
|
2023-05-26 21:37:19 +00:00
|
|
|
#define TEST_MACROS_ENABLEDoff
|
|
|
|
#include <test_macros.h>
|
|
|
|
|
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>
|
|
|
|
|
2019-06-11 23:43:13 +00:00
|
|
|
#include <QPrinter>
|
|
|
|
#include <QPrintDialog>
|
|
|
|
#include <QPainter>
|
|
|
|
#include <QMainWindow>
|
|
|
|
|
2013-09-14 23:32:14 +00:00
|
|
|
#include "mainwindow.h"
|
2014-04-25 05:28:10 +00:00
|
|
|
#include "statistics.h"
|
2019-08-06 19:33:48 +00:00
|
|
|
#include "cprogressbar.h"
|
2019-08-06 17:51:14 +00:00
|
|
|
#include "SleepLib/common.h"
|
2020-01-15 21:34:28 +00:00
|
|
|
#include "version.h"
|
2023-07-08 15:13:35 +00:00
|
|
|
#include "SleepLib/profiles.h"
|
2013-09-14 23:32:14 +00:00
|
|
|
|
|
|
|
extern MainWindow *mainwin;
|
|
|
|
|
2019-06-11 18:34:00 +00:00
|
|
|
// HTML components that make up Statistics page and printed report
|
2019-06-12 20:32:22 +00:00
|
|
|
QString htmlReportHeader = ""; // Page header
|
2021-12-10 07:54:14 +00:00
|
|
|
QString htmlReportHeaderPrint = ""; // Page header
|
2019-06-11 18:34:00 +00:00
|
|
|
QString htmlUsage = ""; // CPAP and Oximetry
|
2022-02-27 14:18:39 +00:00
|
|
|
QString htmlMachineSettings = ""; // Device (formerly Rx) changes
|
|
|
|
QString htmlMachines = ""; // Devices used in this profile
|
2019-06-11 18:34:00 +00:00
|
|
|
QString htmlReportFooter = ""; // Page footer
|
|
|
|
|
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");
|
2020-06-10 21:34:18 +00:00
|
|
|
return QString("<img src='data:image/png;base64,"+byteArray.toBase64()+"' ALT='logo'>");
|
2018-05-28 22:35:48 +00:00
|
|
|
}
|
|
|
|
|
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;
|
2023-03-01 12:51:45 +00:00
|
|
|
return QString::asprintf("%02i:%02i", hours, minutes); //,seconds);
|
2014-05-06 09:11:31 +00:00
|
|
|
}
|
|
|
|
|
2019-05-11 00:53:09 +00:00
|
|
|
|
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)) {
|
2020-08-20 00:09:41 +00:00
|
|
|
qDebug() << "Could not open" << path << "for reading, error code" << file.error() << file.errorString();
|
2014-09-11 14:23:08 +00:00
|
|
|
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;
|
|
|
|
|
2023-09-08 09:22:12 +00:00
|
|
|
{ // Bug Fix. during testing, a crash occured due to a null machie value. in saveRxChanges. out << rx.machine->loaderName()
|
|
|
|
QList<QDate> toErase;
|
|
|
|
for (auto ri = rxitems.begin(); ri != rxitems.end();++ri ) {
|
|
|
|
RXItem rxitem = ri.value();
|
|
|
|
if (rxitem.machine==0) toErase.append(ri.key());
|
|
|
|
}
|
|
|
|
for (auto date : toErase) {
|
|
|
|
rxitems.remove(date) ;
|
|
|
|
}
|
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
2023-09-08 09:22:12 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
void Statistics::saveRXChanges()
|
|
|
|
{
|
|
|
|
QString path = p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" );
|
|
|
|
QFile file(path);
|
|
|
|
if (!file.open(QFile::WriteOnly)) {
|
2020-08-09 17:33:04 +00:00
|
|
|
qWarning() << "Could not open" << path << "for writing, error code" << file.error() << file.errorString();
|
2014-09-11 14:23:08 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2023-06-05 01:53:56 +00:00
|
|
|
void Statistics::updateDisabledInfo()
|
|
|
|
{
|
2023-06-13 16:32:51 +00:00
|
|
|
QDate lastcpap = p_profile->LastGoodDay(MT_CPAP);
|
|
|
|
QDate firstcpap = p_profile->FirstGoodDay(MT_CPAP);
|
2023-06-13 23:25:55 +00:00
|
|
|
if (lastcpap > p_profile->general->statReportDate() ) {
|
|
|
|
lastcpap = p_profile->general->statReportDate();
|
2023-06-13 16:32:51 +00:00
|
|
|
}
|
|
|
|
disabledInfo.update( lastcpap, firstcpap );
|
2023-06-05 01:53:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DisabledInfo::update(QDate latestDate, QDate earliestDate)
|
|
|
|
{
|
|
|
|
clear();
|
2023-06-09 16:47:10 +00:00
|
|
|
if ( (!latestDate.isValid()) || (!earliestDate.isValid()) || (p_profile->cpap->clinicalMode()) ) return;
|
2023-06-08 16:32:51 +00:00
|
|
|
qint64 complianceHours = 3600000.0 * p_profile->cpap->complianceHours(); // conbvert to ms
|
2023-06-05 01:53:56 +00:00
|
|
|
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;};
|
2023-06-08 16:32:51 +00:00
|
|
|
// find basic statistics for a day
|
2023-06-05 01:53:56 +00:00
|
|
|
int numDisabled=0;
|
|
|
|
qint64 sessLength = 0;
|
|
|
|
qint64 dayLength = 0;
|
2023-06-08 16:32:51 +00:00
|
|
|
qint64 enabledLength = 0;
|
2023-06-05 01:53:56 +00:00
|
|
|
QList<Session *> sessions = day->getSessions(MT_CPAP,true);
|
|
|
|
for (auto & sess : sessions) {
|
|
|
|
sessLength = sess->length();
|
2023-06-09 18:47:35 +00:00
|
|
|
//if (sessLength<0) sessLength=0; // some sessions have negative length. Sould solve this issue
|
2023-06-05 01:53:56 +00:00
|
|
|
dayLength += sessLength;
|
2023-06-08 16:32:51 +00:00
|
|
|
if (sess->enabled(true)) {
|
|
|
|
enabledLength += sessLength;
|
|
|
|
} else {
|
2023-06-05 01:53:56 +00:00
|
|
|
numDisabled ++;
|
|
|
|
totalDurationOfDisabledSessions += sessLength;
|
|
|
|
if (maxDurationOfaDisabledsession < sessLength) maxDurationOfaDisabledsession = sessLength;
|
|
|
|
}
|
|
|
|
}
|
2023-06-08 16:32:51 +00:00
|
|
|
// calculate stats for all days
|
|
|
|
// calculate if compliance for a day changed.
|
|
|
|
if ( complianceHours <= enabledLength ) {
|
2023-06-05 01:53:56 +00:00
|
|
|
daysInCompliance ++;
|
|
|
|
} else {
|
2023-06-08 16:32:51 +00:00
|
|
|
if (complianceHours < dayLength ) {
|
2023-06-05 01:53:56 +00:00
|
|
|
numDaysDisabledSessionChangedCompliance++;
|
|
|
|
} else {
|
|
|
|
daysOutOfCompliance ++;
|
|
|
|
}
|
|
|
|
}
|
2023-06-08 16:32:51 +00:00
|
|
|
// update disabled info for all days
|
2023-06-05 01:53:56 +00:00
|
|
|
if ( numDisabled > 0 ) {
|
2023-06-08 16:32:51 +00:00
|
|
|
numDisabledsessions += numDisabled;
|
2023-06-05 01:53:56 +00:00
|
|
|
numDaysWithDisabledsessions++;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
// convect ms to minutes
|
|
|
|
maxDurationOfaDisabledsession/=60000 ;
|
|
|
|
totalDurationOfDisabledSessions/=60000 ;
|
|
|
|
};
|
|
|
|
|
|
|
|
QString DisabledInfo::display(int type)
|
|
|
|
{
|
|
|
|
/*
|
2023-06-08 16:32:51 +00:00
|
|
|
Permissive mode: some sessions are excluded from this report, as follows:
|
|
|
|
Total disabled sessions: xx, found in yy days.
|
2023-06-05 01:53:56 +00:00
|
|
|
Duration of longest disabled session: aa minutes, Total duration of all disabled sessions: bb minutes.
|
2023-06-09 18:47:35 +00:00
|
|
|
+tr("Date: %1 AHI: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "</a><br>";
|
2023-06-05 01:53:56 +00:00
|
|
|
*/
|
|
|
|
switch (type) {
|
|
|
|
default :
|
|
|
|
case 0:
|
2023-06-08 17:37:31 +00:00
|
|
|
//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"));
|
2023-06-05 01:53:56 +00:00
|
|
|
case 1:
|
2023-06-08 16:32:51 +00:00
|
|
|
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) );
|
|
|
|
}
|
2023-06-05 01:53:56 +00:00
|
|
|
case 2:
|
|
|
|
return QString(QObject::tr( "Duration of longest disabled session: %1 minutes, Total duration of all disabled sessions: %2 minutes.")
|
2023-06-09 18:47:35 +00:00
|
|
|
.arg(maxDurationOfaDisabledsession, 0, 'f', 1) .arg(totalDurationOfDisabledSessions, 0, 'f', 1));
|
2023-06-05 01:53:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
void Statistics::updateRXChanges()
|
|
|
|
{
|
2019-08-04 15:36:34 +00:00
|
|
|
// Set conditional progress bar.
|
|
|
|
CProgressBar * progress = new CProgressBar (QObject::tr("Updating Statistics cache"), mainwin, p_profile->daylist.count());
|
2019-07-28 15:18:59 +00:00
|
|
|
|
2019-08-04 15:36:34 +00:00
|
|
|
// Clear loaded rx cache
|
2014-09-11 14:23:08 +00:00
|
|
|
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
|
|
|
|
2019-08-04 15:36:34 +00:00
|
|
|
progress->add (1); // Increment progress bar
|
|
|
|
|
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
|
|
|
|
2019-06-30 15:57:16 +00:00
|
|
|
if (day->first() == 0) { // Ignore invalid dates
|
2023-05-28 00:09:15 +00:00
|
|
|
//qDebug() << "Statistics::updateRXChanges ignoring day with first=0";
|
2019-06-30 15:57:16 +00:00
|
|
|
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();
|
2019-09-02 05:14:36 +00:00
|
|
|
QString mode = day->getCPAPModeStr();
|
2014-09-11 14:23:08 +00:00
|
|
|
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
|
2021-07-25 04:12:15 +00:00
|
|
|
tmp = day->count(AllAhiChannels);
|
2014-09-11 14:23:08 +00:00
|
|
|
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
|
2021-07-25 04:12:15 +00:00
|
|
|
tmp = day->count(AllAhiChannels);
|
2014-09-11 14:23:08 +00:00
|
|
|
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
|
2021-07-25 04:12:15 +00:00
|
|
|
tmp = dy->count(AllAhiChannels);
|
2014-09-11 14:23:08 +00:00
|
|
|
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
|
2021-07-25 04:12:15 +00:00
|
|
|
tmp = dy->count(AllAhiChannels);
|
2014-09-11 14:23:08 +00:00
|
|
|
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
|
|
|
|
2019-05-11 00:53:09 +00:00
|
|
|
// Generate pressure/mode/`strings
|
2014-09-11 14:23:08 +00:00
|
|
|
QString relief = day->getPressureRelief();
|
2019-09-02 05:14:36 +00:00
|
|
|
QString mode = day->getCPAPModeStr();
|
2014-09-11 14:23:08 +00:00
|
|
|
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
|
2021-07-25 04:12:15 +00:00
|
|
|
tmp = day->count(AllAhiChannels);
|
2016-03-02 23:27:23 +00:00
|
|
|
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
|
2021-07-25 04:12:15 +00:00
|
|
|
tmp = day->count(AllAhiChannels);
|
2014-09-11 14:23:08 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2018-06-07 22:09:05 +00:00
|
|
|
std::sort(list.begin(), list.end(), rxAHILessThan);
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-08-04 15:36:34 +00:00
|
|
|
// Close the progress bar
|
|
|
|
progress->close();
|
2020-01-31 00:52:37 +00:00
|
|
|
delete progress;
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2019-05-27 18:22:38 +00:00
|
|
|
// 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.
|
2022-02-27 14:18:39 +00:00
|
|
|
// Also creates "device" names for device types.
|
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));
|
2023-06-09 16:47:10 +00:00
|
|
|
if (!p_profile->cpap->clinicalMode()) {
|
2023-06-05 01:53:56 +00:00
|
|
|
updateDisabledInfo();
|
|
|
|
rows.push_back(StatisticsRow(disabledInfo.display(0),SC_WARNING ,MT_CPAP));
|
|
|
|
rows.push_back(StatisticsRow(disabledInfo.display(1),SC_WARNING2,MT_CPAP));
|
2023-06-08 16:32:51 +00:00
|
|
|
if (disabledInfo.size()>0) {
|
|
|
|
rows.push_back(StatisticsRow(disabledInfo.display(2),SC_WARNING2,MT_CPAP));
|
|
|
|
}
|
2023-06-05 01:53:56 +00:00
|
|
|
}
|
2014-05-06 17:39:05 +00:00
|
|
|
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));
|
2023-11-10 18:54:49 +00:00
|
|
|
rows.push_back(StatisticsRow(tr("Days In Period"), SC_SELECTED_DAYS , MT_CPAP));
|
|
|
|
rows.push_back(StatisticsRow(tr("Days Used"), SC_DAYS_W_DATA , MT_CPAP));
|
|
|
|
rows.push_back(StatisticsRow(tr("Days Not Used"), SC_DAYS_WO_DATA , MT_CPAP));
|
|
|
|
rows.push_back(StatisticsRow(tr("Days (%1 hrs/day)"), SC_COMPLIANCE_DAYS , MT_CPAP));
|
|
|
|
rows.push_back(StatisticsRow(tr("Percent Days (%1 hrs/day)"), SC_COMPLIANCE, MT_CPAP));
|
2014-05-06 17:39:05 +00:00
|
|
|
rows.push_back(StatisticsRow(tr("Average Hours per Night"), SC_HOURS, MT_CPAP));
|
|
|
|
|
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));
|
2021-07-25 04:12:15 +00:00
|
|
|
rows.push_back(StatisticsRow("AllApnea", SC_CPH, MT_CPAP));
|
2014-05-06 17:39:05 +00:00
|
|
|
rows.push_back(StatisticsRow("Obstructive", SC_CPH, MT_CPAP));
|
|
|
|
rows.push_back(StatisticsRow("Hypopnea", SC_CPH, MT_CPAP));
|
2021-07-17 23:20:44 +00:00
|
|
|
rows.push_back(StatisticsRow("Apnea", SC_CPH, MT_CPAP));
|
2014-05-06 17:39:05 +00:00
|
|
|
rows.push_back(StatisticsRow("ClearAirway", SC_CPH, MT_CPAP));
|
|
|
|
rows.push_back(StatisticsRow("FlowLimit", SC_CPH, MT_CPAP));
|
2021-11-04 01:06:50 +00:00
|
|
|
rows.push_back(StatisticsRow("FLG", SC_90P, MT_CPAP));
|
2014-05-06 17:39:05 +00:00
|
|
|
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));
|
2019-12-25 06:12:49 +00:00
|
|
|
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));
|
2014-05-06 17:39:05 +00:00
|
|
|
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));
|
2019-12-25 06:12:49 +00:00
|
|
|
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));
|
2014-05-06 17:39:05 +00:00
|
|
|
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));
|
2019-12-25 06:12:49 +00:00
|
|
|
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));
|
2014-05-06 17:39:05 +00:00
|
|
|
|
2019-06-16 22:30:39 +00:00
|
|
|
rows.push_back(StatisticsRow("", SC_HEADING, MT_OXIMETER)); // Just adds some space
|
2014-05-06 17:39:05 +00:00
|
|
|
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];
|
2023-05-26 21:37:19 +00:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2019-05-27 18:22:38 +00:00
|
|
|
// Get the user information block for displaying at top of page
|
2019-05-22 20:49:28 +00:00
|
|
|
QString Statistics::getUserInfo () {
|
2020-06-09 17:27:11 +00:00
|
|
|
if (!AppSetting->showPersonalData())
|
2020-06-05 04:25:23 +00:00
|
|
|
return "";
|
|
|
|
|
2014-07-11 12:09:38 +00:00
|
|
|
QString address = p_profile->user->address();
|
2020-06-10 21:34:18 +00:00
|
|
|
address.replace("\n", "<br>");
|
2014-05-06 09:33:06 +00:00
|
|
|
|
2019-06-15 18:58:02 +00:00
|
|
|
QString userinfo = "";
|
2014-05-06 09:33:06 +00:00
|
|
|
|
2014-07-11 12:09:38 +00:00
|
|
|
if (!p_profile->user->firstName().isEmpty()) {
|
2020-06-10 21:34:18 +00:00
|
|
|
userinfo = tr("Name: %1, %2").arg(p_profile->user->lastName()).arg(p_profile->user->firstName()) + "<br>";
|
2014-07-11 12:09:38 +00:00
|
|
|
if (!p_profile->user->DOB().isNull()) {
|
2020-06-10 21:34:18 +00:00
|
|
|
userinfo += tr("DOB: %1").arg(p_profile->user->DOB().toString(MedDateFormat)) + "<br>";
|
2014-05-06 09:33:06 +00:00
|
|
|
}
|
2014-07-11 12:09:38 +00:00
|
|
|
if (!p_profile->user->phone().isEmpty()) {
|
2020-06-10 21:34:18 +00:00
|
|
|
userinfo += 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()) {
|
2020-06-10 21:34:18 +00:00
|
|
|
userinfo += 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()) {
|
2020-06-10 21:34:18 +00:00
|
|
|
userinfo += tr("Address:")+"<br>"+address;
|
2014-05-06 12:37:00 +00:00
|
|
|
}
|
2014-05-06 09:33:06 +00:00
|
|
|
}
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
while (userinfo.length() > 0 && userinfo.endsWith("<br>")) // Strip trailing newlines
|
2020-06-11 18:01:28 +00:00
|
|
|
userinfo = userinfo.mid(0, userinfo.length()-4);
|
2019-06-15 18:58:02 +00:00
|
|
|
|
2019-05-22 20:49:28 +00:00
|
|
|
return userinfo;
|
|
|
|
}
|
2014-05-06 09:33:06 +00:00
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
const QString table_width = "width='100%'";
|
2019-05-27 18:22:38 +00:00
|
|
|
// Create the page header in HTML. Includes everything from <head> through <body>
|
2019-06-20 23:13:04 +00:00
|
|
|
QString Statistics::generateHeader(bool onScreen)
|
2019-05-22 20:49:28 +00:00
|
|
|
{
|
2020-06-10 21:34:18 +00:00
|
|
|
QString html = QString("<html><head>");
|
|
|
|
html += "<title>Oscar Statistics Report</title>";
|
|
|
|
html += "<style type='text/css'>";
|
|
|
|
|
2019-06-20 23:13:04 +00:00
|
|
|
if (onScreen) {
|
|
|
|
html += "p,a,td,body { font-family: '" + QApplication::font().family() + "'; }"
|
|
|
|
"p,a,td,body { font-size: " + QString::number(QApplication::font().pointSize() + 2) + "px; }";
|
|
|
|
} else {
|
2020-06-11 18:01:28 +00:00
|
|
|
html += "p,a,td,body { font-family: 'Helvetica'; }";
|
2019-06-20 23:13:04 +00:00
|
|
|
// "p,a,td,body { font-size: 10px; }";
|
|
|
|
}
|
2019-07-25 13:27:02 +00:00
|
|
|
// qDebug() << "generateHeader font" << html;
|
2019-06-20 23:13:04 +00:00
|
|
|
html += "table.curved {" // Borders not supported without webkit
|
|
|
|
// "border: 1px solid gray;"
|
|
|
|
// "border-radius:10px;"
|
|
|
|
// "-moz-border-radius:10px;"
|
|
|
|
// "-webkit-border-radius:10px;"
|
|
|
|
// "page-break-after:auto;"
|
|
|
|
// "-fs-table-paginate: paginate;"
|
|
|
|
"}"
|
|
|
|
|
|
|
|
"tr.datarow:nth-child(even) {"
|
|
|
|
"background-color: #f8f8f8;"
|
|
|
|
"}"
|
|
|
|
"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>"
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
"<link rel='stylesheet' type='text/css' href='qrc:/docs/tooltips.css' >"
|
2019-06-20 23:13:04 +00:00
|
|
|
|
|
|
|
"<script type='text/javascript'>"
|
|
|
|
"function ChangeColor(tableRow, highLight)"
|
|
|
|
"{ tableRow.style.backgroundColor = highLight; }"
|
|
|
|
"function Go(url) { throw(url); }"
|
|
|
|
"</script>"
|
|
|
|
|
2014-07-25 07:53:48 +00:00
|
|
|
"</head>"
|
2019-05-22 20:49:28 +00:00
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
"<body>"; //leftmargin=0 topmargin=5 rightmargin=0>";
|
2018-05-28 22:35:48 +00:00
|
|
|
|
2019-05-22 20:49:28 +00:00
|
|
|
QPixmap logoPixmap(":/icons/logo-lg.png");
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
// html += "<div align=center><table class=curved width='100%'>"
|
|
|
|
html += "<div align=center><table class=curved " + table_width + ">"
|
2019-05-22 20:49:28 +00:00
|
|
|
"<tr>"
|
|
|
|
"<td align='left' valign='middle'>" + getUserInfo() + "</td>"
|
2019-06-15 18:58:02 +00:00
|
|
|
"<td align='right' valign='middle' width='200'>"
|
2020-06-10 21:34:18 +00:00
|
|
|
"<font size='+2'>" + STR_TR_OSCAR + " </font><br>"
|
2019-05-22 20:49:28 +00:00
|
|
|
"<font size='+1'>" + QObject::tr("Usage Statistics") + " </font>"
|
|
|
|
"</td>"
|
2020-06-10 21:34:18 +00:00
|
|
|
"<td align='right' valign='middle' width='110'>" + resizeHTMLPixmap(logoPixmap,80,80)+" <br>"
|
2019-05-22 20:49:28 +00:00
|
|
|
"</td>"
|
|
|
|
"</tr>"
|
|
|
|
"</table>"
|
2019-06-15 18:58:02 +00:00
|
|
|
"</div>";
|
2019-06-20 23:13:04 +00:00
|
|
|
|
2014-07-25 07:53:48 +00:00
|
|
|
return html;
|
2013-09-14 23:32:14 +00:00
|
|
|
}
|
2019-05-27 18:22:38 +00:00
|
|
|
|
|
|
|
// HTML for page footer
|
2019-06-15 04:17:43 +00:00
|
|
|
QString Statistics::generateFooter(bool showinfo)
|
2013-09-14 23:32:14 +00:00
|
|
|
{
|
2014-07-25 07:53:48 +00:00
|
|
|
QString html;
|
|
|
|
|
|
|
|
if (showinfo) {
|
2019-05-22 20:49:28 +00:00
|
|
|
html += "<hr><div align=center><font size='-1'><i>";
|
2019-09-07 05:14:32 +00:00
|
|
|
QDateTime timestamp = QDateTime::currentDateTime();
|
Update version display throughout to use the new information and be consistent.
The full version now includes the build/git information embedded within
it as build metadata according to the Semantic Versioning 2.0.0 spec,
for example: "1.1.0-beta-1+branch-name-a1b2c3d".
Now the full version string, with all detail is always displayed
EXCEPT for release versions, in which case just the simple version
number ("1.1.0") is displayed in the primary UI.
- Main window title: simple version for release versions, full version
string otherwise
- Notifications: same as main window title
- System tray: same as main window title
- About window title: same as main window title
- About window release notes: always include full version string
- Reports: always include full version string
- Under the logo (about dialog, profile selector, new profile
window): removed, as it is largely redundant and can
interfere with the window geometry.
- Database upgrade alert: same as main window title
- Database newer alert: same as main window title
The full version string is also included within the preference and
profile .xml files, but because build metadata is ignored in version
comparisons, differences in builds will not cause any spurious
alerts. However, changes in prerelease versions will continue to
be significant, as they should be.
2020-01-16 18:05:55 +00:00
|
|
|
html += tr("This report was prepared on %1 by OSCAR %2").arg(timestamp.toString(MedDateFormat + " hh:mm"))
|
|
|
|
.arg(getVersion())
|
2020-06-10 21:34:18 +00:00
|
|
|
+ "<br>"
|
2019-09-07 05:14:32 +00:00
|
|
|
+ tr("OSCAR is free open-source CPAP report software");
|
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
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-27 18:22:38 +00:00
|
|
|
// Calculate AHI for a period as total # of events / total hours used
|
|
|
|
// Add RERA if calculating RDI instead of just AHI
|
2013-09-14 23:32:14 +00:00
|
|
|
EventDataType calcAHI(QDate start, QDate end)
|
|
|
|
{
|
2021-07-25 04:12:15 +00:00
|
|
|
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));
|
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
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-05-27 18:22:38 +00:00
|
|
|
// Calculate flow limits per hour
|
2013-09-14 23:32:14 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-05-27 18:22:38 +00:00
|
|
|
// Calculate ...(what are these?)
|
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;
|
|
|
|
}
|
|
|
|
|
2022-02-27 14:18:39 +00:00
|
|
|
// Structure for recording Prescription Changes (now called Device Settings Changes)
|
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 ©) {
|
|
|
|
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 ©) { date = copy.date; ahi = copy.ahi; hours = copy.hours; }
|
2013-09-14 23:32:14 +00:00
|
|
|
QDate date;
|
|
|
|
EventDataType ahi;
|
|
|
|
EventDataType hours;
|
|
|
|
};
|
2019-05-27 18:22:38 +00:00
|
|
|
|
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;
|
|
|
|
}
|
2023-06-12 20:19:14 +00:00
|
|
|
Period(QDate first,QDate last,bool& finished, int advance , bool month,QString name) {
|
|
|
|
if (finished) return;
|
2023-06-11 11:32:50 +00:00
|
|
|
// adds date range to header.
|
|
|
|
// replaces the following
|
|
|
|
// periods.push_back(Period(qMax(last.addDays(-6), first), last, tr("Last Week")));
|
2023-06-12 20:19:14 +00:00
|
|
|
QDate next;
|
2023-06-11 11:32:50 +00:00
|
|
|
if (month) {
|
|
|
|
// note add days or addmonths returns the start of the next day or the next month.
|
2023-06-12 20:19:14 +00:00
|
|
|
// must shorten one day for Month.
|
|
|
|
next = last.addMonths(advance).addDays(+1);
|
2023-06-11 11:32:50 +00:00
|
|
|
} else {
|
2023-06-12 20:19:14 +00:00
|
|
|
next = last.addDays(advance);
|
2023-06-11 11:32:50 +00:00
|
|
|
}
|
2023-06-12 20:19:14 +00:00
|
|
|
if (next<=first) {
|
|
|
|
finished = true;
|
|
|
|
next = first;
|
|
|
|
}
|
|
|
|
name = name + "<br>" + next.toString(Qt::SystemLocaleShortDate) ;
|
2023-06-11 11:32:50 +00:00
|
|
|
if (advance!=0) {
|
2023-06-11 12:48:49 +00:00
|
|
|
name = name + " - " + last.toString(Qt::SystemLocaleShortDate);
|
2023-06-11 11:32:50 +00:00
|
|
|
}
|
|
|
|
this->header = name;
|
2023-06-12 20:19:14 +00:00
|
|
|
this->start = next ;
|
2023-06-11 11:32:50 +00:00
|
|
|
this->end = last ;
|
|
|
|
}
|
2023-03-30 21:11:28 +00:00
|
|
|
Period& operator=(const Period&) = default;
|
|
|
|
~Period() {};
|
2014-05-06 11:54:41 +00:00
|
|
|
QDate start;
|
|
|
|
QDate end;
|
|
|
|
QString header;
|
|
|
|
};
|
|
|
|
|
2023-06-08 16:32:51 +00:00
|
|
|
const QString warning_color="#ffffff";
|
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
|
|
|
|
2022-02-27 14:18:39 +00:00
|
|
|
// Sort devices by first day of use
|
2019-09-16 18:56:40 +00:00
|
|
|
bool machineCompareFirstDay(Machine* left, Machine *right) {
|
|
|
|
return left->FirstDay() > right->FirstDay();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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
|
|
|
|
2019-09-16 18:56:40 +00:00
|
|
|
std::sort(cpap_machines.begin(), cpap_machines.end(), machineCompareFirstDay);
|
|
|
|
std::sort(oximeters.begin(), oximeters.end(), machineCompareFirstDay);
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
mach.append(cpap_machines);
|
|
|
|
mach.append(oximeters);
|
2014-05-06 09:11:31 +00:00
|
|
|
|
2019-09-16 18:56:40 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
QString html;
|
|
|
|
if (mach.size() > 0) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<div align=center><br>";
|
2014-04-17 05:52:25 +00:00
|
|
|
|
2020-06-10 21:34:18 +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>";
|
2022-02-27 14:18:39 +00:00
|
|
|
html += "<tr bgcolor='"+heading_color+"'><th colspan=7 align=center><font size='+2'>" + tr("Device Information") + "</font></th></tr>";
|
2014-05-06 09:11:31 +00:00
|
|
|
|
2019-08-18 21:27:21 +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></tr>")
|
2014-09-11 14:23:08 +00:00
|
|
|
.arg(STR_TR_Brand)
|
|
|
|
.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; }
|
2022-02-27 14:18:39 +00:00
|
|
|
//qDebug() << "Device" << m->brand() << "series" << m->series() << "model" << m->model() << "model number" << m->modelnumber();
|
2014-09-11 14:23:08 +00:00
|
|
|
QDate d1 = m->FirstDay();
|
|
|
|
QDate d2 = m->LastDay();
|
2023-06-13 23:25:55 +00:00
|
|
|
if (d2 > p_profile->general->statReportDate() ) {
|
|
|
|
d2 = p_profile->general->statReportDate();
|
2023-06-13 16:32:51 +00:00
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
QString mn = m->modelnumber();
|
2019-08-18 21:27:21 +00:00
|
|
|
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td></tr>")
|
2014-09-11 14:23:08 +00:00
|
|
|
.arg(m->brand())
|
|
|
|
.arg(m->model() +
|
|
|
|
(mn.isEmpty() ? "" : QString(" (") + mn + QString(")")))
|
|
|
|
.arg(m->serial())
|
2019-08-06 17:51:14 +00:00
|
|
|
.arg(d1.toString(MedDateFormat))
|
|
|
|
.arg(d2.toString(MedDateFormat));
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
}
|
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()
|
|
|
|
{
|
2022-02-27 14:18:39 +00:00
|
|
|
// Generate list only if there are CPAP devices
|
2019-08-28 22:22:31 +00:00
|
|
|
QList<Machine *> cpap_machines = p_profile->GetMachines(MT_CPAP);
|
|
|
|
if (cpap_machines.isEmpty())
|
|
|
|
return "";
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
QString html = "<div align=center><br>";
|
|
|
|
html += QString("<table class=curved style='page-break-before:always' " + table_width+">");
|
2014-09-11 14:23:08 +00:00
|
|
|
html += "<thead>";
|
2022-02-27 14:18:39 +00:00
|
|
|
html += "<tr bgcolor='"+heading_color+"'><th colspan=9 align=center><font size='+2'>" + tr("Changes to Device 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"));
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html+="<tr>";
|
2014-09-11 14:23:08 +00:00
|
|
|
for (int i=0; i < hdrlist.size(); ++i) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+=QString(" <th align=left><b>%1</b></th>").arg(hdrlist.at(i));
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
2020-06-10 21:34:18 +00:00
|
|
|
html+="</tr>";
|
2014-09-11 14:23:08 +00:00
|
|
|
html += "</thead>";
|
|
|
|
// html += "<tfoot>";
|
|
|
|
// html += "<tr><td colspan=10 align=center>";
|
|
|
|
// html += QString("<i>") +
|
|
|
|
// tr("Efficacy highlighting ignores prescription settings with less than %1 days of recorded data.").
|
2020-06-10 21:34:18 +00:00
|
|
|
// arg(rxthresh) + QString("</i><br>");
|
2014-09-11 14:23:08 +00:00
|
|
|
// html += "</td></tr>";
|
|
|
|
// html += "</tfoot>";
|
|
|
|
|
|
|
|
QMapIterator<QDate, RXItem> it(rxitems);
|
|
|
|
it.toBack();
|
|
|
|
while (it.hasPrevious()) {
|
|
|
|
it.previous();
|
|
|
|
const RXItem & rx = it.value();
|
2023-06-13 23:25:55 +00:00
|
|
|
if (rx.start > p_profile->general->statReportDate() ) continue;
|
2023-06-13 16:32:51 +00:00
|
|
|
QDate rxend=rx.end;
|
2023-06-13 23:25:55 +00:00
|
|
|
if (rxend > p_profile->general->statReportDate() ) rxend = p_profile->general->statReportDate();
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
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))
|
2023-06-13 16:32:51 +00:00
|
|
|
.arg(rxend.toString(Qt::ISODate))
|
2014-09-11 14:23:08 +00:00
|
|
|
.arg(datarowclass);
|
|
|
|
|
|
|
|
double ahi = rdi ? (double(rx.rdi) / rx.hours) : (double(rx.ahi) /rx.hours);
|
|
|
|
double fli = double(rx.count(CPAP_FlowLimit)) / rx. hours;
|
|
|
|
|
2019-09-01 00:06:47 +00:00
|
|
|
QString machid = QString("<td>%1 (%2)</td>").arg(rx.machine->model())
|
|
|
|
.arg(rx.machine->modelnumber());
|
|
|
|
if (AppSetting->includeSerial())
|
|
|
|
machid = QString("<td>%1 (%2) [%3]</td>").arg(rx.machine->model())
|
|
|
|
.arg(rx.machine->modelnumber())
|
|
|
|
.arg(rx.machine->serial());
|
|
|
|
|
2019-08-06 17:51:14 +00:00
|
|
|
html += QString("<td>%1</td>").arg(rx.start.toString(MedDateFormat))+
|
2023-06-13 16:32:51 +00:00
|
|
|
QString("<td>%1</td>").arg(rxend.toString(MedDateFormat))+
|
2014-09-11 14:23:08 +00:00
|
|
|
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)+
|
2019-09-01 00:06:47 +00:00
|
|
|
machid +
|
2019-05-11 00:53:09 +00:00
|
|
|
QString("<td>%1</td>").arg(formatRelief(rx.relief))+
|
2014-09-11 14:23:08 +00:00
|
|
|
QString("<td>%1</td>").arg(rx.mode)+
|
|
|
|
QString("<td>%1</td>").arg(rx.pressure)+
|
|
|
|
"</tr>";
|
|
|
|
}
|
|
|
|
html+="</table></div>";
|
|
|
|
|
|
|
|
return html;
|
|
|
|
}
|
|
|
|
|
2019-05-24 19:43:49 +00:00
|
|
|
// Report no data available
|
|
|
|
QString Statistics::htmlNoData()
|
|
|
|
{
|
|
|
|
QString html = "<div align=center>";
|
2020-06-10 21:34:18 +00:00
|
|
|
html += QString( "<p><font size=\"+3\"><br>" + tr("No data found?!?") + "</font></p>"+
|
|
|
|
"<p><img src='qrc:/icons/logo-lm.png' alt='logo' width='100' height='100'></p>"
|
2019-05-24 19:43:49 +00:00
|
|
|
"<p><i>"+tr("Oscar has no data to report :(")+"</i></p>");
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-06-12 20:32:22 +00:00
|
|
|
// Create the HTML for CPAP and Oximetry usage
|
|
|
|
QString Statistics::GenerateCPAPUsage()
|
2014-09-11 14:23:08 +00:00
|
|
|
{
|
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);
|
|
|
|
|
2022-02-27 14:18:39 +00:00
|
|
|
// Go through all CPAP and Oximeter devices and see if any data is present
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-11 18:34:00 +00:00
|
|
|
QString html = "";
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2019-05-24 19:43:49 +00:00
|
|
|
// If we don't have any data, return HTML that says that and we are done
|
|
|
|
if (!havedata) {
|
2019-06-12 20:32:22 +00:00
|
|
|
return "";
|
2019-05-24 19:43:49 +00:00
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
// Find first and last days with valid CPAP data
|
|
|
|
QDate lastcpap = p_profile->LastGoodDay(MT_CPAP);
|
|
|
|
QDate firstcpap = p_profile->FirstGoodDay(MT_CPAP);
|
2023-06-13 23:25:55 +00:00
|
|
|
if (lastcpap > p_profile->general->statReportDate() ) {
|
|
|
|
lastcpap = p_profile->general->statReportDate();
|
2023-06-13 16:32:51 +00:00
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2019-05-24 19:43:49 +00:00
|
|
|
QString ahitxt = getRDIorAHIText();
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2019-05-24 19:43:49 +00:00
|
|
|
// Prepare top of table
|
2014-04-17 05:52:25 +00:00
|
|
|
html += "<div align=center>";
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<table class=curved "+table_width+">";
|
2014-04-17 05:52:25 +00:00
|
|
|
|
2019-05-24 19:43:49 +00:00
|
|
|
// Compute number of monthly periods for a monthly rather than standard time distribution
|
2014-05-06 11:54:41 +00:00
|
|
|
int number_periods = 0;
|
2019-05-24 19:43:49 +00:00
|
|
|
if (p_profile->general->statReportMode() == STAT_MODE_MONTHLY) {
|
2019-06-11 13:28:34 +00:00
|
|
|
int firstMonth = firstcpap.month();
|
2019-05-24 19:43:49 +00:00
|
|
|
int lastMonth = lastcpap.month();
|
2019-07-19 01:16:05 +00:00
|
|
|
int years = lastcpap.year() - firstcpap.year();
|
2019-07-19 05:36:07 +00:00
|
|
|
lastMonth += (12 * years); // handle time extending to next year
|
2019-06-11 13:28:34 +00:00
|
|
|
number_periods = lastMonth - firstMonth + 1;
|
|
|
|
|
2019-05-27 18:22:38 +00:00
|
|
|
if (number_periods < 1) {
|
2019-06-11 13:28:34 +00:00
|
|
|
qDebug() << "*** Begin" << firstcpap << "beginMonth" << firstMonth << "lastMonth" << lastMonth << "periods" << number_periods;
|
2019-05-27 18:22:38 +00:00
|
|
|
number_periods = 1;
|
|
|
|
}
|
2019-07-19 01:16:05 +00:00
|
|
|
qDebug() << "Number of months for stats (trim to 12 max)" << number_periods;
|
2019-05-24 19:43:49 +00:00
|
|
|
// But not more than one year
|
2014-05-28 19:26:05 +00:00
|
|
|
if (number_periods > 12) {
|
|
|
|
number_periods = 12;
|
|
|
|
}
|
2021-12-10 07:54:14 +00:00
|
|
|
// } else if (p_profile->general->statReportMode() == STAT_MODE_RANGE) {
|
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-06 09:11:31 +00:00
|
|
|
bool skipsection = false;;
|
2019-05-27 18:22:38 +00:00
|
|
|
// Loop through all rows of the Statistics report
|
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);
|
2023-06-13 23:25:55 +00:00
|
|
|
if (last > p_profile->general->statReportDate() ) {
|
|
|
|
last = p_profile->general->statReportDate();
|
2023-06-13 16:32:51 +00:00
|
|
|
}
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2019-05-27 18:22:38 +00:00
|
|
|
// Clear the periods (columns)
|
2014-05-06 11:54:41 +00:00
|
|
|
periods.clear();
|
2019-05-24 19:43:49 +00:00
|
|
|
if (p_profile->general->statReportMode() == STAT_MODE_STANDARD) {
|
2023-06-11 11:32:50 +00:00
|
|
|
// note add days or addmonths returns the start of the next day or the next month.
|
2023-06-12 20:19:14 +00:00
|
|
|
// 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")));
|
2023-06-13 00:30:53 +00:00
|
|
|
periods.push_back(Period(first,last,finished, -6,true, tr("Last 6 Months")));
|
2023-06-12 20:19:14 +00:00
|
|
|
periods.push_back(Period(first,last,finished, -12,true,tr("Last Year")));
|
2021-12-10 07:54:14 +00:00
|
|
|
} else if (p_profile->general->statReportMode() == STAT_MODE_MONTHLY) {
|
2014-05-06 11:54:41 +00:00
|
|
|
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) {
|
2023-06-11 11:32:50 +00:00
|
|
|
periods.push_back(Period(s, l, s.toString("MMMM<br>yyyy")));
|
2014-05-28 19:26:05 +00:00
|
|
|
j++;
|
2014-05-06 11:54:41 +00:00
|
|
|
}
|
2023-06-11 00:23:47 +00:00
|
|
|
l = s.addDays(-1);
|
2014-05-28 19:26:05 +00:00
|
|
|
} while ((l > first) && (j < number_periods));
|
|
|
|
|
|
|
|
for (; j < number_periods; ++j) {
|
|
|
|
periods.push_back(Period(last,last, ""));
|
2014-05-06 11:54:41 +00:00
|
|
|
}
|
2021-12-10 07:54:14 +00:00
|
|
|
} else { // STAT_MODE_RANGE
|
|
|
|
first = p_profile->general->statReportRangeStart();
|
|
|
|
last = p_profile->general->statReportRangeEnd();
|
2023-06-13 13:51:43 +00:00
|
|
|
if (first > last) {
|
2023-06-13 16:32:51 +00:00
|
|
|
first = last;
|
2023-06-13 13:51:43 +00:00
|
|
|
}
|
2023-06-11 00:50:59 +00:00
|
|
|
periods.push_back(Period(first,last,first.toString(MedDateFormat)+" - "+last.toString(MedDateFormat)));
|
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) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+=QString("<tr bgcolor='%1'><th colspan=%2 align=center><font size='+2'>%3</font></th></tr>").
|
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;
|
2019-06-03 00:48:27 +00:00
|
|
|
} else if (row.calc == SC_HOURS) {
|
2014-05-06 09:11:31 +00:00
|
|
|
name = row.src;
|
2019-06-03 00:48:27 +00:00
|
|
|
} else if (row.calc == SC_COMPLIANCE) {
|
|
|
|
name = QString(row.src).arg(p_profile->cpap->m_complianceHours);
|
2014-05-06 09:11:31 +00:00
|
|
|
} else if (row.calc == SC_COLUMNHEADERS) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html += QString("<tr><td><b>%1</b></td>").arg(tr("Details"));
|
2014-05-06 11:54:41 +00:00
|
|
|
for (int j=0; j < periods.size(); j++) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html += QString("<td onmouseover='ChangeColor(this, \"#eeeeee\");' onmouseout='ChangeColor(this, \"#ffffff\");' onclick='alert(\"overview=%1,%2\");'><b>%3</b></td>").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
|
|
|
}
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "</tr>";
|
2014-05-06 09:11:31 +00:00
|
|
|
continue;
|
2023-11-10 18:54:49 +00:00
|
|
|
} else if (row.calc == SC_DAYS_WO_DATA) {
|
|
|
|
name = row.src;
|
|
|
|
} else if (row.calc == SC_DAYS_W_DATA) {
|
|
|
|
name = row.src;
|
|
|
|
} else if (row.calc == SC_SELECTED_DAYS) {
|
|
|
|
name = row.src;
|
|
|
|
} else if (row.calc == SC_COMPLIANCE_DAYS) {
|
|
|
|
name = QString(row.src).arg(p_profile->cpap->m_complianceHours);
|
2014-05-06 09:11:31 +00:00
|
|
|
} else if (row.calc == SC_DAYS) {
|
|
|
|
QDate first=p_profile->FirstGoodDay(row.type);
|
|
|
|
QDate last=p_profile->LastGoodDay(row.type);
|
2023-06-13 23:25:55 +00:00
|
|
|
if (last > p_profile->general->statReportDate() ) {
|
|
|
|
last = p_profile->general->statReportDate();
|
2023-06-13 16:32:51 +00:00
|
|
|
}
|
2014-05-06 09:11:31 +00:00
|
|
|
QString & machine = machinenames[row.type];
|
|
|
|
int value=p_profile->countDays(row.type, first, last);
|
|
|
|
|
|
|
|
if (value == 0) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+=QString("<tr><td colspan=%1 align=center>%2</td></tr>").arg(periods.size()+1).
|
2023-06-11 00:50:59 +00:00
|
|
|
arg(tr("Database has No %1 data available.").arg(machine));
|
2014-05-06 09:11:31 +00:00
|
|
|
} else if (value == 1) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+=QString("<tr><td colspan=%1 align=center>%2</td></tr>").arg(periods.size()+1).
|
2023-06-11 00:50:59 +00:00
|
|
|
arg(tr("Database has %1 day of %2 Data on %3")
|
2014-05-06 09:11:31 +00:00
|
|
|
.arg(value)
|
|
|
|
.arg(machine)
|
2019-08-06 17:51:14 +00:00
|
|
|
.arg(last.toString(MedDateFormat)));
|
2014-05-06 09:11:31 +00:00
|
|
|
} else {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+=QString("<tr><td colspan=%1 align=center>%2</td></tr>").arg(periods.size()+1).
|
2023-06-11 00:50:59 +00:00
|
|
|
arg(tr("Database has %1 days of %2 Data, between %3 and %4")
|
2014-05-06 09:11:31 +00:00
|
|
|
.arg(value)
|
|
|
|
.arg(machine)
|
2019-08-06 17:51:14 +00:00
|
|
|
.arg(first.toString(MedDateFormat))
|
|
|
|
.arg(last.toString(MedDateFormat)));
|
2014-05-06 09:11:31 +00:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
} else if (row.calc == SC_SUBHEADING) { // subheading..
|
2020-06-10 21:34:18 +00:00
|
|
|
html+=QString("<tr bgcolor='%1'><td colspan=%2 align=center><b>%3</b></td></tr>").
|
2014-05-06 11:54:41 +00:00
|
|
|
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;
|
2023-05-26 21:37:19 +00:00
|
|
|
} else if (row.calc == SC_WARNING) {
|
2023-06-08 16:32:51 +00:00
|
|
|
//html+=QString("<tr bgcolor='%1'><td colspan=%2 align=center><font size='+1'>%3</font></td></tr>").
|
|
|
|
// arg(warning_color).arg(periods.size()+1).arg(row.src);
|
|
|
|
html+=QString("<tr bgcolor='%1'><th colspan=%2 align=center><font size='+0'><i>%3</i></font></th></tr>").
|
2023-05-26 21:37:19 +00:00
|
|
|
arg(warning_color).arg(periods.size()+1).arg(row.src);
|
|
|
|
continue;
|
2023-06-05 01:53:56 +00:00
|
|
|
} else if (row.calc == SC_WARNING2) {
|
2023-06-08 16:32:51 +00:00
|
|
|
html+=QString("<tr bgcolor='%1'><th colspan=%2 align=center><font size='+0'><i>%3</i></font></th></tr>").
|
2023-06-05 01:53:56 +00:00
|
|
|
arg(warning_color).arg(periods.size()+1).arg(row.src);
|
|
|
|
continue;
|
2014-05-06 09:11:31 +00:00
|
|
|
} 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
|
|
|
}
|
2023-06-12 20:19:14 +00:00
|
|
|
// Defined percentages for columns for diffent modes.
|
2014-05-15 22:48:51 +00:00
|
|
|
QString line;
|
2014-05-28 19:26:05 +00:00
|
|
|
int np = periods.size();
|
|
|
|
int width;
|
2023-06-12 20:19:14 +00:00
|
|
|
// both create header column and 5 data columns for a total of 100
|
2020-06-10 21:34:18 +00:00
|
|
|
int dataWidth = 14;
|
|
|
|
int headerWidth = 30;
|
2023-11-10 18:54:49 +00:00
|
|
|
if (p_profile->general->statReportMode() == STAT_MODE_MONTHLY)
|
|
|
|
{
|
2023-06-12 20:19:14 +00:00
|
|
|
// both create header column and 13 data columns for a total of 100
|
2020-06-10 21:34:18 +00:00
|
|
|
dataWidth = 6;
|
|
|
|
headerWidth = 22;
|
|
|
|
}
|
|
|
|
line += QString("<tr class=datarow><td width='%1%'>%2</td>").arg(headerWidth).arg(name);
|
2014-05-28 19:26:05 +00:00
|
|
|
for (int j=0; j < np; j++) {
|
2020-06-10 21:34:18 +00:00
|
|
|
width = j < np-1 ? dataWidth : 100 - (headerWidth + dataWidth*(np-1));
|
|
|
|
line += QString("<td width='%1%'>").arg(width);
|
2014-05-28 19:26:05 +00:00
|
|
|
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;
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "</tr>";
|
2013-09-14 23:32:14 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 05:52:25 +00:00
|
|
|
html += "</table>";
|
|
|
|
html += "</div>";
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2019-06-12 20:32:22 +00:00
|
|
|
return html;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the HTML that will be the Statistics page.
|
|
|
|
QString Statistics::GenerateHTML()
|
|
|
|
{
|
2019-06-15 04:17:43 +00:00
|
|
|
htmlReportHeader = generateHeader(true);
|
2019-06-20 23:13:04 +00:00
|
|
|
htmlReportHeaderPrint = generateHeader(false);
|
2019-06-15 04:17:43 +00:00
|
|
|
htmlReportFooter = generateFooter(true);
|
2019-06-12 20:32:22 +00:00
|
|
|
|
|
|
|
htmlUsage = GenerateCPAPUsage();
|
|
|
|
|
|
|
|
if (htmlUsage == "") {
|
|
|
|
return htmlReportHeader + htmlNoData() + htmlReportFooter;
|
|
|
|
}
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2019-06-11 18:34:00 +00:00
|
|
|
htmlMachineSettings = GenerateRXChanges();
|
|
|
|
htmlMachines = GenerateMachineList();
|
2014-05-06 09:11:31 +00:00
|
|
|
|
2019-06-11 18:34:00 +00:00
|
|
|
QString htmlScript = "<script type='text/javascript' language='javascript' src='qrc:/docs/script.js'></script>";
|
2019-06-12 20:32:22 +00:00
|
|
|
|
|
|
|
return htmlReportHeader + htmlUsage + htmlMachineSettings + htmlMachines + htmlScript + htmlReportFooter;
|
2013-09-14 23:32:14 +00:00
|
|
|
}
|
2014-05-06 09:11:31 +00:00
|
|
|
|
2019-06-15 04:17:43 +00:00
|
|
|
// Print the Statistics page on printer
|
2019-06-11 23:43:13 +00:00
|
|
|
void Statistics::printReport(QWidget * parent) {
|
|
|
|
|
2019-06-15 18:58:02 +00:00
|
|
|
QPrinter printer(QPrinter::ScreenResolution); // ScreenResolution required for graphics sizing
|
2019-06-15 04:17:43 +00:00
|
|
|
|
2019-06-13 01:37:06 +00:00
|
|
|
#ifdef Q_OS_LINUX
|
2019-06-11 23:43:13 +00:00
|
|
|
printer.setPrinterName("Print to File (PDF)");
|
|
|
|
printer.setOutputFormat(QPrinter::PdfFormat);
|
2019-06-15 22:09:04 +00:00
|
|
|
QString name = "Statistics";
|
|
|
|
QString datestr = QDate::currentDate().toString(Qt::ISODate);
|
|
|
|
|
|
|
|
QString filename = p_pref->Get("{home}/") + name + "_" + p_profile->user->userName() + "_" + datestr + ".pdf";
|
2019-06-11 23:43:13 +00:00
|
|
|
|
|
|
|
printer.setOutputFileName(filename);
|
|
|
|
#endif
|
2019-06-15 04:17:43 +00:00
|
|
|
|
2019-06-11 23:43:13 +00:00
|
|
|
printer.setPrintRange(QPrinter::AllPages);
|
2023-02-17 22:09:18 +00:00
|
|
|
printer.setPageOrientation(QPageLayout::Portrait);
|
2019-06-15 18:58:02 +00:00
|
|
|
printer.setFullPage(false); // Print only on printable area of page and not in non-printable margins
|
2023-02-17 22:09:18 +00:00
|
|
|
printer.setCopyCount(1);
|
2020-06-11 18:01:28 +00:00
|
|
|
|
|
|
|
QMarginsF minMargins = printer.pageLayout().margins(QPageLayout::Millimeter);
|
2023-02-17 22:09:18 +00:00
|
|
|
|
|
|
|
printer.setPageMargins( QMarginsF( fmax(10,minMargins.left()), fmax(10,minMargins.top()), fmax(10,minMargins.right()), fmax(12,minMargins.bottom())), QPageLayout::Millimeter);
|
2020-06-11 18:01:28 +00:00
|
|
|
QMarginsF setMargins = printer.pageLayout().margins(QPageLayout::Millimeter);
|
|
|
|
qDebug () << "Min margins" << minMargins << "Set margins" << setMargins << "millimeters";
|
2019-06-15 04:17:43 +00:00
|
|
|
|
|
|
|
// Show print dialog to user and allow them to change settings as desired
|
2019-06-11 23:43:13 +00:00
|
|
|
QPrintDialog pdlg(&printer, parent);
|
|
|
|
|
|
|
|
if (pdlg.exec() == QPrintDialog::Accepted) {
|
|
|
|
|
2019-06-15 18:58:02 +00:00
|
|
|
QTextDocument doc;
|
2020-06-09 17:27:11 +00:00
|
|
|
QSizeF printArea = printer.pageRect(QPrinter::Point).size();
|
2020-06-11 18:01:28 +00:00
|
|
|
QSizeF originalPrintArea = printArea;
|
2020-06-09 19:07:55 +00:00
|
|
|
printArea.setWidth(printArea.width()*2); // scale up for better font appearance
|
|
|
|
printArea.setHeight(printArea.height()*2);
|
2020-06-09 17:27:11 +00:00
|
|
|
doc.setPageSize(printArea); // Set document to print area, in pixels, removing default 2cm margins
|
2019-06-16 22:30:39 +00:00
|
|
|
|
2020-06-11 18:01:28 +00:00
|
|
|
qDebug() << "print area (points)" << originalPrintArea << "Enlarged print area" << printArea << "paper size" << printer.paperRect(QPrinter::Point).size();
|
2020-06-09 17:27:11 +00:00
|
|
|
|
|
|
|
// Determine appropriate font and font size
|
2020-06-11 18:01:28 +00:00
|
|
|
QFont font = QFont("Helvetica");
|
2020-06-09 19:07:55 +00:00
|
|
|
float fontScalar = 12;
|
2020-06-09 17:27:11 +00:00
|
|
|
float printWidth = printArea.width();
|
2020-06-09 19:07:55 +00:00
|
|
|
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
|
2019-06-16 22:30:39 +00:00
|
|
|
doc.setDefaultFont(font);
|
2020-06-09 17:27:11 +00:00
|
|
|
|
2020-06-11 18:01:28 +00:00
|
|
|
qDebug() << "Enlarged printer font" << font << "printer default font set" << doc.defaultFont();
|
2019-06-16 22:30:39 +00:00
|
|
|
|
2022-01-14 13:39:00 +00:00
|
|
|
doc.setHtml(htmlReportHeaderPrint + htmlUsage + htmlReportFooter + htmlMachineSettings + htmlMachines + htmlReportFooter);
|
2019-06-16 22:30:39 +00:00
|
|
|
|
2020-06-11 18:01:28 +00:00
|
|
|
// Dump HTML for use with HTML4 validator
|
2020-06-10 21:46:55 +00:00
|
|
|
// QString html = htmlReportHeaderPrint + htmlUsage + htmlMachineSettings + htmlMachines + htmlReportFooter;
|
|
|
|
// qDebug() << "Html:" << html;
|
2020-06-10 21:34:18 +00:00
|
|
|
|
2019-06-15 04:17:43 +00:00
|
|
|
doc.print(&printer);
|
2019-06-11 23:43:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-12 20:32:22 +00:00
|
|
|
QString Statistics::UpdateRecordsBox()
|
2014-09-30 08:40:12 +00:00
|
|
|
{
|
|
|
|
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; }"
|
2020-06-10 21:34:18 +00:00
|
|
|
"</style>"
|
2022-02-27 14:18:39 +00:00
|
|
|
"<title>Device Statistics Panel</title>"
|
2020-06-10 21:34:18 +00:00
|
|
|
"</head><body>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
2023-06-15 19:33:33 +00:00
|
|
|
int totalDays = 1+first.daysTo(last);
|
|
|
|
int daysUsed = p_profile->countDays(MT_CPAP, first, last);
|
|
|
|
int daysSkipped = totalDays - daysUsed;
|
2023-11-10 18:54:49 +00:00
|
|
|
int compliant = p_profile->countCompliantDays(MT_CPAP, first, last );
|
2023-06-15 19:33:33 +00:00
|
|
|
int lowUsed = daysUsed - compliant;
|
|
|
|
float comperc = (100.0 / float(totalDays)) * float(compliant);
|
2014-09-30 08:40:12 +00:00
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<b>"+tr("CPAP Usage")+"</b><br>";
|
2023-06-15 19:33:33 +00:00
|
|
|
html += first.toString(Qt::SystemLocaleShortDate) + " - " + last.toString(Qt::SystemLocaleShortDate) + "<br>";
|
|
|
|
if (daysSkipped > 0) {
|
|
|
|
html += tr("Total Days: %1").arg(totalDays) + "<br>";
|
|
|
|
html += tr("Days Not Used: %1").arg(daysSkipped) + "<br>";
|
2023-06-11 22:15:46 +00:00
|
|
|
}
|
2023-06-15 19:33:33 +00:00
|
|
|
html += tr("Days Used: %1").arg(daysUsed) + "<br>";
|
|
|
|
html += tr("Days %1 %2 Hours: %3").arg(">=") .arg(p_profile->cpap->complianceHours(),0,'f',1) .arg(compliant) + "<br>";
|
|
|
|
html += tr("Days %1 %2 Hours: %3").arg("<").arg(p_profile->cpap->complianceHours(),0,'f',1) .arg(lowUsed) + "<br>";
|
2020-06-10 21:34:18 +00:00
|
|
|
html += tr("Compliance: %1%").arg(comperc, 0, 'f', 1) + "<br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// 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)) {
|
2023-06-13 16:32:51 +00:00
|
|
|
Day * day = p_profile->GetDay(date, MT_CPAP);
|
2014-09-30 08:40:12 +00:00
|
|
|
if (!day) continue;
|
|
|
|
|
|
|
|
float ahi = day->calcAHI();
|
|
|
|
if (ahi >= 5) {
|
|
|
|
baddays++;
|
|
|
|
}
|
|
|
|
ahilist.insert(ahi, date);
|
|
|
|
}
|
2020-06-10 21:34:18 +00:00
|
|
|
html += tr("Days AHI of 5 or greater: %1").arg(baddays) + "<br><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
if (ahilist.size() > (show_records * 2)) {
|
|
|
|
it = ahilist.begin();
|
|
|
|
it_end = ahilist.end();
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<b>"+tr("Best AHI")+"</b><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
for (int i=0; (i<show_records) && (it != it_end); ++i, ++it) {
|
|
|
|
html += QString("<a href='daily=%1'>").arg(it.value().toString(Qt::ISODate))
|
2020-06-10 21:34:18 +00:00
|
|
|
+tr("Date: %1 AHI: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "</a><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<b>"+tr("Worst AHI")+"</b><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
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))
|
2020-06-10 21:34:18 +00:00
|
|
|
+tr("Date: %1 AHI: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "</a><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// Flow Limitation Records
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
ahilist.clear();
|
|
|
|
for (QDate date = first; date <= last; date = date.addDays(1)) {
|
2023-06-13 16:32:51 +00:00
|
|
|
Day * day = p_profile->GetDay(date, MT_CPAP);
|
2014-09-30 08:40:12 +00:00
|
|
|
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();
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<b>"+tr("Best Flow Limitation")+"</b><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
for (int i=0; (i<show_records) && (it != it_end); ++i, ++it) {
|
|
|
|
html += QString("<a href='daily=%1'>").arg(it.value().toString(Qt::ISODate))
|
2020-06-10 21:34:18 +00:00
|
|
|
+tr("Date: %1 FL: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "</a><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<b>"+tr("Worst Flow Limtation")+"</b><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
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))
|
2020-06-10 21:34:18 +00:00
|
|
|
+tr("Date: %1 FL: %2").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "</a><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
cnt++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (cnt == 0) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+= "<i>"+tr("No Flow Limitation on record")+"</i><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// Large Leak Records
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
ahilist.clear();
|
|
|
|
for (QDate date = first; date <= last; date = date.addDays(1)) {
|
2023-06-13 16:32:51 +00:00
|
|
|
Day * day = p_profile->GetDay(date, MT_CPAP);
|
2014-09-30 08:40:12 +00:00
|
|
|
if (!day) continue;
|
|
|
|
|
|
|
|
float leak = day->calcPON(CPAP_LargeLeak);
|
|
|
|
ahilist.insert(leak, date);
|
|
|
|
}
|
|
|
|
|
|
|
|
cnt = 0;
|
|
|
|
if (ahilist.size() > (show_records * 2)) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<b>"+tr("Worst Large Leaks")+"</b><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
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))
|
2020-06-10 21:34:18 +00:00
|
|
|
+tr("Date: %1 Leak: %2%").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "</a><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
cnt++;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
if (cnt == 0) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+= "<i>"+tr("No Large Leaks on record")+"</i><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
2020-03-10 01:15:02 +00:00
|
|
|
/// CSR Records
|
2014-09-30 08:40:12 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
cnt = 0;
|
|
|
|
if (p_profile->hasChannel(CPAP_CSR)) {
|
|
|
|
ahilist.clear();
|
|
|
|
for (QDate date = first; date <= last; date = date.addDays(1)) {
|
2023-06-13 16:32:51 +00:00
|
|
|
Day * day = p_profile->GetDay(date, MT_CPAP);
|
2014-09-30 08:40:12 +00:00
|
|
|
if (!day) continue;
|
|
|
|
|
|
|
|
float leak = day->calcPON(CPAP_CSR);
|
|
|
|
ahilist.insert(leak, date);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ahilist.size() > (show_records * 2)) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<b>"+tr("Worst CSR")+"</b><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
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))
|
2020-06-10 21:34:18 +00:00
|
|
|
+tr("Date: %1 CSR: %2%").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "</a><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
cnt++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (cnt == 0) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+= "<i>"+tr("No CSR on record")+"</i><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
}
|
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)) {
|
2023-06-13 16:32:51 +00:00
|
|
|
Day * day = p_profile->GetDay(date, MT_CPAP);
|
2016-02-27 03:37:56 +00:00
|
|
|
if (!day) continue;
|
|
|
|
|
|
|
|
float leak = day->calcPON(CPAP_PB);
|
|
|
|
ahilist.insert(leak, date);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ahilist.size() > (show_records * 2)) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<b>"+tr("Worst PB")+"</b><br>";
|
2016-02-27 03:37:56 +00:00
|
|
|
|
|
|
|
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))
|
2020-06-10 21:34:18 +00:00
|
|
|
+tr("Date: %1 PB: %2%").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "</a><br>";
|
2016-02-27 03:37:56 +00:00
|
|
|
cnt++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (cnt == 0) {
|
2020-06-10 21:34:18 +00:00
|
|
|
html+= "<i>"+tr("No PB on record")+"</i><br>";
|
2016-02-27 03:37:56 +00:00
|
|
|
}
|
|
|
|
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<br>";
|
2016-02-27 03:37:56 +00:00
|
|
|
}
|
|
|
|
}
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
} else {
|
2020-06-10 21:34:18 +00:00
|
|
|
html += "<br><b>"+tr("Want more information?")+"</b><br>";
|
|
|
|
html += "<i>"+tr("OSCAR needs all summary data loaded to calculate best/worst data for individual days.")+"</i><br><br>";
|
|
|
|
html += "<i>"+tr("Please enable Pre-Load Summaries checkbox in preferences to make sure this data is available.")+"</i><br><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// 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;
|
|
|
|
}
|
|
|
|
|
2018-06-07 22:09:05 +00:00
|
|
|
std::sort(list.begin(), list.end(), rxAHILessThan);
|
2014-09-30 08:40:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
if (list.size() >= 2) {
|
2023-06-11 12:17:38 +00:00
|
|
|
html += "<b>"+tr("Best Device Setting")+"</b><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
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)) +
|
2020-06-10 21:34:18 +00:00
|
|
|
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 += tr("AHI: %1").arg(double(rxbest.ahi) / rxbest.hours, 0, 'f', 2) + "<br>";
|
|
|
|
html += tr("Total Hours: %1").arg(rxbest.hours, 0, 'f', 2) + "<br>";
|
|
|
|
html += QString("%1").arg(rxbest.pressure) + "<br>";
|
|
|
|
html += QString("%1").arg(formatRelief(rxbest.relief)) + "<br>";
|
|
|
|
html += "<br>";
|
|
|
|
|
2023-06-11 12:17:38 +00:00
|
|
|
html += "<b>"+tr("Worst Device Setting")+"</b><br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
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)) +
|
2020-06-10 21:34:18 +00:00
|
|
|
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 += tr("AHI: %1").arg(double(rxworst.ahi) / rxworst.hours, 0, 'f', 2) + "<br>";
|
|
|
|
html += tr("Total Hours: %1").arg(rxworst.hours, 0, 'f', 2) + "<br>";
|
|
|
|
|
|
|
|
html += QString("%1").arg(rxworst.pressure) + "<br>";
|
|
|
|
html += QString("%1").arg(formatRelief(rxworst.relief)) + "<br>";
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
html += "</body></html>";
|
2019-06-12 20:32:22 +00:00
|
|
|
|
|
|
|
return html;
|
2014-09-30 08:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
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) {
|
2023-11-10 18:54:49 +00:00
|
|
|
float c = p_profile->countCompliantDays(type, start, end );
|
2019-05-30 02:35:19 +00:00
|
|
|
// float p = (100.0 / days) * c;
|
|
|
|
float realDays = qAbs(start.daysTo(end)) + 1;
|
2023-11-10 18:54:49 +00:00
|
|
|
float p = (100.0 *c / realDays) ;
|
|
|
|
value = QString("%1%").arg(p, 0, 'f', 2);
|
|
|
|
} else if (calc == SC_DAYS_WO_DATA) {
|
|
|
|
value = QString::number((1+start.daysTo(end)) - p_profile->countDays(type, start, end ));
|
|
|
|
} else if (calc == SC_DAYS_W_DATA) {
|
|
|
|
value = QString::number(p_profile->countDays(type, start, end));
|
|
|
|
} else if (calc == SC_SELECTED_DAYS) {
|
|
|
|
value = QString::number(1+start.daysTo(end));
|
|
|
|
} else if (calc == SC_COMPLIANCE_DAYS) {
|
|
|
|
int value1 = p_profile->countCompliantDays(type, start, end );
|
|
|
|
value = QString("%1 ").arg(value1);
|
|
|
|
//value += QString(" / %1").arg(double(100*value1)/double(1+start.daysTo(end)));
|
2014-05-06 09:11:31 +00:00
|
|
|
} 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;
|
|
|
|
}
|
2023-07-08 15:13:35 +00:00
|
|
|
|
|
|
|
QDate lastdate;
|
|
|
|
QDate firstdate;
|
|
|
|
void Statistics::updateReportDate() {
|
|
|
|
if (p_profile) {
|
|
|
|
QDate last = p_profile->LastDay();
|
|
|
|
QDate first = p_profile->FirstDay();
|
|
|
|
if (last == lastdate && first == firstdate) return;
|
|
|
|
p_profile->general->setStatReportRangeStart(first);
|
|
|
|
p_profile->general->setStatReportRangeEnd(last);
|
|
|
|
p_profile->general->setStatReportDate(last);
|
|
|
|
lastdate = last;
|
|
|
|
firstdate = first;
|
|
|
|
}
|
|
|
|
}
|