2018-06-14 08:58:06 +00:00
|
|
|
/* gSessionTimesChart Implementation
|
2014-05-31 21:25:07 +00:00
|
|
|
*
|
2020-04-27 18:12:16 +00:00
|
|
|
* Copyright (c) 2020 The Oscar Team
|
2018-03-28 07:10:52 +00:00
|
|
|
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
2014-05-31 21:25:07 +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. */
|
2014-05-31 21:25:07 +00:00
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
#include <QLabel>
|
|
|
|
#include <QDateTime>
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
#include "mainwindow.h"
|
2014-05-31 21:25:07 +00:00
|
|
|
#include "SleepLib/profiles.h"
|
2014-06-02 11:22:45 +00:00
|
|
|
#include "gSessionTimesChart.h"
|
2014-05-31 21:25:07 +00:00
|
|
|
|
|
|
|
#include "gYAxis.h"
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
extern MainWindow * mainwin;
|
2014-09-16 02:15:19 +00:00
|
|
|
|
|
|
|
short SummaryCalcItem::midcalc;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
gSummaryChart::gSummaryChart(QString label, MachineType machtype)
|
2014-05-31 21:25:07 +00:00
|
|
|
:Layer(NoChannel), m_label(label), m_machtype(machtype)
|
|
|
|
{
|
2014-09-11 14:23:08 +00:00
|
|
|
m_layertype = LT_Overview;
|
2014-05-31 21:25:07 +00:00
|
|
|
QDateTime d1 = QDateTime::currentDateTime();
|
|
|
|
QDateTime d2 = d1;
|
|
|
|
d1.setTimeSpec(Qt::UTC); // CHECK: Does this deal with DST?
|
|
|
|
tz_offset = d2.secsTo(d1);
|
|
|
|
tz_hours = tz_offset / 3600.0;
|
2014-09-11 14:23:08 +00:00
|
|
|
expected_slices = 5;
|
2014-10-03 07:09:35 +00:00
|
|
|
|
|
|
|
idx_end = 0;
|
|
|
|
idx_start = 0;
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
gSummaryChart::gSummaryChart(ChannelID code, MachineType machtype)
|
|
|
|
:Layer(code), m_machtype(machtype)
|
2014-05-31 21:25:07 +00:00
|
|
|
{
|
2014-09-11 14:23:08 +00:00
|
|
|
m_layertype = LT_Overview;
|
|
|
|
QDateTime d1 = QDateTime::currentDateTime();
|
|
|
|
QDateTime d2 = d1;
|
|
|
|
d1.setTimeSpec(Qt::UTC); // CHECK: Does this deal with DST?
|
|
|
|
tz_offset = d2.secsTo(d1);
|
|
|
|
tz_hours = tz_offset / 3600.0;
|
|
|
|
expected_slices = 5;
|
|
|
|
|
2014-10-02 07:56:57 +00:00
|
|
|
addCalc(code, ST_MIN, brighten(schema::channel[code].defaultColor() ,0.60f));
|
|
|
|
addCalc(code, ST_MID, brighten(schema::channel[code].defaultColor() ,1.20f));
|
|
|
|
addCalc(code, ST_90P, brighten(schema::channel[code].defaultColor() ,1.70f));
|
|
|
|
addCalc(code, ST_MAX, brighten(schema::channel[code].defaultColor() ,2.30f));
|
2014-10-03 07:09:35 +00:00
|
|
|
|
|
|
|
idx_end = 0;
|
|
|
|
idx_start = 0;
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
gSummaryChart::~gSummaryChart()
|
2014-05-31 21:25:07 +00:00
|
|
|
{
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void gSummaryChart::SetDay(Day *unused_day)
|
|
|
|
{
|
|
|
|
cache.clear();
|
|
|
|
|
2014-05-31 21:25:07 +00:00
|
|
|
Q_UNUSED(unused_day)
|
|
|
|
Layer::SetDay(nullptr);
|
|
|
|
|
2014-09-04 02:17:59 +00:00
|
|
|
firstday = p_profile->FirstDay(m_machtype);
|
|
|
|
lastday = p_profile->LastDay(m_machtype);
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
dayindex.clear();
|
|
|
|
daylist.clear();
|
2014-09-17 02:34:50 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
if (!firstday.isValid() || !lastday.isValid()) return;
|
|
|
|
// daylist.reserve(firstday.daysTo(lastday)+1);
|
2014-09-04 02:17:59 +00:00
|
|
|
QDate date = firstday;
|
2014-09-11 14:23:08 +00:00
|
|
|
int idx = 0;
|
2014-09-04 02:17:59 +00:00
|
|
|
do {
|
2018-05-05 21:58:11 +00:00
|
|
|
auto di = p_profile->daylist.find(date);
|
2014-09-11 14:23:08 +00:00
|
|
|
Day * day = nullptr;
|
|
|
|
if (di != p_profile->daylist.end()) {
|
|
|
|
day = di.value();
|
2014-09-04 02:17:59 +00:00
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
daylist.append(day);
|
|
|
|
dayindex[date] = idx;
|
|
|
|
idx++;
|
|
|
|
date = date.addDays(1);
|
|
|
|
} while (date <= lastday);
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2016-02-28 14:04:26 +00:00
|
|
|
m_minx = QDateTime(firstday, QTime(0,0,0), Qt::UTC).toMSecsSinceEpoch();
|
|
|
|
m_maxx = QDateTime(lastday, QTime(23,59,59), Qt::UTC).toMSecsSinceEpoch();
|
2014-09-04 02:17:59 +00:00
|
|
|
m_miny = 0;
|
2014-09-11 14:23:08 +00:00
|
|
|
m_maxy = 20;
|
|
|
|
|
2014-09-04 02:17:59 +00:00
|
|
|
m_empty = false;
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-04 02:17:59 +00:00
|
|
|
}
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-17 02:34:50 +00:00
|
|
|
|
|
|
|
//QMap<QDate, int> gSummaryChart::dayindex;
|
|
|
|
//QList<Day *> gSummaryChart::daylist;
|
|
|
|
|
2020-04-27 18:46:49 +00:00
|
|
|
int gSummaryChart::addCalc(ChannelID code, SummaryType type, QColor color)
|
|
|
|
{
|
|
|
|
calcitems.append(SummaryCalcItem(code, type, color));
|
|
|
|
return calcitems.size() - 1; // return the index of the newly appended calc
|
|
|
|
}
|
|
|
|
|
|
|
|
int gSummaryChart::addCalc(ChannelID code, SummaryType type)
|
|
|
|
{
|
|
|
|
return addCalc(code, type, schema::channel[code].defaultColor());
|
|
|
|
}
|
|
|
|
|
2014-09-17 02:34:50 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
bool gSummaryChart::keyPressEvent(QKeyEvent *event, gGraph *graph)
|
|
|
|
{
|
|
|
|
Q_UNUSED(event)
|
|
|
|
Q_UNUSED(graph)
|
|
|
|
return false;
|
|
|
|
}
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
bool gSummaryChart::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
|
|
|
|
{
|
|
|
|
Q_UNUSED(event)
|
|
|
|
Q_UNUSED(graph)
|
|
|
|
return false;
|
|
|
|
}
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
bool gSummaryChart::mousePressEvent(QMouseEvent *event, gGraph *graph)
|
|
|
|
{
|
|
|
|
Q_UNUSED(event)
|
|
|
|
Q_UNUSED(graph)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool gSummaryChart::mouseReleaseEvent(QMouseEvent *event, gGraph *graph)
|
|
|
|
{
|
|
|
|
if (!(event->modifiers() & Qt::ShiftModifier)) return false;
|
|
|
|
|
|
|
|
float x = event->x() - m_rect.left();
|
|
|
|
float y = event->y() - m_rect.top();
|
|
|
|
qDebug() << x << y;
|
|
|
|
|
|
|
|
EventDataType miny;
|
|
|
|
EventDataType maxy;
|
|
|
|
|
|
|
|
graph->roundY(miny, maxy);
|
|
|
|
|
2015-08-11 20:01:24 +00:00
|
|
|
QDate date = QDateTime::fromMSecsSinceEpoch(m_minx, Qt::UTC).date();
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
int days = ceil(double(m_maxx - m_minx) / 86400000.0);
|
|
|
|
|
|
|
|
float barw = float(m_rect.width()) / float(days);
|
|
|
|
|
|
|
|
float idx = x/barw;
|
|
|
|
|
|
|
|
date = date.addDays(idx);
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto it = dayindex.find(date);
|
2014-09-11 14:23:08 +00:00
|
|
|
if (it != dayindex.end()) {
|
|
|
|
Day * day = daylist.at(it.value());
|
|
|
|
if (day) {
|
|
|
|
mainwin->getDaily()->LoadDate(date);
|
|
|
|
mainwin->JumpDaily();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-09-13 11:34:18 +00:00
|
|
|
void gSummaryChart::preCalc()
|
|
|
|
{
|
2014-09-16 02:15:19 +00:00
|
|
|
midcalc = p_profile->general->prefCalcMiddle();
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (auto & calc : calcitems) {
|
2018-04-25 13:00:25 +00:00
|
|
|
calc.reset(idx_end - idx_start, midcalc);
|
2014-09-13 11:34:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-29 14:41:31 +00:00
|
|
|
void gSummaryChart::customCalc(Day *day, QVector<SummaryChartSlice> & slices)
|
2014-09-13 11:34:18 +00:00
|
|
|
{
|
2014-09-29 14:41:31 +00:00
|
|
|
int size = slices.size();
|
|
|
|
if (size != calcitems.size()) {
|
2014-09-13 11:34:18 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
float hour = day->hours(m_machtype);
|
2018-05-05 21:58:11 +00:00
|
|
|
|
2014-09-29 14:41:31 +00:00
|
|
|
for (int i=0; i < size; ++i) {
|
2014-09-13 11:34:18 +00:00
|
|
|
const SummaryChartSlice & slice = slices.at(i);
|
|
|
|
SummaryCalcItem & calc = calcitems[i];
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
calc.update(slice.value, hour);
|
|
|
|
}
|
2014-09-13 11:34:18 +00:00
|
|
|
}
|
|
|
|
|
2018-06-10 07:53:10 +00:00
|
|
|
void gSummaryChart::afterDraw(QPainter &painter, gGraph &graph, QRectF rect)
|
2014-09-13 11:34:18 +00:00
|
|
|
{
|
|
|
|
if (totaldays == nousedays) return;
|
|
|
|
|
|
|
|
if (calcitems.size() == 0) return;
|
|
|
|
|
|
|
|
QStringList strlist;
|
|
|
|
QString txt;
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
int midcalc = p_profile->general->prefCalcMiddle();
|
2014-09-13 11:34:18 +00:00
|
|
|
QString midstr;
|
2014-09-14 15:29:07 +00:00
|
|
|
if (midcalc == 0) {
|
2014-09-13 11:34:18 +00:00
|
|
|
midstr = QObject::tr("Med.");
|
2014-09-14 15:29:07 +00:00
|
|
|
} else if (midcalc == 1) {
|
2014-09-13 11:34:18 +00:00
|
|
|
midstr = QObject::tr("W-Avg");
|
2014-09-14 15:29:07 +00:00
|
|
|
} else {
|
|
|
|
midstr = QObject::tr("Avg");
|
2014-09-13 11:34:18 +00:00
|
|
|
}
|
2014-09-14 15:29:07 +00:00
|
|
|
|
|
|
|
|
2014-09-13 11:34:18 +00:00
|
|
|
float perc = p_profile->general->prefCalcPercentile();
|
2018-06-14 08:58:06 +00:00
|
|
|
QString percstr = QString("%1%").arg(perc, 0, 'f',0);
|
2014-09-13 11:34:18 +00:00
|
|
|
|
|
|
|
schema::Channel & chan = schema::channel[calcitems.at(0).code];
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (auto & calc : calcitems) {
|
|
|
|
|
2014-09-13 11:34:18 +00:00
|
|
|
if (calcitems.size() == 1) {
|
|
|
|
float val = calc.min;
|
|
|
|
if (val < 99998)
|
|
|
|
strlist.append(QObject::tr("Min: %1").arg(val,0,'f',2));
|
|
|
|
}
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
float mid = 0;
|
|
|
|
switch (midcalc) {
|
|
|
|
case 0:
|
|
|
|
if (calc.median_data.size() > 0) {
|
|
|
|
mid = median(calc.median_data.begin(), calc.median_data.end());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
mid = calc.wavg_sum / calc.divisor;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
mid = calc.avg_sum / calc.cnt;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-09-13 11:34:18 +00:00
|
|
|
float val = 0;
|
|
|
|
switch (calc.type) {
|
|
|
|
case ST_CPH:
|
2014-09-14 15:29:07 +00:00
|
|
|
val = mid;
|
|
|
|
txt = midstr+": ";
|
2014-09-13 11:34:18 +00:00
|
|
|
break;
|
|
|
|
case ST_SPH:
|
2014-09-14 15:29:07 +00:00
|
|
|
val = mid;
|
|
|
|
txt = midstr+": ";
|
2014-09-13 11:34:18 +00:00
|
|
|
break;
|
|
|
|
case ST_MIN:
|
|
|
|
val = calc.min;
|
|
|
|
if (val >= 99998) continue;
|
|
|
|
txt = QObject::tr("Min: ");
|
|
|
|
break;
|
|
|
|
case ST_MAX:
|
|
|
|
val = calc.max;
|
|
|
|
if (val <= -99998) continue;
|
|
|
|
txt = QObject::tr("Max: ");
|
|
|
|
break;
|
|
|
|
case ST_SETMIN:
|
|
|
|
val = calc.min;
|
|
|
|
if (val >= 99998) continue;
|
|
|
|
txt = QObject::tr("Min: ");
|
|
|
|
break;
|
|
|
|
case ST_SETMAX:
|
|
|
|
val = calc.max;
|
|
|
|
if (val <= -99998) continue;
|
|
|
|
txt = QObject::tr("Max: ");
|
|
|
|
break;
|
|
|
|
case ST_MID:
|
2014-09-14 15:29:07 +00:00
|
|
|
val = mid;
|
2014-09-13 11:34:18 +00:00
|
|
|
txt = QObject::tr("%1: ").arg(midstr);
|
|
|
|
break;
|
|
|
|
case ST_90P:
|
2014-09-14 15:29:07 +00:00
|
|
|
val = mid;
|
2014-09-13 11:34:18 +00:00
|
|
|
txt = QObject::tr("%1: ").arg(percstr);
|
|
|
|
break;
|
|
|
|
default:
|
2014-09-14 15:29:07 +00:00
|
|
|
val = mid;
|
2014-09-13 11:34:18 +00:00
|
|
|
txt = QObject::tr("???: ");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
strlist.append(QString("%1%2").arg(txt).arg(val,0,'f',2));
|
|
|
|
if (calcitems.size() == 1) {
|
|
|
|
val = calc.max;
|
|
|
|
if (val > -99998)
|
|
|
|
strlist.append(QObject::tr("Max: %1").arg(val,0,'f',2));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QString str;
|
|
|
|
if (totaldays > 1) {
|
|
|
|
str = QObject::tr("%1 (%2 days): ").arg(chan.fullname()).arg(totaldays);
|
|
|
|
} else {
|
|
|
|
str = QObject::tr("%1 (%2 day): ").arg(chan.fullname()).arg(totaldays);
|
|
|
|
}
|
|
|
|
str += " "+strlist.join(", ");
|
|
|
|
|
|
|
|
QRectF rec(rect.left(), rect.top(), 0,0);
|
|
|
|
painter.setFont(*defaultfont);
|
|
|
|
rec = painter.boundingRect(rec, Qt::AlignTop, str);
|
|
|
|
rec.moveBottom(rect.top()-3*graph.printScaleY());
|
|
|
|
painter.drawText(rec, Qt::AlignTop, str);
|
|
|
|
|
|
|
|
// graph.renderText(str, rect.left(), rect.top()-5*graph.printScaleY(), 0);
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
QString gSummaryChart::tooltipData(Day *, int idx)
|
|
|
|
{
|
|
|
|
QString txt;
|
2018-05-05 21:58:11 +00:00
|
|
|
const auto & slices = cache[idx];
|
2020-08-03 03:24:56 +00:00
|
|
|
int i = slices.size();
|
|
|
|
while (i > 0) {
|
|
|
|
i--;
|
|
|
|
txt += QString("\n%1: %2").arg(slices[i].name).arg(float(slices[i].value), 0, 'f', 2);
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
return txt;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gSummaryChart::populate(Day * day, int idx)
|
|
|
|
{
|
2014-09-17 02:34:50 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
bool good = false;
|
2018-05-05 21:58:11 +00:00
|
|
|
for (const auto & item : calcitems) {
|
2014-09-11 14:23:08 +00:00
|
|
|
if (day->hasData(item.code, item.type)) {
|
|
|
|
good = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!good) return;
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto & slices = cache[idx];
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
float hours = day->hours(m_machtype);
|
|
|
|
float base = 0;
|
2018-05-05 21:58:11 +00:00
|
|
|
|
|
|
|
for (auto & item : calcitems) {
|
2014-09-11 14:23:08 +00:00
|
|
|
ChannelID code = item.code;
|
|
|
|
schema::Channel & chan = schema::channel[code];
|
|
|
|
float value = 0;
|
|
|
|
QString name;
|
|
|
|
QColor color;
|
|
|
|
switch (item.type) {
|
|
|
|
case ST_CPH:
|
|
|
|
value = day->count(code) / hours;
|
|
|
|
name = chan.label();
|
2014-09-13 11:34:18 +00:00
|
|
|
color = item.color;
|
|
|
|
slices.append(SummaryChartSlice(&item, value, value, name, color));
|
2014-09-11 14:23:08 +00:00
|
|
|
break;
|
|
|
|
case ST_SPH:
|
|
|
|
value = (100.0 / hours) * (day->sum(code) / 3600.0);
|
|
|
|
name = QObject::tr("% in %1").arg(chan.label());
|
2014-09-13 11:34:18 +00:00
|
|
|
color = item.color;
|
|
|
|
slices.append(SummaryChartSlice(&item, value, value, name, color));
|
2014-09-11 14:23:08 +00:00
|
|
|
break;
|
|
|
|
case ST_HOURS:
|
|
|
|
value = hours;
|
|
|
|
name = QObject::tr("Hours");
|
|
|
|
color = COLOR_LightBlue;
|
2018-03-28 06:22:42 +00:00
|
|
|
slices.append(SummaryChartSlice(&item, value, hours, name, color));
|
2014-09-11 14:23:08 +00:00
|
|
|
break;
|
|
|
|
case ST_MIN:
|
|
|
|
value = day->Min(code);
|
|
|
|
name = QObject::tr("Min %1").arg(chan.label());
|
2014-09-13 11:34:18 +00:00
|
|
|
color = item.color;
|
|
|
|
slices.append(SummaryChartSlice(&item, value, value - base, name, color));
|
2014-09-11 14:23:08 +00:00
|
|
|
base = value;
|
|
|
|
break;
|
|
|
|
case ST_MID:
|
|
|
|
value = day->calcMiddle(code);
|
|
|
|
name = day->calcMiddleLabel(code);
|
2014-09-13 11:34:18 +00:00
|
|
|
color = item.color;
|
|
|
|
slices.append(SummaryChartSlice(&item, value, value - base, name, color));
|
2014-09-11 14:23:08 +00:00
|
|
|
base = value;
|
|
|
|
break;
|
|
|
|
case ST_90P:
|
|
|
|
value = day->calcPercentile(code);
|
|
|
|
name = day->calcPercentileLabel(code);
|
2014-09-13 11:34:18 +00:00
|
|
|
color = item.color;
|
|
|
|
slices.append(SummaryChartSlice(&item, value, value - base, name, color));
|
2014-09-11 14:23:08 +00:00
|
|
|
base = value;
|
|
|
|
break;
|
|
|
|
case ST_MAX:
|
|
|
|
value = day->calcMax(code);
|
|
|
|
name = day->calcMaxLabel(code);
|
2014-09-13 11:34:18 +00:00
|
|
|
color = item.color;
|
|
|
|
slices.append(SummaryChartSlice(&item, value, value - base, name, color));
|
2014-09-11 14:23:08 +00:00
|
|
|
base = value;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gSummaryChart::paint(QPainter &painter, gGraph &graph, const QRegion ®ion)
|
2014-09-04 02:17:59 +00:00
|
|
|
{
|
2018-06-10 07:53:10 +00:00
|
|
|
QRectF rect = region.boundingRect();
|
|
|
|
|
|
|
|
rect.translate(0.0f, 0.001f);
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-04 02:17:59 +00:00
|
|
|
painter.setPen(QColor(Qt::black));
|
|
|
|
painter.drawRect(rect);
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
rect.moveBottom(rect.bottom()+1);
|
|
|
|
|
2014-09-04 02:17:59 +00:00
|
|
|
m_minx = graph.min_x;
|
|
|
|
m_maxx = graph.max_x;
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2015-08-11 20:01:24 +00:00
|
|
|
QDateTime date2 = QDateTime::fromMSecsSinceEpoch(m_minx, Qt::UTC);
|
|
|
|
QDateTime enddate2 = QDateTime::fromMSecsSinceEpoch(m_maxx, Qt::UTC);
|
2014-09-04 02:17:59 +00:00
|
|
|
|
|
|
|
QDate date = date2.date();
|
|
|
|
QDate enddate = enddate2.date();
|
|
|
|
|
|
|
|
int days = ceil(double(m_maxx - m_minx) / 86400000.0);
|
|
|
|
|
2014-09-18 17:58:00 +00:00
|
|
|
//float lasty1 = rect.bottom();
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto it = dayindex.find(date);
|
2016-02-28 13:58:28 +00:00
|
|
|
idx_start = 0;
|
|
|
|
if (it == dayindex.end()) {
|
|
|
|
it = dayindex.begin();
|
|
|
|
} else {
|
2014-09-13 11:34:18 +00:00
|
|
|
idx_start = it.value();
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
2016-02-28 13:58:28 +00:00
|
|
|
|
2014-09-13 11:34:18 +00:00
|
|
|
int idx = idx_start;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2019-08-23 02:59:51 +00:00
|
|
|
// Determine how many days after the first day of the chart that this data is to begin
|
2019-08-22 09:33:18 +00:00
|
|
|
int numDaysOffset = 0;
|
2019-08-23 02:59:51 +00:00
|
|
|
if (firstday > date) { // date = beginning date of chart; firstday = beginning date of data
|
2019-08-22 09:33:18 +00:00
|
|
|
numDaysOffset = date.daysTo(firstday);
|
|
|
|
}
|
|
|
|
|
2019-08-23 02:59:51 +00:00
|
|
|
// Determine how many days before the last day of the chart that this data is to end
|
|
|
|
int numDaysAfter = 0;
|
|
|
|
if (enddate > lastday) { // enddate = last date of chart; lastday = last date of data
|
|
|
|
numDaysAfter = lastday.daysTo(enddate);
|
|
|
|
}
|
|
|
|
if (numDaysAfter > days) // Nothing to do if this data is off the left edge of the chart
|
|
|
|
return;
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto ite = dayindex.find(enddate);
|
2014-09-13 11:34:18 +00:00
|
|
|
idx_end = daylist.size()-1;
|
2014-09-11 14:23:08 +00:00
|
|
|
if (ite != dayindex.end()) {
|
|
|
|
idx_end = ite.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
QPoint mouse = graph.graphView()->currentMousePos();
|
|
|
|
|
|
|
|
nousedays = 0;
|
|
|
|
totaldays = 0;
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
QRectF hl_rect;
|
2014-09-11 14:23:08 +00:00
|
|
|
QDate hl_date;
|
|
|
|
Day * hl_day = nullptr;
|
|
|
|
int hl_idx = -1;
|
|
|
|
bool hl = false;
|
|
|
|
|
2016-02-28 13:58:28 +00:00
|
|
|
if ((daylist.size() == 0) || (it == dayindex.end()))
|
|
|
|
return;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2014-10-02 11:22:30 +00:00
|
|
|
//Day * lastday = nullptr;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
// int dc = 0;
|
|
|
|
// for (int i=idx; i<=idx_end; ++i) {
|
|
|
|
// Day * day = daylist.at(i);
|
|
|
|
// if (day || lastday) {
|
|
|
|
// dc++;
|
|
|
|
// }
|
|
|
|
// lastday = day;
|
|
|
|
// }
|
|
|
|
// days = dc;
|
|
|
|
// lastday = nullptr;
|
2014-09-04 02:17:59 +00:00
|
|
|
float barw = float(rect.width()) / float(days);
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
QString hl2_text = "";
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
QVector<QRectF> outlines;
|
|
|
|
int size = idx_end - idx;
|
|
|
|
outlines.reserve(size * expected_slices);
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
// Virtual call to setup any custom graph stuff
|
|
|
|
preCalc();
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-18 10:53:59 +00:00
|
|
|
float lastx1 = rect.left();
|
2019-08-22 09:33:18 +00:00
|
|
|
lastx1 += numDaysOffset * barw;
|
2014-09-18 10:53:59 +00:00
|
|
|
float right_edge = (rect.left()+rect.width()+1);
|
|
|
|
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
/// Calculate Graph Peaks
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
peak_value = 0;
|
2014-09-18 10:53:59 +00:00
|
|
|
for (int i=idx; i <= idx_end; ++i, lastx1 += barw) {
|
2014-09-11 14:23:08 +00:00
|
|
|
Day * day = daylist.at(i);
|
|
|
|
|
2014-09-18 10:53:59 +00:00
|
|
|
if ((lastx1 + barw) > right_edge)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (!day) {
|
2014-09-11 14:23:08 +00:00
|
|
|
continue;
|
2014-09-18 10:53:59 +00:00
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
day->OpenSummary();
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto cit = cache.find(i);
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
if (cit == cache.end()) {
|
|
|
|
populate(day, i);
|
|
|
|
cit = cache.find(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cit != cache.end()) {
|
|
|
|
float base = 0, val;
|
2018-05-05 21:58:11 +00:00
|
|
|
for (const auto & slice : cit.value()) {
|
2014-09-11 14:23:08 +00:00
|
|
|
val = slice.height;
|
|
|
|
base += val;
|
|
|
|
}
|
|
|
|
peak_value = qMax(peak_value, base);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_miny = 0;
|
|
|
|
m_maxy = ceil(peak_value);
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
/// Y-Axis scaling
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
EventDataType miny;
|
|
|
|
EventDataType maxy;
|
|
|
|
|
|
|
|
graph.roundY(miny, maxy);
|
|
|
|
float ymult = float(rect.height()) / (maxy-miny);
|
|
|
|
|
2014-09-18 10:53:59 +00:00
|
|
|
lastx1 = rect.left();
|
2019-08-22 09:33:18 +00:00
|
|
|
lastx1 += numDaysOffset * barw;
|
2014-09-18 10:53:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
/// Main drawing loop
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
2014-09-04 02:17:59 +00:00
|
|
|
do {
|
2014-09-11 14:23:08 +00:00
|
|
|
Day * day = daylist.at(idx);
|
|
|
|
|
2014-09-18 10:53:59 +00:00
|
|
|
if ((lastx1 + barw) > right_edge)
|
2014-09-11 14:23:08 +00:00
|
|
|
break;
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
totaldays++;
|
|
|
|
|
2014-09-17 02:34:50 +00:00
|
|
|
if (!day) {// || !day->hasMachine(m_machtype)) {
|
2014-09-18 10:53:59 +00:00
|
|
|
// lasty1 = rect.bottom();
|
2014-09-04 02:17:59 +00:00
|
|
|
lastx1 += barw;
|
2014-09-11 14:23:08 +00:00
|
|
|
it++;
|
|
|
|
nousedays++;
|
2014-10-02 11:22:30 +00:00
|
|
|
//lastday = day;
|
2014-09-04 02:17:59 +00:00
|
|
|
continue;
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-10-02 11:22:30 +00:00
|
|
|
//lastday = day;
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
float x1 = lastx1 + barw;
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
day->OpenSummary();
|
|
|
|
QRectF hl2_rect;
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
bool hlday = false;
|
|
|
|
QRectF rec2(lastx1, rect.top(), barw, rect.height());
|
2014-09-04 02:17:59 +00:00
|
|
|
if (rec2.contains(mouse)) {
|
2014-09-11 14:23:08 +00:00
|
|
|
hl_rect = rec2;
|
|
|
|
hl_day = day;
|
|
|
|
hl_date = it.key();
|
|
|
|
hl_idx = idx;
|
|
|
|
|
2014-09-04 02:17:59 +00:00
|
|
|
hl = true;
|
2014-09-11 14:23:08 +00:00
|
|
|
hlday = true;
|
2014-09-04 02:17:59 +00:00
|
|
|
}
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto cit = cache.find(idx);
|
2014-06-02 11:22:45 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
if (cit == cache.end()) {
|
|
|
|
populate(day, idx);
|
|
|
|
cit = cache.find(idx);
|
|
|
|
}
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
float lastval = 0, val, y1,y2;
|
|
|
|
if (cit != cache.end()) {
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// Draw pressure settings
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
2014-09-29 14:41:31 +00:00
|
|
|
QVector<SummaryChartSlice> & list = cit.value();
|
2014-09-11 14:23:08 +00:00
|
|
|
customCalc(day, list);
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-16 02:15:19 +00:00
|
|
|
QLinearGradient gradient(lastx1, 0, lastx1 + barw, 0); //rect.bottom(), barw, rect.bottom());
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (const auto & slice : list) {
|
2014-09-11 14:23:08 +00:00
|
|
|
val = slice.height;
|
|
|
|
y1 = ((lastval-miny) * ymult);
|
|
|
|
y2 = (val * ymult);
|
|
|
|
QColor color = slice.color;
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-16 02:15:19 +00:00
|
|
|
QRectF rec = QRectF(lastx1, rect.bottom() - y1, barw, -y2).intersected(rect);
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
if (hlday) {
|
|
|
|
if (rec.contains(mouse.x(), mouse.y())) {
|
|
|
|
color = Qt::yellow;
|
|
|
|
hl2_rect = rec;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-16 02:15:19 +00:00
|
|
|
if (barw <= 3) {
|
|
|
|
painter.fillRect(rec, QBrush(color));
|
|
|
|
} else if (barw > 8) {
|
2014-09-11 14:23:08 +00:00
|
|
|
gradient.setColorAt(0,color);
|
2014-09-16 02:15:19 +00:00
|
|
|
gradient.setColorAt(1,brighten(color, 2.0));
|
2014-09-11 14:23:08 +00:00
|
|
|
painter.fillRect(rec, QBrush(gradient));
|
2014-09-16 02:15:19 +00:00
|
|
|
// painter.fillRect(rec, slice.brush);
|
2014-09-11 14:23:08 +00:00
|
|
|
outlines.append(rec);
|
2014-09-16 02:15:19 +00:00
|
|
|
} else {
|
2018-04-25 13:00:25 +00:00
|
|
|
painter.fillRect(rec, brighten(color, 1.25));
|
2014-09-11 14:23:08 +00:00
|
|
|
outlines.append(rec);
|
|
|
|
}
|
|
|
|
|
|
|
|
lastval += val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lastx1 = x1;
|
|
|
|
it++;
|
|
|
|
} while (++idx <= idx_end);
|
|
|
|
painter.setPen(QPen(Qt::black,1));
|
|
|
|
painter.drawRects(outlines);
|
|
|
|
|
|
|
|
if (hl) {
|
|
|
|
QColor col2(255,0,0,64);
|
|
|
|
painter.fillRect(hl_rect, QBrush(col2));
|
|
|
|
|
2014-09-24 01:42:14 +00:00
|
|
|
QString txt = hl_date.toString(Qt::SystemLocaleShortDate)+" ";
|
2014-09-11 14:23:08 +00:00
|
|
|
if (hl_day) {
|
|
|
|
// grab extra tooltip data
|
|
|
|
txt += tooltipData(hl_day, hl_idx);
|
|
|
|
if (!hl2_text.isEmpty()) {
|
|
|
|
QColor col = Qt::yellow;
|
|
|
|
col.setAlpha(255);
|
|
|
|
// painter.fillRect(hl2_rect, QBrush(col));
|
|
|
|
txt += hl2_text;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
graph.ToolTip(txt, mouse.x()-15, mouse.y()+5, TT_AlignRight);
|
|
|
|
}
|
2014-09-16 02:15:19 +00:00
|
|
|
try {
|
|
|
|
afterDraw(painter, graph, rect);
|
|
|
|
} catch(...) {
|
|
|
|
qDebug() << "Bad median call in" << m_label;
|
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
// This could be turning off graphs prematurely..
|
|
|
|
if (cache.size() == 0) {
|
|
|
|
|
|
|
|
m_empty = true;
|
|
|
|
graph.graphView()->updateScale();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
QString gUsageChart::tooltipData(Day * day, int)
|
|
|
|
{
|
|
|
|
return QObject::tr("\nHours: %1").arg(day->hours(m_machtype), 0, 'f', 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void gUsageChart::populate(Day *day, int idx)
|
|
|
|
{
|
2014-09-29 14:41:31 +00:00
|
|
|
QVector<SummaryChartSlice> & slices = cache[idx];
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2020-03-29 23:21:34 +00:00
|
|
|
float hours = day->hours(m_machtype);
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2014-09-13 11:34:18 +00:00
|
|
|
QColor cpapcolor = day->summaryOnly() ? QColor(128,128,128) : calcitems[0].color;
|
2014-09-11 14:23:08 +00:00
|
|
|
bool haveoxi = day->hasMachine(MT_OXIMETER);
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
QColor goodcolor = haveoxi ? QColor(128,255,196) : cpapcolor;
|
|
|
|
|
|
|
|
QColor color = (hours < compliance_threshold) ? QColor(255,64,64) : goodcolor;
|
2014-09-13 11:34:18 +00:00
|
|
|
slices.append(SummaryChartSlice(&calcitems[0], hours, hours, QObject::tr("Hours"), color));
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void gUsageChart::preCalc()
|
|
|
|
{
|
2014-09-16 02:15:19 +00:00
|
|
|
midcalc = p_profile->general->prefCalcMiddle();
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
compliance_threshold = p_profile->cpap->complianceHours();
|
|
|
|
incompdays = 0;
|
2014-09-13 11:34:18 +00:00
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
SummaryCalcItem & calc = calcitems[0];
|
2018-04-25 13:00:25 +00:00
|
|
|
calc.reset(idx_end - idx_start, midcalc);
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
|
2014-09-29 14:41:31 +00:00
|
|
|
void gUsageChart::customCalc(Day *, QVector<SummaryChartSlice> &list)
|
2014-09-11 14:23:08 +00:00
|
|
|
{
|
2014-09-14 15:29:07 +00:00
|
|
|
if (list.size() == 0) {
|
|
|
|
incompdays++;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
SummaryChartSlice & slice = list[0];
|
2014-09-14 15:29:07 +00:00
|
|
|
SummaryCalcItem & calc = calcitems[0];
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
if (slice.value < compliance_threshold) incompdays++;
|
2014-09-14 15:29:07 +00:00
|
|
|
|
|
|
|
calc.update(slice.value, 1);
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
|
2018-06-10 07:53:10 +00:00
|
|
|
void gUsageChart::afterDraw(QPainter &, gGraph &graph, QRectF rect)
|
2014-09-11 14:23:08 +00:00
|
|
|
{
|
2014-09-13 11:34:18 +00:00
|
|
|
if (totaldays == nousedays) return;
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
if (totaldays > 1) {
|
|
|
|
float comp = 100.0 - ((float(incompdays + nousedays) / float(totaldays)) * 100.0);
|
2014-09-13 11:34:18 +00:00
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
int midcalc = p_profile->general->prefCalcMiddle();
|
|
|
|
float mid = 0;
|
|
|
|
SummaryCalcItem & calc = calcitems[0];
|
|
|
|
switch (midcalc) {
|
|
|
|
case 0: // median
|
|
|
|
mid = median(calc.median_data.begin(), calc.median_data.end());
|
|
|
|
break;
|
|
|
|
case 1: // w-avg
|
|
|
|
mid = calc.wavg_sum / calc.divisor;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
mid = calc.avg_sum / calc.cnt;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString txt = QObject::tr("%1 low usage, %2 no usage, out of %3 days (%4% compliant.) Length: %5 / %6 / %7").
|
|
|
|
arg(incompdays).arg(nousedays).arg(totaldays).arg(comp,0,'f',1).arg(calc.min, 0, 'f', 2).arg(mid, 0, 'f', 2).arg(calc.max, 0, 'f', 2);;
|
2014-09-11 14:23:08 +00:00
|
|
|
graph.renderText(txt, rect.left(), rect.top()-5*graph.printScaleY(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
void gSessionTimesChart::preCalc() {
|
2014-09-16 02:15:19 +00:00
|
|
|
|
|
|
|
midcalc = p_profile->general->prefCalcMiddle();
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
num_slices = 0;
|
|
|
|
num_days = 0;
|
|
|
|
total_length = 0;
|
|
|
|
SummaryCalcItem & calc = calcitems[0];
|
|
|
|
|
2018-04-25 13:00:25 +00:00
|
|
|
calc.reset((idx_end - idx_start) * 6, midcalc);
|
2014-09-14 15:29:07 +00:00
|
|
|
|
|
|
|
SummaryCalcItem & calc1 = calcitems[1];
|
|
|
|
|
2018-04-25 13:00:25 +00:00
|
|
|
calc1.reset(idx_end - idx_start, midcalc);
|
2014-09-14 15:29:07 +00:00
|
|
|
|
|
|
|
SummaryCalcItem & calc2 = calcitems[2];
|
2018-04-25 13:00:25 +00:00
|
|
|
calc2.reset(idx_end - idx_start, midcalc);
|
2014-09-14 15:29:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-29 14:41:31 +00:00
|
|
|
void gSessionTimesChart::customCalc(Day *, QVector<SummaryChartSlice> & slices) {
|
2014-09-14 15:29:07 +00:00
|
|
|
int size = slices.size();
|
|
|
|
num_slices += size;
|
|
|
|
|
|
|
|
SummaryCalcItem & calc1 = calcitems[1];
|
|
|
|
calc1.update(slices.size(), 1);
|
|
|
|
|
|
|
|
EventDataType max = 0;
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (auto & slice : slices) {
|
|
|
|
slice.calc->update(slice.height, slice.height);
|
2014-09-14 15:29:07 +00:00
|
|
|
|
|
|
|
max = qMax(slice.height, max);
|
|
|
|
}
|
|
|
|
SummaryCalcItem & calc2 = calcitems[2];
|
|
|
|
|
|
|
|
calc2.update(max, max);
|
|
|
|
|
|
|
|
num_days++;
|
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2018-06-10 07:53:10 +00:00
|
|
|
void gSessionTimesChart::afterDraw(QPainter & /*painter */, gGraph &graph, QRectF rect)
|
2014-09-11 14:23:08 +00:00
|
|
|
{
|
2014-09-13 11:34:18 +00:00
|
|
|
if (totaldays == nousedays) return;
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
SummaryCalcItem & calc = calcitems[0]; // session length
|
|
|
|
SummaryCalcItem & calc1 = calcitems[1]; // number of sessions
|
|
|
|
SummaryCalcItem & calc2 = calcitems[2]; // number of sessions
|
|
|
|
|
|
|
|
int midcalc = p_profile->general->prefCalcMiddle();
|
|
|
|
|
|
|
|
float mid = 0, mid1 = 0, midlongest = 0;
|
|
|
|
switch (midcalc) {
|
|
|
|
case 0:
|
|
|
|
if (calc.median_data.size() > 0) {
|
|
|
|
mid = median(calc.median_data.begin(), calc.median_data.end());
|
|
|
|
mid1 = median(calc1.median_data.begin(), calc1.median_data.end());
|
|
|
|
midlongest = median(calc2.median_data.begin(), calc2.median_data.end());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
mid = calc.wavg_sum / calc.divisor;
|
|
|
|
mid1 = calc1.wavg_sum / calc1.divisor;
|
|
|
|
midlongest = calc2.wavg_sum / calc2.divisor;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
mid = calc.avg_sum / calc.cnt;
|
|
|
|
mid1 = calc1.avg_sum / calc1.cnt;
|
|
|
|
midlongest = calc2.avg_sum / calc2.cnt;
|
|
|
|
break;
|
|
|
|
}
|
2014-09-13 11:34:18 +00:00
|
|
|
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
// float avgsess = float(num_slices) / float(num_days);
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
QString txt = QObject::tr("Sessions: %1 / %2 / %3 Length: %4 / %5 / %6 Longest: %7 / %8 / %9")
|
|
|
|
.arg(calc1.min, 0, 'f', 2).arg(mid1, 0, 'f', 2).arg(calc1.max, 0, 'f', 2)
|
|
|
|
.arg(calc.min, 0, 'f', 2).arg(mid, 0, 'f', 2).arg(calc.max, 0, 'f', 2)
|
|
|
|
.arg(calc2.min, 0, 'f', 2).arg(midlongest, 0, 'f', 2).arg(calc2.max, 0, 'f', 2);
|
2014-09-11 14:23:08 +00:00
|
|
|
graph.renderText(txt, rect.left(), rect.top()-5*graph.printScaleY(), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void gSessionTimesChart::paint(QPainter &painter, gGraph &graph, const QRegion ®ion)
|
|
|
|
{
|
2018-06-10 07:53:10 +00:00
|
|
|
QRectF rect = region.boundingRect();
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
painter.setPen(QColor(Qt::black));
|
|
|
|
painter.drawRect(rect);
|
|
|
|
|
|
|
|
m_minx = graph.min_x;
|
|
|
|
m_maxx = graph.max_x;
|
|
|
|
|
2015-08-11 20:01:24 +00:00
|
|
|
QDateTime date2 = QDateTime::fromMSecsSinceEpoch(m_minx, Qt::UTC);
|
|
|
|
QDateTime enddate2 = QDateTime::fromMSecsSinceEpoch(m_maxx, Qt::UTC);
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
QDate date = date2.date();
|
|
|
|
QDate enddate = enddate2.date();
|
|
|
|
|
|
|
|
int days = ceil(double(m_maxx - m_minx) / 86400000.0);
|
|
|
|
|
|
|
|
float barw = float(rect.width()) / float(days);
|
|
|
|
|
|
|
|
QDateTime splittime;
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto it = dayindex.find(date);
|
2014-09-11 14:23:08 +00:00
|
|
|
int idx=0;
|
2016-02-28 13:58:28 +00:00
|
|
|
|
|
|
|
if (it == dayindex.end()) {
|
|
|
|
it = dayindex.begin();
|
|
|
|
} else {
|
2014-09-11 14:23:08 +00:00
|
|
|
idx = it.value();
|
|
|
|
}
|
|
|
|
|
2019-08-23 02:59:51 +00:00
|
|
|
// Determine how many days after the first day of the chart that this data is to begin
|
2019-08-22 09:33:18 +00:00
|
|
|
int numDaysOffset = 0;
|
2019-08-23 02:59:51 +00:00
|
|
|
if (firstday > date) { // date = beginning date of chart; firstday = beginning date of data
|
2019-08-22 09:33:18 +00:00
|
|
|
numDaysOffset = date.daysTo(firstday);
|
|
|
|
}
|
|
|
|
|
2019-08-23 02:59:51 +00:00
|
|
|
// Determine how many days before the last day of the chart that this data is to end
|
|
|
|
int numDaysAfter = 0;
|
|
|
|
if (enddate > lastday) { // enddate = last date of chart; lastday = last date of data
|
|
|
|
numDaysAfter = lastday.daysTo(enddate);
|
|
|
|
}
|
|
|
|
if (numDaysAfter > days) // Nothing to do if this data is off the left edge of the chart
|
|
|
|
return;
|
|
|
|
|
2019-08-22 09:33:18 +00:00
|
|
|
// float lasty1 = rect.bottom();
|
|
|
|
float lastx1 = rect.left();
|
|
|
|
lastx1 += numDaysOffset * barw;
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto ite = dayindex.find(enddate);
|
2014-09-11 14:23:08 +00:00
|
|
|
int idx_end = daylist.size()-1;
|
|
|
|
if (ite != dayindex.end()) {
|
|
|
|
idx_end = ite.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
QPoint mouse = graph.graphView()->currentMousePos();
|
|
|
|
|
|
|
|
if (daylist.size() == 0) return;
|
|
|
|
|
|
|
|
QVector<QRectF> outlines;
|
|
|
|
int size = idx_end - idx;
|
|
|
|
outlines.reserve(size * 5);
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto it2 = it;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
/// Calculate Graph Peaks
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
peak_value = 0;
|
|
|
|
min_value = 999;
|
2018-05-05 21:58:11 +00:00
|
|
|
auto it_end = dayindex.end();
|
2014-09-18 10:53:59 +00:00
|
|
|
|
|
|
|
float right_edge = (rect.left()+rect.width()+1);
|
|
|
|
for (int i=idx; (i <= idx_end) && (it2 != it_end); ++i, ++it2, lastx1 += barw) {
|
2014-09-11 14:23:08 +00:00
|
|
|
Day * day = daylist.at(i);
|
|
|
|
|
2014-09-18 10:53:59 +00:00
|
|
|
if ((lastx1 + barw) > right_edge)
|
|
|
|
break;
|
2014-09-17 02:34:50 +00:00
|
|
|
|
2014-09-18 10:53:59 +00:00
|
|
|
if (!day) {
|
2014-09-11 14:23:08 +00:00
|
|
|
continue;
|
2014-09-18 10:53:59 +00:00
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto cit = cache.find(i);
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
if (cit == cache.end()) {
|
|
|
|
day->OpenSummary();
|
|
|
|
date = it2.key();
|
|
|
|
splittime = QDateTime(date, split);
|
|
|
|
|
2014-09-29 14:41:31 +00:00
|
|
|
QVector<SummaryChartSlice> & slices = cache[i];
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
bool haveoxi = day->hasMachine(MT_OXIMETER);
|
|
|
|
|
|
|
|
QColor goodcolor = haveoxi ? QColor(128,255,196) : QColor(64,128,255);
|
|
|
|
|
2014-09-24 01:42:14 +00:00
|
|
|
QString datestr = date.toString(Qt::SystemLocaleShortDate);
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (const auto & sess : day->sessions) {
|
2014-09-17 06:12:38 +00:00
|
|
|
if (!sess->enabled() || (sess->type() != m_machtype)) continue;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
// Look at mask on/off slices...
|
2018-05-05 21:58:11 +00:00
|
|
|
if (sess->m_slices.size() > 0) {
|
2014-09-11 14:23:08 +00:00
|
|
|
// segments
|
2018-05-05 21:58:11 +00:00
|
|
|
for (const auto & slice : sess->m_slices) {
|
2015-08-11 20:16:03 +00:00
|
|
|
QDateTime st = QDateTime::fromMSecsSinceEpoch(slice.start, Qt::LocalTime);
|
2014-09-18 14:31:31 +00:00
|
|
|
|
|
|
|
float s1 = float(splittime.secsTo(st)) / 3600.0;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
float s2 = double(slice.end - slice.start) / 3600000.0;
|
|
|
|
|
2019-06-04 00:12:17 +00:00
|
|
|
QColor col = (slice.status == MaskOn) ? goodcolor : Qt::black;
|
2014-09-24 01:42:14 +00:00
|
|
|
QString txt = QObject::tr("%1\nLength: %3\nStart: %2\n").arg(datestr).arg(st.time().toString("hh:mm:ss")).arg(s2,0,'f',2);
|
2014-09-18 14:31:31 +00:00
|
|
|
|
2019-06-04 00:12:17 +00:00
|
|
|
txt += (slice.status == MaskOn) ? QObject::tr("Mask On") : QObject::tr("Mask Off");
|
2014-09-18 14:31:31 +00:00
|
|
|
slices.append(SummaryChartSlice(&calcitems[0], s1, s2, txt, col));
|
2014-09-04 02:17:59 +00:00
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
} else {
|
|
|
|
// otherwise just show session duration
|
|
|
|
qint64 sf = sess->first();
|
2015-08-11 20:16:03 +00:00
|
|
|
QDateTime st = QDateTime::fromMSecsSinceEpoch(sf, Qt::LocalTime);
|
2014-09-11 14:23:08 +00:00
|
|
|
float s1 = float(splittime.secsTo(st)) / 3600.0;
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
float s2 = sess->hours();
|
|
|
|
|
2014-09-24 01:42:14 +00:00
|
|
|
QString txt = QObject::tr("%1\nLength: %3\nStart: %2").arg(datestr).arg(st.time().toString("hh:mm:ss")).arg(s2,0,'f',2);
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-13 11:34:18 +00:00
|
|
|
slices.append(SummaryChartSlice(&calcitems[0], s1, s2, txt, goodcolor));
|
2014-09-04 02:17:59 +00:00
|
|
|
}
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cit = cache.find(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cit != cache.end()) {
|
|
|
|
float peak = 0, base = 999;
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (const auto & slice : cit.value()) {
|
2014-09-11 14:23:08 +00:00
|
|
|
float s1 = slice.value;
|
|
|
|
float s2 = slice.height;
|
|
|
|
|
|
|
|
peak = qMax(peak, s1+s2);
|
|
|
|
base = qMin(base, s1);
|
|
|
|
}
|
|
|
|
peak_value = qMax(peak_value, peak);
|
|
|
|
min_value = qMin(min_value, base);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_miny = (min_value < 999) ? floor(min_value) : 0;
|
|
|
|
m_maxy = ceil(peak_value);
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
/// Y-Axis scaling
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
EventDataType miny;
|
|
|
|
EventDataType maxy;
|
|
|
|
|
|
|
|
graph.roundY(miny, maxy);
|
|
|
|
float ymult = float(rect.height()) / (maxy-miny);
|
|
|
|
|
|
|
|
|
|
|
|
preCalc();
|
2014-09-14 15:29:07 +00:00
|
|
|
totaldays = 0;
|
|
|
|
nousedays = 0;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2014-09-18 10:53:59 +00:00
|
|
|
lastx1 = rect.left();
|
2019-08-22 09:33:18 +00:00
|
|
|
lastx1 += numDaysOffset * barw;
|
2014-09-18 10:53:59 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
/// Main Loop scaling
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
do {
|
|
|
|
Day * day = daylist.at(idx);
|
|
|
|
|
2014-09-18 10:53:59 +00:00
|
|
|
if ((lastx1 + barw) > right_edge)
|
2014-09-11 14:23:08 +00:00
|
|
|
break;
|
|
|
|
|
2014-09-13 11:34:18 +00:00
|
|
|
totaldays++;
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2014-09-17 02:34:50 +00:00
|
|
|
if (!day) { // || !day->hasMachine(m_machtype)) {
|
2014-09-18 10:53:59 +00:00
|
|
|
// lasty1 = rect.bottom();
|
2014-09-11 14:23:08 +00:00
|
|
|
lastx1 += barw;
|
2014-09-13 11:34:18 +00:00
|
|
|
nousedays++;
|
2014-09-11 14:23:08 +00:00
|
|
|
// it++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
auto cit = cache.find(idx);
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
float x1 = lastx1 + barw;
|
|
|
|
|
2014-10-02 11:22:30 +00:00
|
|
|
//bool hl = false;
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2018-06-10 07:53:10 +00:00
|
|
|
QRectF rec2(lastx1, rect.top(), barw, rect.height());
|
2014-09-11 14:23:08 +00:00
|
|
|
if (rec2.contains(mouse)) {
|
|
|
|
QColor col2(255,0,0,64);
|
|
|
|
painter.fillRect(rec2, QBrush(col2));
|
2014-10-02 11:22:30 +00:00
|
|
|
//hl = true;
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (cit != cache.end()) {
|
2014-09-29 14:41:31 +00:00
|
|
|
QVector<SummaryChartSlice> & slices = cit.value();
|
2014-09-11 14:23:08 +00:00
|
|
|
|
|
|
|
customCalc(day, slices);
|
|
|
|
|
2014-09-16 02:15:19 +00:00
|
|
|
QLinearGradient gradient(lastx1, rect.bottom(), lastx1+barw, rect.bottom());
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (const auto & slice : slices) {
|
2014-09-11 14:23:08 +00:00
|
|
|
float s1 = slice.value - miny;
|
|
|
|
float s2 = slice.height;
|
2014-09-04 02:17:59 +00:00
|
|
|
|
|
|
|
float y1 = (s1 * ymult);
|
|
|
|
float y2 = (s2 * ymult);
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
QColor col = slice.color;
|
2014-09-04 02:17:59 +00:00
|
|
|
|
2018-06-10 07:53:10 +00:00
|
|
|
QRectF rec(lastx1, rect.bottom() - y1 - y2, barw, y2);
|
2014-09-11 14:23:08 +00:00
|
|
|
rec = rec.intersected(rect);
|
|
|
|
|
2014-09-04 02:17:59 +00:00
|
|
|
if (rec.contains(mouse)) {
|
2014-09-11 14:23:08 +00:00
|
|
|
col = Qt::yellow;
|
|
|
|
graph.ToolTip(slice.name, mouse.x() - 15,mouse.y() + 15, TT_AlignRight);
|
2014-09-04 02:17:59 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
if (barw > 8) {
|
|
|
|
gradient.setColorAt(0,col);
|
2014-09-16 02:15:19 +00:00
|
|
|
gradient.setColorAt(1,brighten(col, 2.0));
|
2014-09-11 14:23:08 +00:00
|
|
|
painter.fillRect(rec, QBrush(gradient));
|
2014-09-16 02:15:19 +00:00
|
|
|
// painter.fillRect(rec, brush);
|
2014-09-11 14:23:08 +00:00
|
|
|
outlines.append(rec);
|
|
|
|
} else if (barw > 3) {
|
|
|
|
painter.fillRect(rec, QBrush(brighten(col,1.25)));
|
|
|
|
outlines.append(rec);
|
|
|
|
} else {
|
|
|
|
painter.fillRect(rec, QBrush(col));
|
|
|
|
}
|
2014-09-04 02:17:59 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2014-05-31 21:25:07 +00:00
|
|
|
|
|
|
|
|
2014-09-04 02:17:59 +00:00
|
|
|
lastx1 = x1;
|
2014-09-11 14:23:08 +00:00
|
|
|
} while (++idx <= idx_end);
|
2014-05-31 21:25:07 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
painter.setPen(QPen(Qt::black,1));
|
|
|
|
painter.drawRects(outlines);
|
|
|
|
afterDraw(painter, graph, rect);
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 17:58:00 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// Total Time in Apnea Chart Stuff
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void gTTIAChart::preCalc()
|
|
|
|
{
|
|
|
|
gSummaryChart::preCalc();
|
|
|
|
}
|
2018-05-05 21:58:11 +00:00
|
|
|
|
2014-09-29 14:41:31 +00:00
|
|
|
void gTTIAChart::customCalc(Day *, QVector<SummaryChartSlice> & slices)
|
2014-09-18 17:58:00 +00:00
|
|
|
{
|
|
|
|
if (slices.size() == 0) return;
|
|
|
|
const SummaryChartSlice & slice = slices.at(0);
|
|
|
|
|
|
|
|
calcitems[0].update(slice.value, slice.value);
|
|
|
|
}
|
2018-05-05 21:58:11 +00:00
|
|
|
|
2018-06-10 07:53:10 +00:00
|
|
|
void gTTIAChart::afterDraw(QPainter &, gGraph &graph, QRectF rect)
|
2014-09-18 17:58:00 +00:00
|
|
|
{
|
|
|
|
QStringList txtlist;
|
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (auto & calc : calcitems) {
|
2014-09-29 14:41:31 +00:00
|
|
|
//ChannelID code = calc.code;
|
|
|
|
//schema::Channel & chan = schema::channel[code];
|
2014-09-18 17:58:00 +00:00
|
|
|
float mid = 0;
|
|
|
|
switch (midcalc) {
|
|
|
|
case 0:
|
|
|
|
if (calc.median_data.size() > 0) {
|
|
|
|
mid = median(calc.median_data.begin(), calc.median_data.end());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
if (calc.divisor > 0) {
|
|
|
|
mid = calc.wavg_sum / calc.divisor;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
2019-05-08 21:08:12 +00:00
|
|
|
if (calc.cnt > 0) {
|
|
|
|
mid = calc.avg_sum / calc.cnt;
|
2014-09-18 17:58:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
txtlist.append(QString("%1 %2 / %3 / %4").arg(QObject::tr("TTIA:")).arg(calc.min, 0, 'f', 2).arg(mid, 0, 'f', 2).arg(calc.max, 0, 'f', 2));
|
|
|
|
}
|
|
|
|
QString txt = txtlist.join(", ");
|
|
|
|
graph.renderText(txt, rect.left(), rect.top()-5*graph.printScaleY(), 0);
|
|
|
|
}
|
2018-05-05 21:58:11 +00:00
|
|
|
|
2014-09-18 17:58:00 +00:00
|
|
|
void gTTIAChart::populate(Day *day, int idx)
|
|
|
|
{
|
2014-09-29 14:41:31 +00:00
|
|
|
QVector<SummaryChartSlice> & slices = cache[idx];
|
2014-09-18 17:58:00 +00:00
|
|
|
float ttia = day->sum(CPAP_Obstructive) + day->sum(CPAP_ClearAirway) + day->sum(CPAP_Apnea) + day->sum(CPAP_Hypopnea);
|
|
|
|
int h = ttia / 3600;
|
|
|
|
int m = int(ttia) / 60 % 60;
|
|
|
|
int s = int(ttia) % 60;
|
|
|
|
slices.append(SummaryChartSlice(&calcitems[0], ttia / 60.0, ttia / 60.0, QObject::tr("\nTTIA: %1").arg(QString().sprintf("%02i:%02i:%02i",h,m,s)), QColor(255,147,150)));
|
|
|
|
}
|
2018-05-05 21:58:11 +00:00
|
|
|
|
2014-09-18 17:58:00 +00:00
|
|
|
QString gTTIAChart::tooltipData(Day *, int idx)
|
|
|
|
{
|
2014-09-29 14:41:31 +00:00
|
|
|
QVector<SummaryChartSlice> & slices = cache[idx];
|
2014-09-18 17:58:00 +00:00
|
|
|
if (slices.size() == 0) return QString();
|
|
|
|
|
|
|
|
const SummaryChartSlice & slice = slices.at(0);
|
|
|
|
return slice.name;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// AHI Chart Stuff
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
2014-09-11 14:23:08 +00:00
|
|
|
void gAHIChart::preCalc()
|
2014-05-31 21:25:07 +00:00
|
|
|
{
|
2014-09-13 11:34:18 +00:00
|
|
|
gSummaryChart::preCalc();
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
ahi_wavg = 0;
|
|
|
|
ahi_avg = 0;
|
2017-09-21 14:50:18 +00:00
|
|
|
total_days = 0;
|
2014-09-11 14:23:08 +00:00
|
|
|
total_hours = 0;
|
2014-09-13 11:34:18 +00:00
|
|
|
min_ahi = 99999;
|
|
|
|
max_ahi = -99999;
|
|
|
|
|
|
|
|
ahi_data.clear();
|
|
|
|
ahi_data.reserve(idx_end-idx_start);
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|
2014-09-29 14:41:31 +00:00
|
|
|
void gAHIChart::customCalc(Day *day, QVector<SummaryChartSlice> &list)
|
2014-05-31 21:25:07 +00:00
|
|
|
{
|
2014-09-11 14:23:08 +00:00
|
|
|
int size = list.size();
|
2014-09-13 11:34:18 +00:00
|
|
|
if (size == 0) return;
|
|
|
|
EventDataType hours = day->hours(m_machtype);
|
|
|
|
EventDataType ahi_cnt = 0;
|
2018-05-05 21:58:11 +00:00
|
|
|
|
|
|
|
for (auto & slice : list) {
|
2014-09-13 11:34:18 +00:00
|
|
|
SummaryCalcItem * calc = slice.calc;
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
EventDataType value = slice.value;
|
2014-09-16 02:15:19 +00:00
|
|
|
float valh = value/ hours;
|
2014-09-13 11:34:18 +00:00
|
|
|
|
2014-09-16 02:15:19 +00:00
|
|
|
switch (midcalc) {
|
|
|
|
case 0:
|
|
|
|
calc->median_data.append(valh);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
calc->wavg_sum += value;
|
|
|
|
calc->divisor += hours;
|
2019-05-08 21:06:17 +00:00
|
|
|
break;
|
|
|
|
case 2:
|
2014-09-16 02:15:19 +00:00
|
|
|
calc->avg_sum += value;
|
|
|
|
calc->cnt++;
|
|
|
|
break;
|
|
|
|
}
|
2014-09-14 15:29:07 +00:00
|
|
|
|
|
|
|
calc->min = qMin(valh, calc->min);
|
|
|
|
calc->max = qMax(valh, calc->max);
|
2014-09-13 11:34:18 +00:00
|
|
|
|
|
|
|
ahi_cnt += value;
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
2014-09-13 11:34:18 +00:00
|
|
|
min_ahi = qMin(ahi_cnt / hours, min_ahi);
|
|
|
|
max_ahi = qMax(ahi_cnt / hours, max_ahi);
|
|
|
|
|
|
|
|
ahi_data.append(ahi_cnt / hours);
|
|
|
|
|
2014-09-14 15:29:07 +00:00
|
|
|
ahi_wavg += ahi_cnt;
|
|
|
|
ahi_avg += ahi_cnt;
|
2014-09-11 14:23:08 +00:00
|
|
|
total_hours += hours;
|
2017-09-21 14:50:18 +00:00
|
|
|
total_days++;
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|
2018-06-10 07:53:10 +00:00
|
|
|
void gAHIChart::afterDraw(QPainter & /*painter */, gGraph &graph, QRectF rect)
|
2014-05-31 21:25:07 +00:00
|
|
|
{
|
2014-09-13 11:34:18 +00:00
|
|
|
if (totaldays == nousedays) return;
|
|
|
|
|
2014-09-16 02:15:19 +00:00
|
|
|
//int size = idx_end - idx_start;
|
2014-09-13 11:34:18 +00:00
|
|
|
|
2014-09-29 14:41:31 +00:00
|
|
|
bool skip = true;
|
2014-09-13 11:34:18 +00:00
|
|
|
float med = 0;
|
2014-09-16 02:15:19 +00:00
|
|
|
switch (midcalc) {
|
|
|
|
case 0:
|
|
|
|
if (ahi_data.size() > 0) {
|
|
|
|
med = median(ahi_data.begin(), ahi_data.end());
|
2014-09-29 14:41:31 +00:00
|
|
|
skip = false;
|
2014-09-16 02:15:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1: // wavg
|
2014-09-18 17:58:00 +00:00
|
|
|
if (total_hours > 0) {
|
|
|
|
med = ahi_wavg / total_hours;
|
2014-09-29 14:41:31 +00:00
|
|
|
skip = false;
|
2014-09-18 17:58:00 +00:00
|
|
|
}
|
2014-09-16 02:15:19 +00:00
|
|
|
break;
|
|
|
|
case 2: // avg
|
2017-09-21 14:50:18 +00:00
|
|
|
if (total_days > 0) {
|
|
|
|
med = ahi_avg / total_days;
|
2014-09-29 14:41:31 +00:00
|
|
|
skip = false;
|
2014-09-18 17:58:00 +00:00
|
|
|
}
|
2014-09-16 02:15:19 +00:00
|
|
|
break;
|
2014-09-13 11:34:18 +00:00
|
|
|
}
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
QStringList txtlist;
|
2014-09-29 14:41:31 +00:00
|
|
|
if (!skip) txtlist.append(QObject::tr("%1 %2 / %3 / %4").arg(STR_TR_AHI).arg(min_ahi, 0, 'f', 2).arg(med, 0, 'f', 2).arg(max_ahi, 0, 'f', 2));
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2020-08-03 03:24:56 +00:00
|
|
|
int i = calcitems.size();
|
|
|
|
while (i > 0) {
|
|
|
|
i--;
|
|
|
|
ChannelID code = calcitems[i].code;
|
2014-09-18 17:58:00 +00:00
|
|
|
schema::Channel & chan = schema::channel[code];
|
|
|
|
float mid = 0;
|
2014-09-29 14:41:31 +00:00
|
|
|
skip = true;
|
2014-09-18 17:58:00 +00:00
|
|
|
switch (midcalc) {
|
|
|
|
case 0:
|
2020-08-03 03:24:56 +00:00
|
|
|
if (calcitems[i].median_data.size() > 0) {
|
|
|
|
mid = median(calcitems[i].median_data.begin(), calcitems[i].median_data.end());
|
2014-09-29 14:41:31 +00:00
|
|
|
skip = false;
|
2014-09-18 17:58:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
2020-08-03 03:24:56 +00:00
|
|
|
if (calcitems[i].divisor > 0) {
|
|
|
|
mid = calcitems[i].wavg_sum / calcitems[i].divisor;
|
2014-09-29 14:41:31 +00:00
|
|
|
skip = false;
|
2014-09-18 17:58:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
2020-08-03 03:24:56 +00:00
|
|
|
if (calcitems[i].cnt > 0) {
|
|
|
|
mid = calcitems[i].avg_sum / calcitems[i].cnt;
|
2014-09-29 14:41:31 +00:00
|
|
|
skip = false;
|
2014-09-14 15:29:07 +00:00
|
|
|
}
|
2014-09-18 17:58:00 +00:00
|
|
|
break;
|
2014-09-13 11:34:18 +00:00
|
|
|
}
|
2014-09-18 17:58:00 +00:00
|
|
|
|
2020-08-03 03:24:56 +00:00
|
|
|
if (!skip) txtlist.append(QString("%1 %2 / %3 / %4").arg(chan.label()).arg(calcitems[i].min, 0, 'f', 2).arg(mid, 0, 'f', 2).arg(calcitems[i].max, 0, 'f', 2));
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
2014-09-13 11:34:18 +00:00
|
|
|
QString txt = txtlist.join(", ");
|
2014-09-11 14:23:08 +00:00
|
|
|
graph.renderText(txt, rect.left(), rect.top()-5*graph.printScaleY(), 0);
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
void gAHIChart::populate(Day *day, int idx)
|
2014-05-31 21:25:07 +00:00
|
|
|
{
|
2014-09-29 14:41:31 +00:00
|
|
|
QVector<SummaryChartSlice> & slices = cache[idx];
|
2014-09-11 14:23:08 +00:00
|
|
|
|
2020-03-29 23:21:34 +00:00
|
|
|
float hours = day->hours(m_machtype);
|
2014-09-13 11:34:18 +00:00
|
|
|
|
2018-05-05 21:58:11 +00:00
|
|
|
for (auto & calc : calcitems) {
|
2014-09-13 11:34:18 +00:00
|
|
|
ChannelID code = calc.code;
|
2014-09-11 14:23:08 +00:00
|
|
|
if (!day->hasData(code, ST_CNT)) continue;
|
2014-09-13 11:34:18 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
schema::Channel *chan = schema::channel.channels.find(code).value();
|
2014-09-13 11:34:18 +00:00
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
float c = day->count(code);
|
2014-09-13 11:34:18 +00:00
|
|
|
slices.append(SummaryChartSlice(&calc, c, c / hours, chan->label(), calc.color));
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
QString gAHIChart::tooltipData(Day *day, int idx)
|
|
|
|
{
|
2014-09-29 14:41:31 +00:00
|
|
|
QVector<SummaryChartSlice> & slices = cache[idx];
|
2014-09-11 14:23:08 +00:00
|
|
|
float total = 0;
|
|
|
|
float hour = day->hours(m_machtype);
|
|
|
|
QString txt;
|
2020-08-03 03:24:56 +00:00
|
|
|
int i = slices.size();
|
|
|
|
while (i > 0) {
|
|
|
|
i--;
|
|
|
|
total += slices[i].value;
|
|
|
|
txt += QString("\n%1: %2").arg(slices[i].name).arg(float(slices[i].value) / hour, 0, 'f', 2);
|
2014-09-11 14:23:08 +00:00
|
|
|
}
|
|
|
|
return QString("\n%1: %2").arg(STR_TR_AHI).arg(float(total) / hour,0,'f',2)+txt;
|
2014-05-31 21:25:07 +00:00
|
|
|
}
|