OSCAR-code/sleepyhead/Graphs/gLineChart.cpp

1150 lines
37 KiB
C++
Raw Normal View History

2014-08-17 12:56:05 +00:00
/* gLineChart Implementation
*
* 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
#include "Graphs/gLineChart.h"
2011-06-26 08:30:44 +00:00
#include <QString>
2011-07-01 10:10:44 +00:00
#include <QDebug>
#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
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());
}
gLineChart::gLineChart(ChannelID code, bool square_plot, bool disable_accel)
: Layer(code), m_square_plot(square_plot), m_disable_accel(disable_accel)
2011-06-26 08:30:44 +00:00
{
addPlot(code, square_plot);
m_report_empty = false;
lines.reserve(50000);
lasttime = 0;
2014-08-17 12:56:05 +00:00
m_layertype = LT_LineChart;
2011-06-26 08:30:44 +00:00
}
gLineChart::~gLineChart()
{
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
}
bool gLineChart::isEmpty()
{
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++) {
Session *sess = m_day->sessions[i];
if (sess->channelExists(code)) {
return false;
}
}
}
return true;
}
2014-08-17 12:56:05 +00:00
void gLineChart::SetDay(Day *d)
{
// Layer::SetDay(d);
m_day = d;
m_minx = 0, m_maxx = 0;
m_miny = 0, m_maxy = 0;
m_physminy = 0, m_physmaxy = 0;
if (!d) {
return;
}
qint64 t64;
EventDataType tmp;
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++) {
Session *sess = d->sessions[i];
if (!sess->enabled()) continue;
if (code == CPAP_MaskPressure) {
if (sess->channelExists(CPAP_MaskPressureHi)) {
code = m_codes[j] = CPAP_MaskPressureHi;
m_enabled[code] = schema::channel[CPAP_MaskPressureHi].enabled();
goto skipcheck; // why not :P
}
}
if (!sess->channelExists(code)) {
continue;
}
skipcheck:
if (first) {
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;
} else {
tmp = sess->physMin(code);
if (m_physminy > tmp) {
m_physminy = tmp;
}
tmp = sess->physMax(code);
if (m_physmaxy < tmp) {
m_physmaxy = tmp;
}
tmp = sess->Min(code);
if (m_miny > tmp) {
m_miny = tmp;
}
tmp = sess->Max(code);
if (m_maxy < tmp) {
m_maxy = tmp;
}
t64 = sess->first(code);
if (m_minx > t64) {
m_minx = t64;
}
t64 = sess->last(code);
if (m_maxx < t64) {
m_maxx = t64;
}
}
}
}
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();
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-11 19:00:18 +00:00
if (!m_day->channelExists(code)) continue;
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-11 18:29:44 +00:00
lob->setOverlayDisplayType(((m_codes[0] == CPAP_FlowRate) || (m_codes[0] == CPAP_MaskPressureHi))? (OverlayDisplayType)p_profile->appearance->overlayType() : ODT_TopAndBottom);
lob->SetDay(m_day);
flags[code] = lob;
2014-08-05 11:17:03 +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));
}
if (m_day) {
for (int i=0; i < m_dotlines.size(); i++) {
m_dotlines[i].calc(m_day);
}
}
}
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-08-17 12:56:05 +00:00
for (int i=0; i< size; ++i) {
ChannelID code = m_codes[i];
if (!m_enabled[code]) continue;
2014-08-17 12:56:05 +00:00
tmp = m_day->Min(code);
if (!first) {
min = tmp;
first = true;
2014-08-17 12:56:05 +00:00
} else {
min = qMin(tmp, min);
}
}
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-08-17 12:56:05 +00:00
// return m;
}
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];
if (!m_enabled[code]) continue;
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-07-27 09:21:53 +00:00
bool gLineChart::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
{
Q_UNUSED(event)
2014-07-21 09:39:26 +00:00
Q_UNUSED(graph)
graph->timedRedraw(0);
return false;
}
QString gLineChart::getMetaString(qint64 time)
{
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];
if (m_day->channelHasData(code)) {
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());
if (code == CPAP_IPAP) {
ipap = val;
addPS = true;
}
if (code == CPAP_EPAP) {
epap = val;
}
}
}
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());
}
lasttime = time;
return lasttext;
}
2011-06-26 08:30:44 +00:00
// Time Domain Line Chart
void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion &region)
2011-06-26 08:30:44 +00:00
{
2014-08-05 11:17:03 +00:00
QRect rect = region.boundingRect();
// 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();
if (!m_visible) {
2011-06-26 08:30:44 +00:00
return;
}
if (!m_day) {
2011-06-26 08:30:44 +00:00
return;
}
//if (!m_day->channelExists(m_code)) return;
2011-09-17 12:39:00 +00:00
if (width < 0) {
return;
}
2014-05-13 21:44:00 +00:00
2014-05-06 02:37:29 +00:00
top++;
double minx, maxx;
if (w.blockZoom()) {
minx = w.rmin_x, maxx = w.rmax_x;
} else {
maxx = w.max_x, minx = w.min_x;
}
// hmmm.. subtract_offset..
EventDataType miny = m_physminy;
EventDataType maxy = m_physmaxy;
2011-07-27 09:21:53 +00:00
w.roundY(miny, maxy);
2014-05-20 12:07:32 +00:00
//#define DEBUG_AUTOSCALER
#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();
double xx = maxx - minx;
double xmult = double(width) / xx;
2011-07-30 06:03:08 +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
if (xx < 0) {
2011-06-26 08:30:44 +00:00
return;
}
if (yy <= 0) {
if (miny == 0) {
return;
}
}
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-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
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));
painter.drawLine(left+xpos, top-w.marginTop()-3, left+xpos, top+height+w.bottom-1);
}
if ((time != lasttime) || lasttext.isEmpty()) {
getMetaString(time);
2014-07-21 13:47:17 +00:00
}
2014-08-17 12:56:05 +00:00
if (m_codes[0] != CPAP_FlowRate) {
QString text = lasttext;
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
}
}
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;
double x0, xL;
2011-07-27 09:21:53 +00:00
double sr;
int sam;
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--;
height -= 2;
2011-06-26 08:30:44 +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;
qint64 drift = 0;
2011-06-26 08:30:44 +00:00
QHash<ChannelID, QVector<EventList *> >::iterator ci;
//m_line_color=schema::channel[m_code].defaultColor();
int legendx = left + width;
2011-09-17 13:21:18 +00:00
int codepoints;
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;
}
for (int gi = 0; gi < m_codes.size(); gi++) {
ChannelID code = m_codes[gi];
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];
if ((dot.code != code) || (!dot.enabled) || (!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-08-17 12:56:05 +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();
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++) {
Session *sess = (*m_day)[svi];
2011-12-28 12:03:48 +00:00
if (!sess) {
qWarning() << "gLineChart::Plot() nullptr Session Record.. This should not happen";
continue;
}
drift = (sess->machine()->type() == MT_CPAP) ? clockdrift : 0;
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());
if (ci != (*m_day)[svi]->eventlist.end()) {
fndbetter = true;
break;
}
2011-09-17 13:21:18 +00:00
}
if (!fndbetter) {
ci = (*m_day)[svi]->eventlist.find(code);
if (ci == (*m_day)[svi]->eventlist.end()) { continue; }
}
QVector<EventList *> &evec = ci.value();
num_points = 0;
2014-05-09 07:05:54 +00:00
int evecsize=evec.size();
2014-05-09 07:05:54 +00:00
for (int i = 0; i < evecsize; i++) {
num_points += evec[i]->count();
}
2011-06-26 08:30:44 +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-05-08 11:18:19 +00:00
for (int n = 0; n < evecsize; ++n) { // for each segment
EventList &el = *evec[n];
accel = (el.type() == EVL_Waveform); // Turn on acceleration if this is a waveform.
2011-06-26 08:30:44 +00:00
if (accel) {
sr = el.rate(); // Time distance between samples
if (sr <= 0) {
qWarning() << "qLineChart::Plot() assert(sr>0)";
continue;
}
}
2011-07-29 14:58:44 +00:00
if (m_disable_accel) { accel = false; }
2011-06-26 08:30:44 +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-06-26 08:30:44 +00:00
int siz = evec[n]->count();
if (siz <= 1) { continue; } // Don't bother drawing 1 point or less.
2011-06-26 08:30:44 +00:00
x0 = el.time(0) + drift;
xL = el.time(siz - 1) + drift;
2011-06-26 08:30:44 +00:00
if (maxx < x0) { continue; }
2011-06-26 08:30:44 +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
} else {
qDebug() << "Reversed order sample fed to gLineChart - ignored.";
continue;
//assert(x1<x2);
}
2011-07-27 09:21:53 +00:00
}
if (accel) {
//x1=el.time(1);
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;
}
if (ZW < num_averages) {
sam = 1;
accel = false;
} else {
sam = ZW / num_averages;
if (sam < 1) {
sam = 1;
accel = false;
}
2011-07-27 09:21:53 +00:00
}
// Prepare the min max y values if we still are accelerating this plot
if (accel) {
for (int i = 0; i < width; i++) {
m_drawlist[i].setX(height);
m_drawlist[i].setY(0);
}
minz = width;
maxz = 0;
2011-07-27 09:21:53 +00:00
}
total_visible += visible_points;
} else {
sam = 1;
2011-07-27 09:21:53 +00:00
}
2011-06-26 08:30:44 +00:00
// these calculations over estimate
// The Z? values are much more accurate
2011-06-26 08:30:44 +00:00
idx = 0;
2011-06-26 08:30:44 +00:00
if (el.type() == EVL_Waveform) {
// We can skip data previous to minx if this is a waveform
2011-06-26 08:30:44 +00:00
if (minx > x0) {
double j = minx - x0; // == starting min of first sample in this segment
idx = (j / sr);
//idx/=(sam*num_averages);
//idx*=(sam*num_averages);
// Loose the precision
idx += sam - (idx % sam);
2011-06-26 08:30:44 +00:00
} // else just start from the beginning
}
2011-07-27 09:21:53 +00:00
int xst = left + 1;
int yst = top + height + 1;
2011-07-27 09:21:53 +00:00
double time;
EventDataType data;
EventDataType gain = el.gain();
done = false;
2011-07-27 09:21:53 +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 14:16:23 +00:00
if (siz > el.count())
siz = el.count();
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) {
time += rate;
2012-01-05 11:35:23 +00:00
// This is much faster than QVector access.
data = *ptr + el.offset();
2012-01-05 11:35:23 +00:00
data *= gain;
// Scale the time scale X to pixel scale X
px = ((time - minx) * xmult);
2012-01-05 11:35:23 +00:00
// Same for Y scale, with gain factored in nmult
py = ((data - miny) * ymult);
// In accel mode, each pixel has a min/max Y value.
// m_drawlist's index is the pixel index for the X pixel axis.
int z = round(px); // Hmmm... round may screw this up.
2012-01-05 11:35:23 +00:00
if (z < minz) {
minz = z; // minz=First pixel
}
2012-01-05 11:35:23 +00:00
if (z > maxz) {
maxz = z; // maxz=Last pixel
}
2012-01-05 11:35:23 +00:00
if (minz < 0) {
qDebug() << "gLineChart::Plot() minz<0 should never happen!! minz =" << minz;
minz = 0;
}
if (maxz > max_drawlist_size) {
qDebug() << "gLineChart::Plot() maxz>max_drawlist_size!!!! maxz = " << maxz <<
" max_drawlist_size =" << max_drawlist_size;
maxz = max_drawlist_size;
}
// Update the Y pixel bounds.
if (py < m_drawlist[z].x()) {
2012-01-05 11:35:23 +00:00
m_drawlist[z].setX(py);
}
if (py > m_drawlist[z].y()) {
2012-01-05 11:35:23 +00:00
m_drawlist[z].setY(py);
}
if (time > maxx) {
done = true;
break;
}
2012-01-05 11:35:23 +00:00
}
// Plot compressed accelerated vertex list
if (maxz > width) {
maxz = width;
}
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-07-27 09:21:53 +00:00
} else { // Zoomed in Waveform
2012-01-05 11:35:23 +00:00
//////////////////////////////////////////////////////////////////
// Normal Waveform Plot
//////////////////////////////////////////////////////////////////
// Prime first point
data = (*ptr + el.offset()) * gain;
lastpx = xst + ((time - minx) * xmult);
lastpy = yst - ((data - miny) * ymult);
2014-08-21 14:59:18 +00:00
EventStoreType *eptr = el.rawData() + el.count()-1;
2012-01-05 11:35:23 +00:00
siz--;
for (int i = idx; i < siz; i += sam) {
ptr += sam;
2014-08-21 14:59:18 +00:00
if (ptr >= eptr) break;
time += rate;
2012-01-05 11:35:23 +00:00
data = (*ptr + el.offset()) * gain;
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
2014-05-08 02:05:01 +00:00
lines.append(QLine(lastpx, lastpy, px, py));
lastpx = px;
lastpy = py;
2012-01-05 11:35:23 +00:00
2014-08-21 14:16:23 +00:00
if (time >= maxx) {
done = true;
break;
}
2011-07-27 09:21:53 +00:00
}
}
painter.setPen(QPen(chan.defaultColor(), 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
} else {
//////////////////////////////////////////////////////////////////
// Standard events/zoomed in Plot
//////////////////////////////////////////////////////////////////
double start = el.first() + drift;
2012-01-05 11:35:23 +00:00
quint32 *tptr = el.rawTime();
2012-01-05 11:35:23 +00:00
int idx = 0;
if (siz > 15) {
2014-08-21 14:16:23 +00:00
// Prime a bit...
for (; idx < siz; ++idx) {
time = start + *tptr++;
2012-01-05 11:35:23 +00:00
if (time >= minx) {
break;
}
}
2012-01-05 11:35:23 +00:00
if (idx > 0) {
idx--;
}
}
// Step one backwards if possible (to draw through the left margin)
EventStoreType *dptr = el.rawData() + idx;
tptr = el.rawTime() + idx;
2012-01-05 11:35:23 +00:00
time = start + *tptr++;
data = (*dptr++ + el.offset()) * gain;
2012-01-05 11:35:23 +00:00
idx++;
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
siz -= idx;
2012-01-05 11:35:23 +00:00
int gs = siz << 1;
if (square_plot) {
2012-01-05 11:35:23 +00:00
gs <<= 1;
}
2012-01-05 11:35:23 +00:00
// Unrolling square plot outside of loop to gain a minor speed improvement.
EventStoreType *eptr = dptr + siz;
2012-01-05 11:35:23 +00:00
if (square_plot) {
for (; dptr < eptr; dptr++) {
time = start + *tptr++;
data = gain * (*dptr + el.offset());
2012-01-05 11:35:23 +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
if (py == lastpy) {
2012-01-05 11:35:23 +00:00
// Cap px to left margin
if (lastpx < xst) { lastpx = xst; }
2012-01-05 11:35:23 +00:00
// Cap px to right margin
if (px > xst + width) { px = xst + width; }
2012-01-05 11:35:23 +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));
// }
lastpx = px;
lastpy = py;
2012-01-05 11:35:23 +00:00
if (time > maxx) {
done = true; // Let this iteration finish.. (This point will be in far clipping)
break;
}
2011-07-27 09:21:53 +00:00
}
2012-01-05 11:35:23 +00:00
} else {
for (; dptr < eptr; dptr++) {
//for (int i=0;i<siz;i++) {
time = start + *tptr++;
data = gain * (*dptr + el.offset());
2012-01-05 11:35:23 +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
if (py == lastpy) {
2012-01-05 11:35:23 +00:00
// Cap px to left margin
if (lastpx < xst) { lastpx = xst; }
2012-01-05 11:35:23 +00:00
// Cap px to right margin
if (px > xst + width) { px = xst + width; }
2012-01-05 11:35:23 +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));
//}
2012-01-05 11:35:23 +00:00
lastpx = px;
lastpy = py;
2012-01-05 11:35:23 +00:00
if (time > maxx) { // Past right edge, abort further drawing..
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
}
painter.setPen(QPen(chan.defaultColor(),p_profile->appearance->lineThickness()));
painter.drawLines(lines);
w.graphView()->lines_drawn_this_frame+=lines.count();
lines.clear();
2011-06-26 08:30:44 +00:00
}
if (done) { break; }
2011-06-26 08:30:44 +00:00
}
}
2014-08-05 11:17:03 +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)) {
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-08-17 12:56:05 +00:00
legendx -= linewidth + (2*ratioX);
}
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);
}
// for (int gi = 0; gi < m_codes.size(); gi++) {
// ChannelID code = m_codes[gi];
// schema::Channel &chan = schema::channel[code];
// int linewidth = (10 * ratioX);
// QRectF rec(0, rect.top()-3, 0,0);
// if (chan.upperThreshold() > 0) {
// QString text = m_threshold.at(gi);
// 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.upperThresholdColor();
// 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);
// }
// if (chan.lowerThreshold() > 0) {
// QString text = m_threshold.at(gi);
// 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.lowerThresholdColor();
// color.setAlpha(200);
// painter.setPen(QPen(QBrush(color),1,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);
2011-07-27 09:21:53 +00:00
if (!total_points) { // No Data?
2014-08-17 12:56:05 +00:00
// if (m_report_empty) {
QString msg = QObject::tr("Plots Disabled");
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);
// DrawText(w,msg,left+(width/2.0)-(x/2.0),rect.top()-w.GetBottomMargin()-height/2.0+y/2.0,0,Qt::gray,bigfont);
// }
2011-06-26 08:30:44 +00:00
}
2014-08-17 12:56:05 +00:00
painter.setClipping(false);
// Calculate combined session times within selected area...
double first, last;
double time = 0;
// Calculate the session time.
for (QList<Session *>::iterator s = m_day->begin(); s != m_day->end(); s++) {
if (!(*s)->enabled()) { continue; }
first = (*s)->first();
last = (*s)->last();
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);
double sum = 0;
int cnt = 0;
// Draw the linechart overlays
if (m_day && (p_profile->appearance->lineCursorMode() || (m_codes[0]==CPAP_FlowRate))) {
QHash<ChannelID, gLineOverlayBar *>::iterator fit;
bool blockhover = false;
for (fit = flags.begin(); fit != flags.end(); ++fit) {
ChannelID code = fit.key();
2014-08-11 18:29:44 +00:00
if (!m_day->channelExists(code)) continue;
gLineOverlayBar * lob = fit.value();
lob->setBlockHover(blockhover);
lob->paint(painter, w, region);
if (lob->hover()) blockhover = true; // did it render a hover over?
if (ahilist.contains(code)) {
sum += lob->sum();
cnt += lob->count();
}
}
}
if (m_codes[0] == OXI_SPO2Drop) {
}
if (m_codes[0] == CPAP_FlowRate) {
2014-08-11 18:29:44 +00:00
float hours = time / 3600.0;
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);
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;
w.renderText(txt,left,top-4);
}
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
}