OSCAR-code/sleepyhead/Graphs/gSessionTimesChart.h

385 lines
10 KiB
C++

/* gSessionTimesChart Header
*
* Copyright (c) 2011-2014 Mark Watkins <jedimark@users.sourceforge.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 Linux
* distribution for more details. */
#ifndef GSESSIONTIMESCHART_H
#define GSESSIONTIMESCHART_H
#include "SleepLib/day.h"
#include "SleepLib/profiles.h"
#include "gGraphView.h"
///Represents the exception for taking the median of an empty list
class median_of_empty_list_exception:public std::exception{
virtual const char* what() const throw() {
return "Attempt to take the median of an empty list of numbers. "
"The median of an empty list is undefined.";
}
};
///Return the median of a sequence of numbers defined by the random
///access iterators begin and end. The sequence must not be empty
///(median is undefined for an empty set).
///
///The numbers must be convertible to double.
template<class RandAccessIter>
double median(RandAccessIter begin, RandAccessIter end)
throw(median_of_empty_list_exception){
if(begin == end){ throw median_of_empty_list_exception(); }
std::size_t size = end - begin;
std::size_t middleIdx = size/2;
RandAccessIter target = begin + middleIdx;
std::nth_element(begin, target, end);
if(size % 2 != 0){ //Odd number of elements
return *target;
}else{ //Even number of elements
double a = *target;
RandAccessIter targetNeighbor= target-1;
std::nth_element(begin, targetNeighbor, end);
return (a+*targetNeighbor)/2.0;
}
}
struct TimeSpan
{
public:
TimeSpan():begin(0), end(0) {}
TimeSpan(float b, float e) : begin(b), end(e) {}
TimeSpan(const TimeSpan & copy) {
begin = copy.begin;
end = copy.end;
}
~TimeSpan() {}
float begin;
float end;
};
struct SummaryCalcItem {
SummaryCalcItem() {
code = 0;
type = ST_CNT;
color = Qt::black;
}
SummaryCalcItem(const SummaryCalcItem & copy) {
code = copy.code;
type = copy.type;
color = copy.color;
sum = copy.sum;
divisor = copy.divisor;
min = copy.min;
max = copy.max;
}
SummaryCalcItem(ChannelID code, SummaryType type, QColor color)
:code(code), type(type), color(color) {}
ChannelID code;
SummaryType type;
QColor color;
double sum;
double divisor;
EventDataType min;
EventDataType max;
};
struct SummaryChartSlice {
SummaryChartSlice() {
calc = nullptr;
height = 0;
value = 0;
name = ST_CNT;
color = Qt::black;
}
SummaryChartSlice(const SummaryChartSlice & copy) {
calc = copy.calc;
value = copy.value;
height = copy.height;
name = copy.name;
color = copy.color;
}
SummaryChartSlice(SummaryCalcItem * calc, EventDataType value, EventDataType height, QString name, QColor color)
:calc(calc), value(value), height(height), name(name), color(color) {}
SummaryCalcItem * calc;
EventDataType value;
EventDataType height;
QString name;
QColor color;
};
class gSummaryChart : public Layer
{
public:
gSummaryChart(QString label, MachineType machtype);
gSummaryChart(ChannelID code, MachineType machtype);
virtual ~gSummaryChart();
//! \brief Renders the graph to the QPainter object
virtual void paint(QPainter &, gGraph &, const QRegion &);
//! \brief Called whenever data model changes underneath. Day object is not needed here, it's just here for Layer compatability.
virtual void SetDay(Day *day = nullptr);
//! \brief Returns true if no data was found for this day during SetDay
virtual bool isEmpty() { return m_empty; }
virtual void populate(Day *, int idx);
//! \brief Override to setup custom stuff before main loop
virtual void preCalc();
//! \brief Override to call stuff in main loop
virtual void customCalc(Day *, QList<SummaryChartSlice> &);
//! \brief Override to call stuff after draw is complete
virtual void afterDraw(QPainter &, gGraph &, QRect);
//! \brief Return any extra data to show beneath the date in the hover over tooltip
virtual QString tooltipData(Day *, int);
void addCalc(ChannelID code, SummaryType type, QColor color) {
calcitems.append(SummaryCalcItem(code, type, color));
}
void addCalc(ChannelID code, SummaryType type) {
calcitems.append(SummaryCalcItem(code, type, schema::channel[code].defaultColor()));
}
virtual Layer * Clone() {
gSummaryChart * sc = new gSummaryChart(m_label, m_machtype);
Layer::CloneInto(sc);
CloneInto(sc);
return sc;
}
void CloneInto(gSummaryChart * layer) {
layer->m_empty = m_empty;
layer->firstday = firstday;
layer->lastday = lastday;
layer->cache = cache;
layer->calcitems = calcitems;
layer->expected_slices = expected_slices;
layer->nousedays = nousedays;
layer->totaldays = totaldays;
layer->peak_value = peak_value;
}
protected:
//! \brief Key was pressed that effects this layer
virtual bool keyPressEvent(QKeyEvent *event, gGraph *graph);
//! \brief Mouse moved over this layers area (shows the hover-over tooltips here)
virtual bool mouseMoveEvent(QMouseEvent *event, gGraph *graph);
//! \brief Mouse Button was pressed over this area
virtual bool mousePressEvent(QMouseEvent *event, gGraph *graph);
//! \brief Mouse Button was released over this area. (jumps to daily view here)
virtual bool mouseReleaseEvent(QMouseEvent *event, gGraph *graph);
QString m_label;
MachineType m_machtype;
bool m_empty;
int hl_day;
int tz_offset;
float tz_hours;
QDate firstday;
QDate lastday;
static QMap<QDate, int> dayindex;
static QList<Day *> daylist;
QHash<int, QList<SummaryChartSlice> > cache;
QList<SummaryCalcItem> calcitems;
int expected_slices;
int nousedays;
int totaldays;
EventDataType peak_value;
EventDataType min_value;
int idx_start;
int idx_end;
};
/*! \class gSessionTimesChart
\brief Displays a summary of session times
*/
class gSessionTimesChart : public gSummaryChart
{
public:
gSessionTimesChart()
:gSummaryChart("SessionTimes", MT_CPAP) {
addCalc(NoChannel, ST_SESSIONS, QColor(64,128,255));
}
virtual ~gSessionTimesChart() {}
virtual void SetDay(Day * day = nullptr) {
gSummaryChart::SetDay(day);
split = p_profile->session->daySplitTime();
m_miny = 0;
m_maxy = 28;
}
virtual void preCalc() {
num_slices = 0;
num_days = 0;
total_length = 0;
session_data.clear();
session_data.reserve(idx_end-idx_start);
}
virtual void customCalc(Day *, QList<SummaryChartSlice> & slices) {
int size = slices.size();
num_slices += size;
for (int i=0; i<size; ++i) {
const SummaryChartSlice & slice = slices.at(i);
total_length += slice.height;
session_data.append(slice.height);
}
num_days++;
}
virtual void afterDraw(QPainter &, gGraph &, QRect);
//! \brief Renders the graph to the QPainter object
virtual void paint(QPainter &painter, gGraph &graph, const QRegion &region);
virtual Layer * Clone() {
gSessionTimesChart * sc = new gSessionTimesChart();
gSummaryChart::CloneInto(sc);
CloneInto(sc);
return sc;
}
void CloneInto(gSessionTimesChart * layer) {
layer->split = split;
}
QTime split;
int num_slices;
int num_days;
int total_slices;
double total_length;
QList<float> session_data;
};
class gUsageChart : public gSummaryChart
{
public:
gUsageChart()
:gSummaryChart("Usage", MT_CPAP) {
addCalc(NoChannel, ST_HOURS, QColor(64,128,255));
}
virtual ~gUsageChart() {}
virtual void preCalc();
virtual void customCalc(Day *, QList<SummaryChartSlice> &);
virtual void afterDraw(QPainter &, gGraph &, QRect);
virtual void populate(Day *day, int idx);
virtual QString tooltipData(Day * day, int);
virtual Layer * Clone() {
gUsageChart * sc = new gUsageChart();
gSummaryChart::CloneInto(sc);
CloneInto(sc);
return sc;
}
void CloneInto(gUsageChart * layer) {
layer->incompdays = incompdays;
layer->compliance_threshold = compliance_threshold;
}
private:
int incompdays;
EventDataType compliance_threshold;
double totalhours;
int totaldays;
QList<float> hour_data;
};
class gAHIChart : public gSummaryChart
{
public:
gAHIChart()
:gSummaryChart("AHIChart", MT_CPAP) {
addCalc(CPAP_ClearAirway, ST_CPH);
addCalc(CPAP_Obstructive, ST_CPH);
addCalc(CPAP_Apnea, ST_CPH);
addCalc(CPAP_Hypopnea, ST_CPH);
if (p_profile->general->calculateRDI())
addCalc(CPAP_RERA, ST_CPH);
}
virtual ~gAHIChart() {}
virtual void preCalc();
virtual void customCalc(Day *, QList<SummaryChartSlice> &);
virtual void afterDraw(QPainter &, gGraph &, QRect);
virtual void populate(Day *, int idx);
virtual QString tooltipData(Day * day, int);
virtual Layer * Clone() {
gAHIChart * sc = new gAHIChart();
gSummaryChart::CloneInto(sc);
CloneInto(sc);
return sc;
}
void CloneInto(gAHIChart * layer) {
layer->ahi_total = ahi_total;
layer->calc_cnt = calc_cnt;
}
double ahi_total;
double total_hours;
float max_ahi;
float min_ahi;
int calc_cnt;
QList<float> ahi_data;
};
class gPressureChart : public gSummaryChart
{
public:
gPressureChart();
virtual ~gPressureChart() {}
virtual Layer * Clone() {
gPressureChart * sc = new gPressureChart();
gSummaryChart::CloneInto(sc);
return sc;
}
// virtual void afterDraw(QPainter &, gGraph &, QRect);
virtual void populate(Day * day, int idx);
virtual QString tooltipData(Day * day, int idx) {
return day->getCPAPMode() + "\n" + day->getPressureSettings() + gSummaryChart::tooltipData(day, idx);
}
};
#endif // GSESSIONTIMESCHART_H