OSCAR-code/oscar/statistics.h

246 lines
6.6 KiB
C
Raw Normal View History

2014-08-17 12:56:05 +00:00
/* Statistics Report Generator Header
*
* Copyright (c) 2019-2022 The OSCAR Team
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
*
* This file is subject to the terms and conditions of the GNU General Public
* License. See the file COPYING in the main directory of the source code
* for more details. */
#ifndef SUMMARY_H
#define SUMMARY_H
#include <QObject>
#include <QMainWindow>
#include <QPrinter>
#include <QPainter>
#include <QHash>
#include <QList>
#include "SleepLib/schema.h"
#include "SleepLib/machine.h"
class DisabledInfo
{
public:
QString display(int);
void update(QDate latest, QDate earliest) ;
private:
int totalDays ;
int daysNoData ;
int daysOutOfCompliance ;
int daysInCompliance ;
int numDisabledsessions ;
int numDaysWithDisabledsessions ;
int maxDurationOfaDisabledsession ;
int numDaysDisabledSessionChangedCompliance ;
int totalDurationOfDisabledSessions ;
void clear () {
totalDays = 0;
daysNoData = 0;
daysOutOfCompliance = 0;
daysInCompliance = 0;
numDisabledsessions = 0;
numDaysWithDisabledsessions = 0;
maxDurationOfaDisabledsession = 0;
numDaysDisabledSessionChangedCompliance = 0;
totalDurationOfDisabledSessions = 0;
};
};
2019-05-27 18:22:38 +00:00
//! \brief Type of calculation on one statistics row
enum StatCalcType {
SC_UNDEFINED=0, SC_COLUMNHEADERS, SC_HEADING, SC_SUBHEADING, SC_MEDIAN, SC_AVG, SC_WAVG, SC_90P, SC_MIN, SC_MAX, SC_CPH, SC_SPH, SC_AHI, SC_HOURS, SC_COMPLIANCE, SC_DAYS, SC_ABOVE, SC_BELOW , SC_WARNING , SC_WARNING2
};
2019-05-27 18:22:38 +00:00
/*! \struct StatisticsRow
\brief Describes a single row on the statistics page
*/
struct StatisticsRow {
StatisticsRow() { calc=SC_UNDEFINED; }
StatisticsRow(QString src, QString calc, QString type) {
this->src = src;
this->calc = lookupCalc(calc);
this->type = lookupType(type);
}
StatisticsRow(QString src, StatCalcType calc, MachineType type) {
this->src = src;
this->calc = calc;
this->type = type;
}
StatisticsRow(const StatisticsRow &copy) {
src=copy.src;
calc=copy.calc;
type=copy.type;
}
StatisticsRow& operator=(const StatisticsRow&) = default;
~StatisticsRow() {};
QString src;
StatCalcType calc;
MachineType type;
2019-05-27 18:22:38 +00:00
//! \brief Looks up calculation type for this row
StatCalcType lookupCalc(QString calc)
{
if (calc.compare("avg",Qt::CaseInsensitive)==0) {
return SC_AVG;
} else if (calc.compare("w-avg",Qt::CaseInsensitive)==0) {
return SC_WAVG;
} else if (calc.compare("median",Qt::CaseInsensitive)==0) {
return SC_MEDIAN;
} else if (calc.compare("90%",Qt::CaseInsensitive)==0) {
return SC_90P;
} else if (calc.compare("min", Qt::CaseInsensitive)==0) {
return SC_MIN;
} else if (calc.compare("max", Qt::CaseInsensitive)==0) {
return SC_MAX;
} else if (calc.compare("cph", Qt::CaseInsensitive)==0) {
return SC_CPH;
} else if (calc.compare("sph", Qt::CaseInsensitive)==0) {
return SC_SPH;
} else if (calc.compare("ahi", Qt::CaseInsensitive)==0) {
return SC_AHI;
} else if (calc.compare("hours", Qt::CaseInsensitive)==0) {
return SC_HOURS;
} else if (calc.compare("compliance", Qt::CaseInsensitive)==0) {
return SC_COMPLIANCE;
} else if (calc.compare("days", Qt::CaseInsensitive)==0) {
return SC_DAYS;
} else if (calc.compare("heading", Qt::CaseInsensitive)==0) {
return SC_HEADING;
} else if (calc.compare("subheading", Qt::CaseInsensitive)==0) {
return SC_SUBHEADING;
}
return SC_UNDEFINED;
}
//! \brief Look up device type
MachineType lookupType(QString type)
{
if (type.compare("cpap", Qt::CaseInsensitive)==0) {
return MT_CPAP;
} else if (type.compare("oximeter", Qt::CaseInsensitive)==0) {
return MT_OXIMETER;
} else if (type.compare("sleepstage", Qt::CaseInsensitive)==0) {
return MT_SLEEPSTAGE;
}
return MT_UNKNOWN;
}
ChannelID channel() {
return schema::channel[src].id();
}
QString value(QDate start, QDate end);
};
//! \class Prescription (device) setting
class RXItem {
public:
RXItem() {
machine = nullptr;
ahi = rdi = 0;
highlight = 0;
hours = 0;
}
RXItem(const RXItem & copy) {
start = copy.start;
end = copy.end;
days = copy.days;
s_count = copy.s_count;
s_sum = copy.s_sum;
ahi = copy.ahi;
rdi = copy.rdi;
hours = copy.hours;
machine = copy.machine;
relief = copy.relief;
mode = copy.mode;
pressure = copy.pressure;
dates = copy.dates;
highlight = copy.highlight;
}
2021-05-16 03:57:44 +00:00
RXItem& operator=(const RXItem&) = default;
inline quint64 count(ChannelID id) const {
QHash<ChannelID, quint64>::const_iterator it = s_count.find(id);
if (it == s_count.end()) return 0;
return it.value();
}
inline double sum(ChannelID id) const{
QHash<ChannelID, double>::const_iterator it = s_sum.find(id);
if (it == s_sum.end()) return 0;
return it.value();
}
QDate start;
QDate end;
int days;
QHash<ChannelID, quint64> s_count;
QHash<ChannelID, double> s_sum;
quint64 ahi;
quint64 rdi;
double hours;
Machine * machine;
QString relief;
QString mode;
QString pressure;
QMap<QDate, Day *> dates;
short highlight;
};
2014-09-30 08:40:12 +00:00
class Statistics : public QObject
{
Q_OBJECT
public:
explicit Statistics(QObject *parent = 0);
QString GenerateHTML();
2019-06-12 20:32:22 +00:00
QString UpdateRecordsBox();
2014-09-30 08:40:12 +00:00
static void printReport(QWidget *parent = nullptr);
void updateDisabledInfo();
protected:
2023-05-26 21:37:19 +00:00
void loadRXChanges();
void saveRXChanges();
void updateRXChanges();
QString getUserInfo();
QString getRDIorAHIText();
QString htmlNoData();
QString generateHeader(bool showheader);
QString generateFooter(bool showinfo=true);
QString GenerateMachineList();
QString GenerateRXChanges();
QString GenerateCPAPUsage();
// Using a map to maintain order
QList<StatisticsRow> rows;
QMap<StatCalcType, QString> calcnames;
QMap<MachineType, QString> machinenames;
QMap<QDate, RXItem> rxitems;
2014-09-30 08:40:12 +00:00
QList<QDate> record_best_ahi;
QList<QDate> record_worst_ahi;
DisabledInfo disabledInfo;
2014-09-30 08:40:12 +00:00
signals:
public slots:
};
#endif // SUMMARY_H