2014-08-17 12:56:05 +00:00
|
|
|
/* gLineChart Implementation
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
|
|
|
* 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. */
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-23 17:20:38 +00:00
|
|
|
#include "Graphs/gLineChart.h"
|
|
|
|
|
2011-06-26 08:30:44 +00:00
|
|
|
#include <QString>
|
2011-07-01 10:10:44 +00:00
|
|
|
#include <QDebug>
|
2014-04-23 17:20:38 +00:00
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
#include "Graphs/glcommon.h"
|
|
|
|
#include "Graphs/gGraph.h"
|
|
|
|
#include "Graphs/gGraphView.h"
|
|
|
|
#include "SleepLib/profiles.h"
|
2014-08-05 11:17:03 +00:00
|
|
|
#include "Graphs/gLineOverlay.h"
|
2011-06-26 08:30:44 +00:00
|
|
|
|
|
|
|
#define EXTRA_ASSERTS 1
|
2014-08-17 12:56:05 +00:00
|
|
|
|
2014-08-23 09:24:22 +00:00
|
|
|
QDataStream & operator<<(QDataStream & stream, const DottedLine & dot)
|
|
|
|
{
|
|
|
|
stream << dot.code;
|
|
|
|
stream << dot.type;
|
|
|
|
stream << dot.value;
|
|
|
|
stream << dot.visible;
|
|
|
|
stream << dot.available;
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
QDataStream & operator>>(QDataStream & stream, DottedLine & dot)
|
|
|
|
{
|
|
|
|
quint32 tmp;
|
|
|
|
stream >> dot.code;
|
|
|
|
stream >> tmp;
|
|
|
|
stream >> dot.value;
|
|
|
|
stream >> dot.visible;
|
|
|
|
stream >> dot.available;
|
|
|
|
dot.type = (ChannelCalcType)tmp;
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
QColor darken(QColor color, float p)
|
|
|
|
{
|
|
|
|
int r = qMin(int(color.red() * p), 255);
|
|
|
|
int g = qMin(int(color.green() * p), 255);
|
|
|
|
int b = qMin(int(color.blue() * p), 255);
|
|
|
|
|
|
|
|
|
|
|
|
return QColor(r,g,b, color.alpha());
|
|
|
|
}
|
|
|
|
|
2014-08-17 15:36:53 +00:00
|
|
|
gLineChart::gLineChart(ChannelID code, bool square_plot, bool disable_accel)
|
2014-04-17 05:55:38 +00:00
|
|
|
: Layer(code), m_square_plot(square_plot), m_disable_accel(disable_accel)
|
2011-06-26 08:30:44 +00:00
|
|
|
{
|
2014-08-17 15:36:53 +00:00
|
|
|
addPlot(code, square_plot);
|
2014-04-17 05:55:38 +00:00
|
|
|
m_report_empty = false;
|
2014-05-10 00:50:35 +00:00
|
|
|
lines.reserve(50000);
|
2014-07-21 16:14:07 +00:00
|
|
|
lasttime = 0;
|
2014-08-17 12:56:05 +00:00
|
|
|
m_layertype = LT_LineChart;
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
|
|
|
gLineChart::~gLineChart()
|
|
|
|
{
|
2014-08-05 11:30:25 +00:00
|
|
|
QHash<ChannelID, gLineOverlayBar *>::iterator fit;
|
|
|
|
for (fit = flags.begin(); fit != flags.end(); ++fit) {
|
|
|
|
// destroy any overlay bar from previous day
|
|
|
|
delete fit.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
flags.clear();
|
|
|
|
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
bool gLineChart::isEmpty()
|
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
if (!m_day) { return true; }
|
|
|
|
|
|
|
|
for (int j = 0; j < m_codes.size(); j++) {
|
|
|
|
ChannelID code = m_codes[j];
|
|
|
|
|
|
|
|
for (int i = 0; i < m_day->size(); i++) {
|
2014-07-25 07:53:48 +00:00
|
|
|
Session *sess = m_day->sessions[i];
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
if (sess->channelExists(code)) {
|
2011-12-27 13:21:10 +00:00
|
|
|
return false;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
|
2011-09-23 03:54:48 +00:00
|
|
|
void gLineChart::SetDay(Day *d)
|
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
// Layer::SetDay(d);
|
|
|
|
m_day = d;
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
m_minx = 0, m_maxx = 0;
|
|
|
|
m_miny = 0, m_maxy = 0;
|
|
|
|
m_physminy = 0, m_physmaxy = 0;
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (!d) {
|
2013-10-25 10:39:30 +00:00
|
|
|
return;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2011-12-27 13:21:10 +00:00
|
|
|
|
|
|
|
qint64 t64;
|
2013-10-25 10:39:30 +00:00
|
|
|
EventDataType tmp;
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
bool first = true;
|
|
|
|
|
|
|
|
for (int j = 0; j < m_codes.size(); j++) {
|
|
|
|
ChannelID code = m_codes[j];
|
|
|
|
|
|
|
|
for (int i = 0; i < d->size(); i++) {
|
2014-07-25 07:53:48 +00:00
|
|
|
Session *sess = d->sessions[i];
|
2014-08-20 20:50:55 +00:00
|
|
|
if (!sess->enabled()) continue;
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2014-07-02 05:11:32 +00:00
|
|
|
if (code == CPAP_MaskPressure) {
|
|
|
|
if (sess->channelExists(CPAP_MaskPressureHi)) {
|
|
|
|
code = m_codes[j] = CPAP_MaskPressureHi;
|
2014-08-17 15:36:53 +00:00
|
|
|
m_enabled[code] = schema::channel[CPAP_MaskPressureHi].enabled();
|
2014-08-20 20:50:55 +00:00
|
|
|
|
2014-07-02 05:11:32 +00:00
|
|
|
goto skipcheck; // why not :P
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!sess->channelExists(code)) {
|
|
|
|
continue;
|
|
|
|
}
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-07-02 05:11:32 +00:00
|
|
|
skipcheck:
|
2013-10-25 10:39:30 +00:00
|
|
|
if (first) {
|
2014-04-17 05:55:38 +00:00
|
|
|
m_miny = sess->Min(code);
|
|
|
|
m_maxy = sess->Max(code);
|
|
|
|
m_physminy = sess->physMin(code);
|
|
|
|
m_physmaxy = sess->physMax(code);
|
|
|
|
m_minx = sess->first(code);
|
|
|
|
m_maxx = sess->last(code);
|
|
|
|
first = false;
|
2013-10-25 10:39:30 +00:00
|
|
|
} else {
|
2014-04-17 05:55:38 +00:00
|
|
|
tmp = sess->physMin(code);
|
|
|
|
|
|
|
|
if (m_physminy > tmp) {
|
|
|
|
m_physminy = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = sess->physMax(code);
|
|
|
|
|
|
|
|
if (m_physmaxy < tmp) {
|
|
|
|
m_physmaxy = tmp;
|
|
|
|
}
|
2013-10-25 15:25:13 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
tmp = sess->Min(code);
|
2013-10-25 15:25:13 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (m_miny > tmp) {
|
|
|
|
m_miny = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = sess->Max(code);
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (m_maxy < tmp) {
|
|
|
|
m_maxy = tmp;
|
|
|
|
}
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
t64 = sess->first(code);
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (m_minx > t64) {
|
|
|
|
m_minx = t64;
|
|
|
|
}
|
|
|
|
|
|
|
|
t64 = sess->last(code);
|
|
|
|
|
|
|
|
if (m_maxx < t64) {
|
|
|
|
m_maxx = t64;
|
|
|
|
}
|
2013-10-25 10:39:30 +00:00
|
|
|
}
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2011-10-05 07:41:56 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
subtract_offset = 0;
|
2014-08-05 11:17:03 +00:00
|
|
|
|
|
|
|
QHash<ChannelID, gLineOverlayBar *>::iterator fit;
|
|
|
|
for (fit = flags.begin(); fit != flags.end(); ++fit) {
|
|
|
|
// destroy any overlay bar from previous day
|
|
|
|
delete fit.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
flags.clear();
|
|
|
|
|
2014-08-07 20:27:23 +00:00
|
|
|
quint32 z = schema::FLAG | schema::MINOR_FLAG | schema::SPAN;
|
|
|
|
if (p_profile->general->showUnknownFlags()) z |= schema::UNKNOWN;
|
|
|
|
QList<ChannelID> available = m_day->getSortedMachineChannels(z);
|
|
|
|
|
|
|
|
for (int i=0; i < available.size(); ++i) {
|
|
|
|
ChannelID code = available.at(i);
|
2014-08-23 06:21:50 +00:00
|
|
|
if (!m_flags_enabled.contains(code)) {
|
|
|
|
bool b = false;
|
|
|
|
|
|
|
|
if (((m_codes[0] == CPAP_FlowRate) || ((m_codes[0] == CPAP_MaskPressureHi))) && (schema::channel[code].machtype() == MT_CPAP)) b = true;
|
|
|
|
if ((m_codes[0] == CPAP_Leak) && (code == CPAP_LargeLeak)) b = true;
|
|
|
|
m_flags_enabled[code] = b;
|
|
|
|
}
|
2014-08-11 19:00:18 +00:00
|
|
|
if (!m_day->channelExists(code)) continue;
|
|
|
|
|
2014-08-07 20:27:23 +00:00
|
|
|
|
|
|
|
schema::Channel * chan = &schema::channel[code];
|
|
|
|
gLineOverlayBar * lob = nullptr;
|
|
|
|
|
|
|
|
if (chan->type() == schema::FLAG) {
|
|
|
|
lob = new gLineOverlayBar(code, chan->defaultColor(), chan->label(), FT_Bar);
|
|
|
|
} else if ((chan->type() == schema::MINOR_FLAG) || (chan->type() == schema::UNKNOWN)) {
|
|
|
|
lob = new gLineOverlayBar(code, chan->defaultColor(), chan->label(), FT_Dot);
|
|
|
|
} else if (chan->type() == schema::SPAN) {
|
|
|
|
lob = new gLineOverlayBar(code, chan->defaultColor(), chan->label(), FT_Span);
|
|
|
|
}
|
|
|
|
if (lob != nullptr) {
|
2014-08-27 09:00:55 +00:00
|
|
|
lob->setOverlayDisplayType(((m_codes[0] == CPAP_FlowRate))? (OverlayDisplayType)p_profile->appearance->overlayType() : ODT_TopAndBottom);
|
2014-08-07 20:27:23 +00:00
|
|
|
lob->SetDay(m_day);
|
|
|
|
flags[code] = lob;
|
2014-08-05 11:17:03 +00:00
|
|
|
}
|
|
|
|
}
|
2014-08-23 09:24:22 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
m_dotlines.clear();
|
|
|
|
|
|
|
|
for (int i=0; i< m_codes.size(); i++) {
|
|
|
|
ChannelID code = m_codes[i];
|
|
|
|
schema::Channel & chan = schema::channel[code];
|
|
|
|
addDotLine(DottedLine(code, Calc_Max,chan.calc[Calc_Max].enabled));
|
|
|
|
if ((code != CPAP_FlowRate) && (code != CPAP_MaskPressure) && (code != CPAP_MaskPressureHi)) {
|
|
|
|
addDotLine(DottedLine(code, Calc_Perc,chan.calc[Calc_Perc].enabled));
|
|
|
|
addDotLine(DottedLine(code, Calc_Middle, chan.calc[Calc_Middle].enabled));
|
|
|
|
}
|
2014-08-17 23:29:30 +00:00
|
|
|
if ((code != CPAP_Snore) && (code != CPAP_FlowLimit) && (code != CPAP_RDI) && (code != CPAP_AHI)) {
|
|
|
|
addDotLine(DottedLine(code, Calc_Min, chan.calc[Calc_Min].enabled));
|
|
|
|
}
|
2014-08-17 12:56:05 +00:00
|
|
|
}
|
|
|
|
if (m_codes[0] == CPAP_Leak) {
|
|
|
|
addDotLine(DottedLine(CPAP_Leak, Calc_UpperThresh, schema::channel[CPAP_Leak].calc[Calc_UpperThresh].enabled));
|
|
|
|
} else if (m_codes[0] == CPAP_FlowRate) {
|
|
|
|
addDotLine(DottedLine(CPAP_FlowRate, Calc_Zero, schema::channel[CPAP_FlowRate].calc[Calc_Zero].enabled));
|
2014-08-23 09:54:51 +00:00
|
|
|
} else if (m_codes[0] == OXI_Pulse) {
|
|
|
|
addDotLine(DottedLine(OXI_Pulse, Calc_UpperThresh, schema::channel[OXI_Pulse].calc[Calc_UpperThresh].enabled));
|
|
|
|
addDotLine(DottedLine(OXI_Pulse, Calc_LowerThresh, schema::channel[OXI_Pulse].calc[Calc_LowerThresh].enabled));
|
|
|
|
} else if (m_codes[0] == OXI_SPO2) {
|
|
|
|
addDotLine(DottedLine(OXI_SPO2, Calc_LowerThresh, schema::channel[OXI_SPO2].calc[Calc_LowerThresh].enabled));
|
2014-08-17 12:56:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (m_day) {
|
|
|
|
for (int i=0; i < m_dotlines.size(); i++) {
|
|
|
|
m_dotlines[i].calc(m_day);
|
2014-08-23 09:24:22 +00:00
|
|
|
|
|
|
|
ChannelID code = m_dotlines[i].code;
|
|
|
|
ChannelCalcType type = m_dotlines[i].type;
|
|
|
|
|
|
|
|
bool b = false; // default value
|
|
|
|
|
|
|
|
QHash<ChannelID, QHash<quint32, bool> >::iterator cit = m_dot_enabled.find(code);
|
|
|
|
|
|
|
|
if (cit == m_dot_enabled.end()) {
|
|
|
|
m_dot_enabled[code].insert(type, b);
|
|
|
|
} else {
|
|
|
|
QHash<quint32, bool>::iterator it = cit.value().find(type);
|
|
|
|
if (it == cit.value().end()) {
|
|
|
|
cit.value().insert(type, b);
|
|
|
|
}
|
|
|
|
}
|
2014-08-17 12:56:05 +00:00
|
|
|
}
|
|
|
|
}
|
2011-09-23 03:54:48 +00:00
|
|
|
}
|
|
|
|
EventDataType gLineChart::Miny()
|
|
|
|
{
|
2014-08-17 12:56:05 +00:00
|
|
|
int size = m_codes.size();
|
|
|
|
if (size == 0) return 0;
|
|
|
|
if (!m_day) return 0;
|
|
|
|
|
|
|
|
bool first = false;
|
|
|
|
EventDataType min = 0, tmp;
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
for (int i=0; i< size; ++i) {
|
|
|
|
ChannelID code = m_codes[i];
|
2014-08-27 09:00:55 +00:00
|
|
|
if (!m_enabled[code] || !m_day->channelExists(code)) continue;
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
tmp = m_day->Min(code);
|
|
|
|
|
|
|
|
if (!first) {
|
|
|
|
min = tmp;
|
2014-08-17 15:36:53 +00:00
|
|
|
first = true;
|
2014-08-17 12:56:05 +00:00
|
|
|
} else {
|
|
|
|
min = qMin(tmp, min);
|
|
|
|
}
|
2011-09-23 03:54:48 +00:00
|
|
|
}
|
2014-08-17 12:56:05 +00:00
|
|
|
if (!first) min = 0;
|
|
|
|
|
|
|
|
m_miny = min;
|
|
|
|
|
|
|
|
return min;
|
|
|
|
// int m = Layer::Miny();
|
|
|
|
|
|
|
|
// if (subtract_offset > 0) {
|
|
|
|
// m -= subtract_offset;
|
|
|
|
|
|
|
|
// if (m < 0) { m = 0; }
|
|
|
|
// }
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
// return m;
|
2011-09-23 03:54:48 +00:00
|
|
|
}
|
|
|
|
EventDataType gLineChart::Maxy()
|
|
|
|
{
|
2014-08-17 12:56:05 +00:00
|
|
|
int size = m_codes.size();
|
|
|
|
if (size == 0) return 0;
|
|
|
|
if (!m_day) return 0;
|
|
|
|
|
|
|
|
bool first = false;
|
|
|
|
EventDataType max = 0, tmp;
|
|
|
|
|
|
|
|
for (int i=0; i< size; ++i) {
|
|
|
|
ChannelID code = m_codes[i];
|
2014-08-27 09:00:55 +00:00
|
|
|
if (!m_enabled[code] || !m_day->channelExists(code)) continue;
|
2014-08-17 12:56:05 +00:00
|
|
|
|
|
|
|
tmp = m_day->Max(code);
|
|
|
|
if (!first) {
|
|
|
|
max = tmp;
|
|
|
|
first = true;
|
|
|
|
} else {
|
|
|
|
max = qMax(tmp, max);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!first) max = 0;
|
|
|
|
m_maxy = max;
|
|
|
|
return max;
|
|
|
|
|
|
|
|
// return Layer::Maxy() - subtract_offset;
|
2011-09-23 03:54:48 +00:00
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2014-07-20 09:22:31 +00:00
|
|
|
bool gLineChart::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
|
|
|
|
{
|
|
|
|
Q_UNUSED(event)
|
2014-07-21 09:39:26 +00:00
|
|
|
Q_UNUSED(graph)
|
2014-08-12 10:44:05 +00:00
|
|
|
graph->timedRedraw(0);
|
|
|
|
|
|
|
|
return false;
|
2014-07-20 09:22:31 +00:00
|
|
|
}
|
|
|
|
|
2014-07-21 16:14:07 +00:00
|
|
|
QString gLineChart::getMetaString(qint64 time)
|
2014-07-21 02:41:25 +00:00
|
|
|
{
|
2014-07-21 16:14:07 +00:00
|
|
|
lasttext = QString();
|
|
|
|
if (!m_day) return lasttext;
|
|
|
|
|
|
|
|
|
|
|
|
EventDataType val;
|
|
|
|
|
|
|
|
EventDataType ipap = 0, epap = 0;
|
|
|
|
bool addPS = false;
|
|
|
|
|
|
|
|
for (int i=0; i<m_codes.size(); ++i) {
|
|
|
|
ChannelID code = m_codes[i];
|
2014-08-22 13:59:27 +00:00
|
|
|
|
2014-07-21 16:14:07 +00:00
|
|
|
if (m_day->channelHasData(code)) {
|
2014-08-06 07:28:24 +00:00
|
|
|
val = m_day->lookupValue(code, time, m_square_plot);
|
2014-08-17 12:56:05 +00:00
|
|
|
lasttext += " "+QString("%1: %2").arg(schema::channel[code].label()).arg(val,0,'f',2); //.arg(schema::channel[code].units());
|
2014-07-21 16:14:07 +00:00
|
|
|
|
|
|
|
if (code == CPAP_IPAP) {
|
|
|
|
ipap = val;
|
|
|
|
addPS = true;
|
|
|
|
}
|
|
|
|
if (code == CPAP_EPAP) {
|
|
|
|
epap = val;
|
2014-07-21 02:41:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-21 16:14:07 +00:00
|
|
|
}
|
|
|
|
if (addPS) {
|
|
|
|
val = ipap - epap;
|
2014-08-17 12:56:05 +00:00
|
|
|
lasttext += " "+QString("%1: %2").arg(schema::channel[CPAP_PS].label()).arg(val,0,'f',2);//.arg(schema::channel[CPAP_PS].units());
|
2014-07-21 02:41:25 +00:00
|
|
|
}
|
|
|
|
|
2014-07-21 16:14:07 +00:00
|
|
|
lasttime = time;
|
|
|
|
return lasttext;
|
2014-07-21 02:41:25 +00:00
|
|
|
}
|
2014-07-20 16:22:51 +00:00
|
|
|
|
2011-06-26 08:30:44 +00:00
|
|
|
// Time Domain Line Chart
|
2014-05-13 17:47:33 +00:00
|
|
|
void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
2011-06-26 08:30:44 +00:00
|
|
|
{
|
2014-08-05 11:17:03 +00:00
|
|
|
QRect rect = region.boundingRect();
|
2014-05-13 17:47:33 +00:00
|
|
|
// TODO: Just use QRect directly.
|
2014-08-05 11:17:03 +00:00
|
|
|
int left = rect.left();
|
|
|
|
int top = rect.top();
|
|
|
|
int width = rect.width();
|
|
|
|
int height = rect.height();
|
2014-05-13 17:47:33 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (!m_visible) {
|
2011-06-26 08:30:44 +00:00
|
|
|
return;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2011-08-25 06:11:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (!m_day) {
|
2011-06-26 08:30:44 +00:00
|
|
|
return;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2011-08-25 06:11:44 +00:00
|
|
|
|
2011-11-25 12:13:35 +00:00
|
|
|
//if (!m_day->channelExists(m_code)) return;
|
2011-09-17 12:39:00 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (width < 0) {
|
2011-08-01 04:34:55 +00:00
|
|
|
return;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2011-08-25 06:11:44 +00:00
|
|
|
|
2014-05-13 21:44:00 +00:00
|
|
|
|
2014-05-06 02:37:29 +00:00
|
|
|
top++;
|
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
double minx, maxx;
|
2013-10-25 15:25:13 +00:00
|
|
|
|
2011-08-29 11:42:40 +00:00
|
|
|
|
|
|
|
if (w.blockZoom()) {
|
2014-04-17 05:55:38 +00:00
|
|
|
minx = w.rmin_x, maxx = w.rmax_x;
|
2011-08-29 11:42:40 +00:00
|
|
|
} else {
|
2014-04-17 05:55:38 +00:00
|
|
|
maxx = w.max_x, minx = w.min_x;
|
2011-08-29 11:42:40 +00:00
|
|
|
}
|
|
|
|
|
2011-09-23 03:54:48 +00:00
|
|
|
// hmmm.. subtract_offset..
|
2011-08-29 11:42:40 +00:00
|
|
|
|
2014-05-20 14:52:29 +00:00
|
|
|
EventDataType miny = m_physminy;
|
|
|
|
EventDataType maxy = m_physmaxy;
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
w.roundY(miny, maxy);
|
|
|
|
|
2014-05-20 14:52:29 +00:00
|
|
|
|
2014-05-20 12:07:32 +00:00
|
|
|
//#define DEBUG_AUTOSCALER
|
2014-05-20 11:51:47 +00:00
|
|
|
#ifdef DEBUG_AUTOSCALER
|
|
|
|
QString a = QString().sprintf("%.2f - %.2f",miny, maxy);
|
|
|
|
w.renderText(a,width/2,top-5);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// the middle of minx and maxy does not have to be the center...
|
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
double logX = painter.device()->logicalDpiX();
|
|
|
|
double physX = painter.device()->physicalDpiX();
|
|
|
|
double ratioX = physX / logX * w.printScaleX();
|
|
|
|
double logY = painter.device()->logicalDpiY();
|
|
|
|
double physY = painter.device()->physicalDpiY();
|
|
|
|
double ratioY = physY / logY * w.printScaleY();
|
2014-05-20 11:51:47 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
double xx = maxx - minx;
|
|
|
|
double xmult = double(width) / xx;
|
2011-07-30 06:03:08 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
EventDataType yy = maxy - miny;
|
|
|
|
EventDataType ymult = EventDataType(height - 3) / yy; // time to pixel conversion multiplier
|
2011-06-26 08:30:44 +00:00
|
|
|
|
|
|
|
// Return on screwy min/max conditions
|
2014-04-17 05:55:38 +00:00
|
|
|
if (xx < 0) {
|
2011-06-26 08:30:44 +00:00
|
|
|
return;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (yy <= 0) {
|
|
|
|
if (miny == 0) {
|
2011-08-07 06:21:09 +00:00
|
|
|
return;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-08-05 11:17:03 +00:00
|
|
|
bool mouseover = false;
|
|
|
|
if (rect.contains(w.graphView()->currentMousePos())) {
|
|
|
|
mouseover = true;
|
|
|
|
|
|
|
|
painter.fillRect(rect, QBrush(QColor(255,255,245,128)));
|
|
|
|
}
|
|
|
|
|
2014-07-21 02:41:25 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
bool linecursormode = p_profile->appearance->lineCursorMode();
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
2014-07-29 14:38:59 +00:00
|
|
|
// Display Line Cursor
|
2014-08-17 12:56:05 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
if (linecursormode) {
|
2014-07-29 14:38:59 +00:00
|
|
|
double time = w.currentTime();
|
2014-07-21 13:47:17 +00:00
|
|
|
|
2014-07-21 07:14:02 +00:00
|
|
|
if ((time > minx) && (time < maxx)) {
|
2014-07-29 14:38:59 +00:00
|
|
|
double xpos = (time - double(minx)) * xmult;
|
2014-07-21 09:39:26 +00:00
|
|
|
painter.setPen(QPen(QBrush(QColor(0,255,0,255)),1));
|
2014-07-21 07:14:02 +00:00
|
|
|
painter.drawLine(left+xpos, top-w.marginTop()-3, left+xpos, top+height+w.bottom-1);
|
|
|
|
}
|
2014-07-20 09:22:31 +00:00
|
|
|
|
2014-07-21 16:14:07 +00:00
|
|
|
if ((time != lasttime) || lasttext.isEmpty()) {
|
|
|
|
getMetaString(time);
|
2014-07-21 13:47:17 +00:00
|
|
|
}
|
2014-07-21 02:41:25 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
if (m_codes[0] != CPAP_FlowRate) {
|
|
|
|
QString text = lasttext;
|
2014-07-20 09:22:31 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
int wid, h;
|
|
|
|
GetTextExtent(text, wid, h);
|
|
|
|
w.renderText(text, left , top-h+5); //+ width/2 - wid/2
|
|
|
|
}
|
2014-07-20 09:22:31 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
EventDataType lastpx, lastpy;
|
|
|
|
EventDataType px, py;
|
2011-07-30 05:32:56 +00:00
|
|
|
int idx;
|
2012-01-05 11:35:23 +00:00
|
|
|
bool done;
|
2014-04-17 05:55:38 +00:00
|
|
|
double x0, xL;
|
2011-07-27 09:21:53 +00:00
|
|
|
double sr;
|
|
|
|
int sam;
|
2014-04-17 05:55:38 +00:00
|
|
|
int minz, maxz;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2011-07-27 09:21:53 +00:00
|
|
|
// Draw bounding box
|
2014-05-07 19:52:59 +00:00
|
|
|
painter.setPen(QColor(Qt::black));
|
|
|
|
painter.drawLine(left, top, left, top + height);
|
|
|
|
painter.drawLine(left, top + height, left + width, top + height);
|
|
|
|
painter.drawLine(left + width, top + height, left + width, top);
|
|
|
|
painter.drawLine(left + width, top, left, top);
|
|
|
|
|
2011-06-26 08:30:44 +00:00
|
|
|
width--;
|
2014-04-17 05:55:38 +00:00
|
|
|
height -= 2;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
int num_points = 0;
|
|
|
|
int visible_points = 0;
|
|
|
|
int total_points = 0;
|
|
|
|
int total_visible = 0;
|
|
|
|
bool square_plot, accel;
|
2014-07-11 12:09:38 +00:00
|
|
|
qint64 clockdrift = qint64(p_profile->cpap->clockDrift()) * 1000L;
|
2014-04-17 05:55:38 +00:00
|
|
|
qint64 drift = 0;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
QHash<ChannelID, QVector<EventList *> >::iterator ci;
|
2011-08-01 04:34:55 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
//m_line_color=schema::channel[m_code].defaultColor();
|
2014-04-17 05:55:38 +00:00
|
|
|
int legendx = left + width;
|
2011-09-17 13:21:18 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
int codepoints;
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2014-05-08 02:05:01 +00:00
|
|
|
painter.setClipRect(left, top, width, height+1);
|
2014-05-07 19:52:59 +00:00
|
|
|
painter.setClipping(true);
|
2014-07-11 12:09:38 +00:00
|
|
|
painter.setRenderHint(QPainter::Antialiasing, p_profile->appearance->antiAliasing());
|
2014-05-07 19:52:59 +00:00
|
|
|
|
2014-08-11 18:29:44 +00:00
|
|
|
painter.setFont(*defaultfont);
|
2014-08-17 12:56:05 +00:00
|
|
|
bool showDottedLines = true;
|
|
|
|
|
|
|
|
int dotlinesize = m_dotlines.size();
|
|
|
|
|
|
|
|
// Unset Dotted lines visible status, so we only draw necessary labels later
|
|
|
|
for (int i=0; i < dotlinesize; i++) {
|
|
|
|
DottedLine & dot = m_dotlines[i];
|
|
|
|
dot.visible = false;
|
|
|
|
}
|
2014-05-15 19:45:46 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
for (int gi = 0; gi < m_codes.size(); gi++) {
|
|
|
|
ChannelID code = m_codes[gi];
|
2014-05-15 21:48:53 +00:00
|
|
|
schema::Channel &chan = schema::channel[code];
|
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Draw the Channel Threshold dotted lines, and flow waveform centreline
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
if (showDottedLines) {
|
|
|
|
for (int i=0; i < dotlinesize; i++) {
|
|
|
|
DottedLine & dot = m_dotlines[i];
|
2014-08-23 09:24:22 +00:00
|
|
|
if ((dot.code != code) || (!m_dot_enabled[dot.code][dot.type]) || (!dot.available) || (!m_enabled[dot.code])) {
|
2014-08-17 12:56:05 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
schema::Channel & chan = schema::channel[code];
|
|
|
|
|
|
|
|
dot.visible = true;
|
|
|
|
QColor color = chan.calc[dot.type].color;
|
|
|
|
color.setAlpha(200);
|
2014-08-17 23:29:30 +00:00
|
|
|
painter.setPen(QPen(QBrush(color), p_profile->appearance->lineThickness(), Qt::DotLine));
|
2014-08-17 12:56:05 +00:00
|
|
|
EventDataType y=top + height + 1 - ((dot.value - miny) * ymult);
|
|
|
|
painter.drawLine(left + 1, y, left + 1 + width, y);
|
2014-05-15 21:48:53 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
}
|
2014-05-15 21:48:53 +00:00
|
|
|
}
|
2014-08-17 12:56:05 +00:00
|
|
|
if (!m_enabled[code]) continue;
|
|
|
|
|
2014-05-08 02:05:01 +00:00
|
|
|
|
|
|
|
lines.clear();
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
codepoints = 0;
|
|
|
|
|
2014-08-21 14:16:23 +00:00
|
|
|
// For each session...
|
2014-05-08 11:18:19 +00:00
|
|
|
int daysize = m_day->size();
|
|
|
|
for (int svi = 0; svi < daysize; svi++) {
|
2014-04-17 05:55:38 +00:00
|
|
|
Session *sess = (*m_day)[svi];
|
|
|
|
|
2011-12-28 12:03:48 +00:00
|
|
|
if (!sess) {
|
2014-04-23 13:19:56 +00:00
|
|
|
qWarning() << "gLineChart::Plot() nullptr Session Record.. This should not happen";
|
2011-12-27 13:21:10 +00:00
|
|
|
continue;
|
|
|
|
}
|
2012-01-19 15:18:34 +00:00
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
drift = (sess->machine()->type() == MT_CPAP) ? clockdrift : 0;
|
2012-01-19 15:18:34 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (!sess->enabled()) { continue; }
|
|
|
|
|
|
|
|
schema::Channel ch = schema::channel[code];
|
|
|
|
bool fndbetter = false;
|
|
|
|
|
2014-05-08 11:18:19 +00:00
|
|
|
QList<schema::Channel *>::iterator mlend=ch.m_links.end();
|
|
|
|
for (QList<schema::Channel *>::iterator l = ch.m_links.begin(); l != mlend; l++) {
|
|
|
|
schema::Channel &c = *(*l);
|
|
|
|
ci = (*m_day)[svi]->eventlist.find(c.id());
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
if (ci != (*m_day)[svi]->eventlist.end()) {
|
|
|
|
fndbetter = true;
|
2011-12-27 13:21:10 +00:00
|
|
|
break;
|
|
|
|
}
|
2011-09-17 13:21:18 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
if (!fndbetter) {
|
2014-04-17 05:55:38 +00:00
|
|
|
ci = (*m_day)[svi]->eventlist.find(code);
|
|
|
|
|
|
|
|
if (ci == (*m_day)[svi]->eventlist.end()) { continue; }
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QVector<EventList *> &evec = ci.value();
|
|
|
|
num_points = 0;
|
|
|
|
|
2014-08-21 15:59:09 +00:00
|
|
|
QVector<EventList *>::iterator evec_end = evec.end();
|
|
|
|
QVector<EventList *>::iterator ni;
|
|
|
|
|
|
|
|
for (ni = evec.begin(); ni != evec_end; ++ni) {
|
|
|
|
num_points += (*ni)->count();
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
total_points += num_points;
|
|
|
|
codepoints += num_points;
|
2014-05-07 19:52:59 +00:00
|
|
|
|
|
|
|
// Max number of samples taken from samples per pixel for better min/max values
|
|
|
|
const int num_averages = 20;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-08-21 15:59:09 +00:00
|
|
|
int n=0;
|
|
|
|
for (ni = evec.begin(); ni != evec_end; ++ni, ++n) {
|
|
|
|
EventList & el = *(EventList*) (*ni);
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
accel = (el.type() == EVL_Waveform); // Turn on acceleration if this is a waveform.
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
if (accel) {
|
2014-04-17 05:55:38 +00:00
|
|
|
sr = el.rate(); // Time distance between samples
|
|
|
|
|
|
|
|
if (sr <= 0) {
|
2011-12-27 13:21:10 +00:00
|
|
|
qWarning() << "qLineChart::Plot() assert(sr>0)";
|
|
|
|
continue;
|
|
|
|
}
|
2011-08-07 13:57:10 +00:00
|
|
|
}
|
2011-07-29 14:58:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (m_disable_accel) { accel = false; }
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
square_plot = m_square_plot;
|
|
|
|
|
|
|
|
if (accel || num_points > 20000) { // Don't square plot if too many points or waveform
|
|
|
|
square_plot = false;
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-08-21 15:59:09 +00:00
|
|
|
int siz = el.count();
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
if (siz <= 1) { continue; } // Don't bother drawing 1 point or less.
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
x0 = el.time(0) + drift;
|
|
|
|
xL = el.time(siz - 1) + drift;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (maxx < x0) { continue; }
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (xL < minx) { continue; }
|
|
|
|
|
|
|
|
if (x0 > xL) {
|
|
|
|
if (siz == 2) { // this happens on CPAP
|
|
|
|
quint32 t = el.getTime()[0];
|
|
|
|
el.getTime()[0] = el.getTime()[1];
|
|
|
|
el.getTime()[1] = t;
|
|
|
|
EventStoreType d = el.getData()[0];
|
|
|
|
el.getData()[0] = el.getData()[1];
|
|
|
|
el.getData()[1] = d;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
} else {
|
|
|
|
qDebug() << "Reversed order sample fed to gLineChart - ignored.";
|
|
|
|
continue;
|
|
|
|
//assert(x1<x2);
|
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
if (accel) {
|
|
|
|
//x1=el.time(1);
|
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
double XR = xx / sr;
|
|
|
|
double Z1 = MAX(x0, minx);
|
|
|
|
double Z2 = MIN(xL, maxx);
|
|
|
|
double ZD = Z2 - Z1;
|
|
|
|
double ZR = ZD / sr;
|
|
|
|
double ZQ = ZR / XR;
|
|
|
|
double ZW = ZR / (width * ZQ);
|
|
|
|
visible_points += ZR * ZQ;
|
|
|
|
|
|
|
|
if (accel && n > 0) {
|
|
|
|
sam = 1;
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
if (ZW < num_averages) {
|
|
|
|
sam = 1;
|
|
|
|
accel = false;
|
2011-12-27 13:21:10 +00:00
|
|
|
} else {
|
2014-04-17 05:55:38 +00:00
|
|
|
sam = ZW / num_averages;
|
|
|
|
|
|
|
|
if (sam < 1) {
|
|
|
|
sam = 1;
|
|
|
|
accel = false;
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
// Prepare the min max y values if we still are accelerating this plot
|
|
|
|
if (accel) {
|
2014-04-17 05:55:38 +00:00
|
|
|
for (int i = 0; i < width; i++) {
|
2011-12-27 13:21:10 +00:00
|
|
|
m_drawlist[i].setX(height);
|
|
|
|
m_drawlist[i].setY(0);
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
minz = width;
|
|
|
|
maxz = 0;
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
total_visible += visible_points;
|
2011-12-27 13:21:10 +00:00
|
|
|
} else {
|
2014-04-17 05:55:38 +00:00
|
|
|
sam = 1;
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
// these calculations over estimate
|
|
|
|
// The Z? values are much more accurate
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
idx = 0;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (el.type() == EVL_Waveform) {
|
2011-12-27 13:21:10 +00:00
|
|
|
// We can skip data previous to minx if this is a waveform
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (minx > x0) {
|
|
|
|
double j = minx - x0; // == starting min of first sample in this segment
|
|
|
|
idx = (j / sr);
|
2011-12-27 13:21:10 +00:00
|
|
|
//idx/=(sam*num_averages);
|
|
|
|
//idx*=(sam*num_averages);
|
|
|
|
// Loose the precision
|
2014-04-17 05:55:38 +00:00
|
|
|
idx += sam - (idx % sam);
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
} // else just start from the beginning
|
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
int xst = left + 1;
|
|
|
|
int yst = top + height + 1;
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
double time;
|
|
|
|
EventDataType data;
|
2014-04-17 05:55:38 +00:00
|
|
|
EventDataType gain = el.gain();
|
2011-07-27 09:53:29 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
done = false;
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (el.type() == EVL_Waveform) { // Waveform Plot
|
|
|
|
if (idx > sam) { idx -= sam; }
|
|
|
|
|
|
|
|
time = el.time(idx) + drift;
|
|
|
|
double rate = double(sr) * double(sam);
|
|
|
|
EventStoreType *ptr = el.rawData() + idx;
|
2014-08-21 15:59:09 +00:00
|
|
|
if ((unsigned) siz > el.count())
|
2014-08-21 14:16:23 +00:00
|
|
|
siz = el.count();
|
2011-12-27 13:21:10 +00:00
|
|
|
|
|
|
|
if (accel) {
|
2012-01-05 11:35:23 +00:00
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
// Accelerated Waveform Plot
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
2014-08-21 14:16:23 +00:00
|
|
|
for (int i = idx; i <= siz; i += sam, ptr += sam) {
|
2014-04-17 05:55:38 +00:00
|
|
|
time += rate;
|
2012-01-05 11:35:23 +00:00
|
|
|
// This is much faster than QVector access.
|
2014-04-17 05:55:38 +00:00
|
|
|
data = *ptr + el.offset();
|
2012-01-05 11:35:23 +00:00
|
|
|
data *= gain;
|
|
|
|
|
|
|
|
// Scale the time scale X to pixel scale X
|
2014-04-17 05:55:38 +00:00
|
|
|
px = ((time - minx) * xmult);
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2012-01-05 11:35:23 +00:00
|
|
|
// Same for Y scale, with gain factored in nmult
|
2014-04-17 05:55:38 +00:00
|
|
|
py = ((data - miny) * ymult);
|
2011-12-27 13:21:10 +00:00
|
|
|
|
|
|
|
// In accel mode, each pixel has a min/max Y value.
|
|
|
|
// m_drawlist's index is the pixel index for the X pixel axis.
|
2014-04-17 05:55:38 +00:00
|
|
|
int z = round(px); // Hmmm... round may screw this up.
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (z < minz) {
|
|
|
|
minz = z; // minz=First pixel
|
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (z > maxz) {
|
|
|
|
maxz = z; // maxz=Last pixel
|
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (minz < 0) {
|
2011-12-27 13:21:10 +00:00
|
|
|
qDebug() << "gLineChart::Plot() minz<0 should never happen!! minz =" << minz;
|
2014-04-17 05:55:38 +00:00
|
|
|
minz = 0;
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
if (maxz > max_drawlist_size) {
|
|
|
|
qDebug() << "gLineChart::Plot() maxz>max_drawlist_size!!!! maxz = " << maxz <<
|
|
|
|
" max_drawlist_size =" << max_drawlist_size;
|
|
|
|
maxz = max_drawlist_size;
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Update the Y pixel bounds.
|
2014-04-17 05:55:38 +00:00
|
|
|
if (py < m_drawlist[z].x()) {
|
2012-01-05 11:35:23 +00:00
|
|
|
m_drawlist[z].setX(py);
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (py > m_drawlist[z].y()) {
|
2012-01-05 11:35:23 +00:00
|
|
|
m_drawlist[z].setY(py);
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (time > maxx) {
|
|
|
|
done = true;
|
2011-12-27 13:21:10 +00:00
|
|
|
break;
|
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2011-08-01 04:34:55 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
// Plot compressed accelerated vertex list
|
2014-04-17 05:55:38 +00:00
|
|
|
if (maxz > width) {
|
|
|
|
maxz = width;
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
float ax1, ay1;
|
|
|
|
QPoint *drl = m_drawlist + minz;
|
2012-01-05 11:35:23 +00:00
|
|
|
// Don't need to cap VertexBuffer here, as it's limited to max_drawlist_size anyway
|
|
|
|
|
|
|
|
|
|
|
|
// Cap within VertexBuffer capacity, one vertex per line point
|
2014-05-07 19:52:59 +00:00
|
|
|
// int np = (maxz - minz) * 2;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-05-07 19:52:59 +00:00
|
|
|
for (int i = minz; i < maxz; i++, drl++) {
|
|
|
|
ax1 = drl->x();
|
|
|
|
ay1 = drl->y();
|
2014-05-08 02:05:01 +00:00
|
|
|
lines.append(QLine(xst + i, yst - ax1, xst + i, yst - ay1));
|
2011-08-01 04:34:55 +00:00
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
} else { // Zoomed in Waveform
|
2012-01-05 11:35:23 +00:00
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
// Normal Waveform Plot
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
// Prime first point
|
2014-04-17 05:55:38 +00:00
|
|
|
data = (*ptr + el.offset()) * gain;
|
|
|
|
lastpx = xst + ((time - minx) * xmult);
|
|
|
|
lastpy = yst - ((data - miny) * ymult);
|
2014-05-14 05:58:47 +00:00
|
|
|
siz--;
|
2014-04-17 05:55:38 +00:00
|
|
|
for (int i = idx; i < siz; i += sam) {
|
|
|
|
ptr += sam;
|
|
|
|
time += rate;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
data = (*ptr + el.offset()) * gain;
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
px = xst + ((time - minx) * xmult); // Scale the time scale X to pixel scale X
|
|
|
|
py = yst - ((data - miny) * ymult); // Same for Y scale, with precomputed gain
|
2012-01-05 11:35:23 +00:00
|
|
|
//py=yst-((data - ymin) * nmult); // Same for Y scale, with precomputed gain
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-05-08 02:05:01 +00:00
|
|
|
lines.append(QLine(lastpx, lastpy, px, py));
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
lastpx = px;
|
|
|
|
lastpy = py;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-08-21 14:16:23 +00:00
|
|
|
if (time >= maxx) {
|
2014-04-17 05:55:38 +00:00
|
|
|
done = true;
|
2011-12-27 13:21:10 +00:00
|
|
|
break;
|
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-09 22:43:21 +00:00
|
|
|
painter.setPen(QPen(chan.defaultColor(), p_profile->appearance->lineThickness()));
|
2014-05-31 21:25:07 +00:00
|
|
|
painter.drawLines(lines);
|
2014-08-06 14:59:40 +00:00
|
|
|
w.graphView()->lines_drawn_this_frame += lines.count();
|
2014-05-31 21:25:07 +00:00
|
|
|
lines.clear();
|
|
|
|
|
|
|
|
|
2012-01-05 11:35:23 +00:00
|
|
|
} else {
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
// Standard events/zoomed in Plot
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
double start = el.first() + drift;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
quint32 *tptr = el.rawTime();
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
int idx = 0;
|
|
|
|
|
|
|
|
if (siz > 15) {
|
2014-08-21 14:16:23 +00:00
|
|
|
// Prime a bit...
|
2014-04-17 05:55:38 +00:00
|
|
|
for (; idx < siz; ++idx) {
|
|
|
|
time = start + *tptr++;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
if (time >= minx) {
|
|
|
|
break;
|
|
|
|
}
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
if (idx > 0) {
|
|
|
|
idx--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Step one backwards if possible (to draw through the left margin)
|
2014-04-17 05:55:38 +00:00
|
|
|
EventStoreType *dptr = el.rawData() + idx;
|
|
|
|
tptr = el.rawTime() + idx;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
time = start + *tptr++;
|
|
|
|
data = (*dptr++ + el.offset()) * gain;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
idx++;
|
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
lastpx = xst + ((time - minx) * xmult); // Scale the time scale X to pixel scale X
|
|
|
|
lastpy = yst - ((data - miny) * ymult); // Same for Y scale without precomputed gain
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
siz -= idx;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
int gs = siz << 1;
|
|
|
|
|
|
|
|
if (square_plot) {
|
2012-01-05 11:35:23 +00:00
|
|
|
gs <<= 1;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
|
|
|
|
2012-01-05 11:35:23 +00:00
|
|
|
// Unrolling square plot outside of loop to gain a minor speed improvement.
|
2014-04-17 05:55:38 +00:00
|
|
|
EventStoreType *eptr = dptr + siz;
|
|
|
|
|
2012-01-05 11:35:23 +00:00
|
|
|
if (square_plot) {
|
2012-01-09 17:06:03 +00:00
|
|
|
for (; dptr < eptr; dptr++) {
|
2014-04-17 05:55:38 +00:00
|
|
|
time = start + *tptr++;
|
|
|
|
data = gain * (*dptr + el.offset());
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
px = xst + ((time - minx) * xmult); // Scale the time scale X to pixel scale X
|
|
|
|
py = yst - ((data - miny) * ymult); // Same for Y scale without precomputed gain
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
// Horizontal lines are easy to cap
|
2014-04-17 05:55:38 +00:00
|
|
|
if (py == lastpy) {
|
2012-01-05 11:35:23 +00:00
|
|
|
// Cap px to left margin
|
2014-04-17 05:55:38 +00:00
|
|
|
if (lastpx < xst) { lastpx = xst; }
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
// Cap px to right margin
|
2014-04-17 05:55:38 +00:00
|
|
|
if (px > xst + width) { px = xst + width; }
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-05-14 05:58:47 +00:00
|
|
|
// lines.append(QLine(lastpx, lastpy, px, lastpy));
|
|
|
|
// lines.append(QLine(px, lastpy, px, py));
|
|
|
|
} // else {
|
2012-01-05 11:35:23 +00:00
|
|
|
// Letting the scissor do the dirty work for non horizontal lines
|
|
|
|
// This really should be changed, as it might be cause that weird
|
|
|
|
// display glitch on Linux..
|
2014-05-07 19:52:59 +00:00
|
|
|
|
2014-05-08 02:05:01 +00:00
|
|
|
lines.append(QLine(lastpx, lastpy, px, lastpy));
|
|
|
|
lines.append(QLine(px, lastpy, px, py));
|
2014-05-14 05:58:47 +00:00
|
|
|
// }
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
lastpx = px;
|
|
|
|
lastpy = py;
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2012-01-05 11:35:23 +00:00
|
|
|
if (time > maxx) {
|
2014-04-17 05:55:38 +00:00
|
|
|
done = true; // Let this iteration finish.. (This point will be in far clipping)
|
2011-12-27 13:21:10 +00:00
|
|
|
break;
|
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
} else {
|
2012-01-09 17:06:03 +00:00
|
|
|
for (; dptr < eptr; dptr++) {
|
2014-04-17 05:55:38 +00:00
|
|
|
//for (int i=0;i<siz;i++) {
|
|
|
|
time = start + *tptr++;
|
|
|
|
data = gain * (*dptr + el.offset());
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
px = xst + ((time - minx) * xmult); // Scale the time scale X to pixel scale X
|
|
|
|
py = yst - ((data - miny) * ymult); // Same for Y scale without precomputed gain
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
// Horizontal lines are easy to cap
|
2014-04-17 05:55:38 +00:00
|
|
|
if (py == lastpy) {
|
2012-01-05 11:35:23 +00:00
|
|
|
// Cap px to left margin
|
2014-04-17 05:55:38 +00:00
|
|
|
if (lastpx < xst) { lastpx = xst; }
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
// Cap px to right margin
|
2014-04-17 05:55:38 +00:00
|
|
|
if (px > xst + width) { px = xst + width; }
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-05-14 05:58:47 +00:00
|
|
|
// lines.append(QLine(lastpx, lastpy, px, py));
|
|
|
|
} //else {
|
2012-01-05 11:35:23 +00:00
|
|
|
// Letting the scissor do the dirty work for non horizontal lines
|
|
|
|
// This really should be changed, as it might be cause that weird
|
|
|
|
// display glitch on Linux..
|
2014-05-08 02:05:01 +00:00
|
|
|
lines.append(QLine(lastpx, lastpy, px, py));
|
2014-05-14 05:58:47 +00:00
|
|
|
//}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
lastpx = px;
|
|
|
|
lastpy = py;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
if (time > maxx) { // Past right edge, abort further drawing..
|
2014-04-17 05:55:38 +00:00
|
|
|
done = true;
|
2012-01-05 11:35:23 +00:00
|
|
|
break;
|
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-08-06 14:59:40 +00:00
|
|
|
painter.setPen(QPen(chan.defaultColor(),p_profile->appearance->lineThickness()));
|
2014-05-14 05:58:47 +00:00
|
|
|
painter.drawLines(lines);
|
|
|
|
w.graphView()->lines_drawn_this_frame+=lines.count();
|
|
|
|
lines.clear();
|
|
|
|
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2011-07-27 09:53:29 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (done) { break; }
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
|
|
|
}
|
2014-05-14 05:58:47 +00:00
|
|
|
|
2014-08-05 11:17:03 +00:00
|
|
|
|
2014-05-31 21:25:07 +00:00
|
|
|
// painter.setPen(QPen(m_colors[gi],p_profile->appearance->lineThickness()));
|
|
|
|
// painter.drawLines(lines);
|
|
|
|
// w.graphView()->lines_drawn_this_frame+=lines.count();
|
|
|
|
// lines.clear();
|
2012-01-05 11:35:23 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Draw Legends on the top line
|
|
|
|
////////////////////////////////////////////////////////////////////
|
2013-10-19 08:00:52 +00:00
|
|
|
|
2014-05-07 19:52:59 +00:00
|
|
|
if ((codepoints > 0)) {
|
2014-04-17 05:55:38 +00:00
|
|
|
QString text = schema::channel[code].label();
|
2014-08-17 12:56:05 +00:00
|
|
|
QRectF rec(0, rect.top()-3, 0,0);
|
|
|
|
rec = painter.boundingRect(rec, Qt::AlignBottom | Qt::AlignLeft, text);
|
|
|
|
rec.moveRight(legendx);
|
|
|
|
legendx -= rec.width();
|
2014-05-07 19:52:59 +00:00
|
|
|
painter.setClipping(false);
|
2014-08-17 12:56:05 +00:00
|
|
|
painter.setPen(Qt::black);
|
|
|
|
painter.drawText(rec, Qt::AlignBottom | Qt::AlignRight, text);
|
|
|
|
|
|
|
|
painter.setPen(QPen(chan.defaultColor(), 1 * ratioY));
|
|
|
|
int linewidth = (10 * ratioX);
|
|
|
|
int yp = rec.top()+(rec.height()/2);
|
|
|
|
painter.drawLine(rec.left()-linewidth, yp , rec.left()-(2 * ratioX), yp);
|
|
|
|
|
2014-05-07 19:52:59 +00:00
|
|
|
painter.setClipping(true);
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
legendx -= linewidth + (2*ratioX);
|
2011-12-27 13:21:10 +00:00
|
|
|
}
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-08-17 12:56:05 +00:00
|
|
|
painter.setClipping(false);
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Draw Channel Threshold legend markers
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
for (int i=0; i < dotlinesize; i++) {
|
|
|
|
DottedLine & dot = m_dotlines[i];
|
|
|
|
if (!dot.visible) continue;
|
|
|
|
ChannelID code = dot.code;
|
|
|
|
schema::Channel &chan = schema::channel[code];
|
|
|
|
int linewidth = (10 * ratioX);
|
|
|
|
QRectF rec(0, rect.top()-3, 0,0);
|
|
|
|
|
|
|
|
QString text = chan.calc[dot.type].label();
|
|
|
|
rec = painter.boundingRect(rec, Qt::AlignBottom | Qt::AlignLeft, text);
|
|
|
|
rec.moveRight(legendx);
|
|
|
|
legendx -= rec.width();
|
|
|
|
painter.setPen(Qt::black);
|
|
|
|
painter.drawText(rec, Qt::AlignBottom | Qt::AlignRight, text);
|
|
|
|
|
|
|
|
QColor color = chan.calc[dot.type].color;
|
|
|
|
color.setAlpha(200);
|
|
|
|
painter.setPen(QPen(QBrush(color),1 * ratioY,Qt::DotLine));
|
|
|
|
|
|
|
|
int yp = rec.top()+(rec.height()/2);
|
|
|
|
painter.drawLine(rec.left()-linewidth, yp , rec.left()-(2 * ratioX), yp);
|
|
|
|
legendx -= linewidth + (2*ratioX);
|
|
|
|
|
|
|
|
|
|
|
|
}
|
2014-08-05 11:17:03 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
painter.setClipping(true);
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-07-27 09:21:53 +00:00
|
|
|
if (!total_points) { // No Data?
|
2014-08-17 12:56:05 +00:00
|
|
|
QString msg = QObject::tr("Plots Disabled");
|
2014-04-17 05:55:38 +00:00
|
|
|
int x, y;
|
|
|
|
GetTextExtent(msg, x, y, bigfont);
|
2014-08-17 12:56:05 +00:00
|
|
|
w.renderText(msg, rect, Qt::AlignCenter, 0, Qt::gray, bigfont);
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-08-11 01:20:09 +00:00
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
painter.setClipping(false);
|
2014-08-11 01:20:09 +00:00
|
|
|
|
|
|
|
// Calculate combined session times within selected area...
|
|
|
|
double first, last;
|
|
|
|
double time = 0;
|
|
|
|
|
2014-08-27 09:55:31 +00:00
|
|
|
// Calculate the CPAP session time.
|
2014-08-11 01:20:09 +00:00
|
|
|
for (QList<Session *>::iterator s = m_day->begin(); s != m_day->end(); s++) {
|
2014-08-27 09:55:31 +00:00
|
|
|
Session * sess = *s;
|
|
|
|
if (!sess->enabled() || (sess->machine()->type() != MT_CPAP)) continue;
|
2014-08-11 01:20:09 +00:00
|
|
|
|
2014-08-27 09:55:31 +00:00
|
|
|
first = sess->first();
|
|
|
|
last = sess->last();
|
2014-08-11 01:20:09 +00:00
|
|
|
|
|
|
|
if (last < w.min_x) { continue; }
|
|
|
|
if (first > w.max_x) { continue; }
|
|
|
|
|
|
|
|
if (first < w.min_x) {
|
|
|
|
first = w.min_x;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (last > w.max_x) {
|
|
|
|
last = w.max_x;
|
|
|
|
}
|
|
|
|
|
|
|
|
time += last - first;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
time /= 1000;
|
|
|
|
|
|
|
|
QList<ChannelID> ahilist;
|
|
|
|
ahilist.push_back(CPAP_Hypopnea);
|
|
|
|
ahilist.push_back(CPAP_Obstructive);
|
|
|
|
ahilist.push_back(CPAP_Apnea);
|
|
|
|
ahilist.push_back(CPAP_ClearAirway);
|
|
|
|
|
|
|
|
QList<ChannelID> extras;
|
2014-08-11 19:00:18 +00:00
|
|
|
extras.push_back(CPAP_NRI);
|
|
|
|
extras.push_back(CPAP_UserFlag1);
|
|
|
|
extras.push_back(CPAP_UserFlag2);
|
2014-08-11 01:20:09 +00:00
|
|
|
|
|
|
|
double sum = 0;
|
|
|
|
int cnt = 0;
|
|
|
|
|
|
|
|
// Draw the linechart overlays
|
2014-08-06 01:25:11 +00:00
|
|
|
if (m_day && (p_profile->appearance->lineCursorMode() || (m_codes[0]==CPAP_FlowRate))) {
|
|
|
|
QHash<ChannelID, gLineOverlayBar *>::iterator fit;
|
2014-08-06 02:08:19 +00:00
|
|
|
bool blockhover = false;
|
|
|
|
|
2014-08-06 01:25:11 +00:00
|
|
|
for (fit = flags.begin(); fit != flags.end(); ++fit) {
|
2014-08-11 01:20:09 +00:00
|
|
|
ChannelID code = fit.key();
|
2014-08-23 06:21:50 +00:00
|
|
|
if (code == 4098) {
|
|
|
|
int i=5; Q_UNUSED(i);
|
|
|
|
}
|
|
|
|
if ((!m_flags_enabled[code]) || (!m_day->channelExists(code))) continue;
|
2014-08-06 02:08:19 +00:00
|
|
|
gLineOverlayBar * lob = fit.value();
|
|
|
|
lob->setBlockHover(blockhover);
|
|
|
|
lob->paint(painter, w, region);
|
|
|
|
if (lob->hover()) blockhover = true; // did it render a hover over?
|
2014-08-11 01:20:09 +00:00
|
|
|
|
|
|
|
if (ahilist.contains(code)) {
|
|
|
|
sum += lob->sum();
|
|
|
|
cnt += lob->count();
|
|
|
|
}
|
2014-08-06 01:25:11 +00:00
|
|
|
}
|
|
|
|
}
|
2014-08-20 09:31:14 +00:00
|
|
|
if (m_codes[0] == OXI_SPO2Drop) {
|
|
|
|
}
|
2014-08-11 01:20:09 +00:00
|
|
|
if (m_codes[0] == CPAP_FlowRate) {
|
2014-08-11 18:29:44 +00:00
|
|
|
float hours = time / 3600.0;
|
2014-08-11 01:20:09 +00:00
|
|
|
int h = time / 3600;
|
|
|
|
int m = int(time / 60) % 60;
|
|
|
|
int s = int(time) % 60;
|
|
|
|
|
2014-08-11 18:29:44 +00:00
|
|
|
double f = double(cnt) / hours; // / (sum / 3600.0);
|
2014-08-11 01:20:09 +00:00
|
|
|
QString txt = QObject::tr("Duration %1:%2:%3").arg(h,2,10,QChar('0')).arg(m,2,10,QChar('0')).arg(s,2,10,QChar('0')) + " "+
|
2014-08-11 19:00:18 +00:00
|
|
|
QObject::tr("AHI %1").arg(f,0,'f',2);// +" " +
|
|
|
|
// QObject::tr("Events %1").arg(cnt) + " " +
|
|
|
|
// QObject::tr("Hours %1").arg(hours,0,'f',2);
|
2014-08-17 12:56:05 +00:00
|
|
|
|
|
|
|
if (linecursormode) txt+=lasttext;
|
|
|
|
|
2014-08-11 06:21:05 +00:00
|
|
|
w.renderText(txt,left,top-4);
|
2014-08-11 01:20:09 +00:00
|
|
|
}
|
|
|
|
|
2014-08-17 12:56:05 +00:00
|
|
|
|
2014-05-08 11:17:45 +00:00
|
|
|
painter.setRenderHint(QPainter::Antialiasing, false);
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|