OSCAR-code/sleepyhead/Graphs/gLineChart.cpp

897 lines
27 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* vim: set ts=8 sts=4 et sw=4 tw=99:
*
* 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"
2011-06-26 08:30:44 +00:00
#define EXTRA_ASSERTS 1
gLineChart::gLineChart(ChannelID code, QColor col, 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, col, square_plot);
m_line_color = col;
m_report_empty = false;
2011-06-26 08:30:44 +00:00
}
gLineChart::~gLineChart()
{
}
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->getSessions()[i];
if (sess->channelExists(code)) {
return false;
}
}
}
return true;
}
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->getSessions()[i];
if (!sess->channelExists(code)) { continue; }
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;
}
}
}
}
2011-12-21 12:47:47 +00:00
//if (m_code==CPAP_Leak) {
// subtract_offset=profile.cpap.[IntentionalLeak].toDouble();
//} else
subtract_offset = 0;
}
EventDataType gLineChart::Miny()
{
int m = Layer::Miny();
if (subtract_offset > 0) {
m -= subtract_offset;
if (m < 0) { m = 0; }
}
return m;
}
EventDataType gLineChart::Maxy()
{
return Layer::Maxy() - subtract_offset;
}
2011-07-27 09:21:53 +00:00
2011-06-26 08:30:44 +00:00
// Time Domain Line Chart
2014-05-07 19:52:59 +00:00
void gLineChart::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height)
2011-06-26 08:30:44 +00:00
{
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-06 02:37:29 +00:00
top++;
EventDataType miny, maxy;
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..
/*if (miny<0) {
2011-08-07 11:37:56 +00:00
miny=-MAX(fabs(miny),fabs(maxy));
}*/
2011-07-27 09:21:53 +00:00
if (w.zoomY() == 0 && PROFILE.appearance->allowYAxisScaling()) {
miny = m_physminy, maxy = m_physmaxy;
} else {
miny = w.min_y, maxy = w.max_y;
}
2011-06-26 08:30:44 +00:00
w.roundY(miny, maxy);
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
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;
qint64 clockdrift = qint64(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-07 19:52:59 +00:00
painter.setClipRect(left, top, width, height);
painter.setClipping(true);
for (int gi = 0; gi < m_codes.size(); gi++) {
ChannelID code = m_codes[gi];
2014-05-07 19:52:59 +00:00
painter.setPen(m_colors[gi]);
codepoints = 0;
for (int svi = 0; svi < m_day->size(); 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()->GetType() == MT_CPAP) ? clockdrift : 0;
if (!sess->enabled()) { continue; }
schema::Channel ch = schema::channel[code];
bool fndbetter = false;
for (QList<schema::Channel *>::iterator l = ch.m_links.begin(); l != ch.m_links.end(); 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;
for (int i = 0; i < evec.size(); 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
for (int n = 0; n < evec.size(); 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;
if (accel) {
2012-01-05 11:35:23 +00:00
//////////////////////////////////////////////////////////////////
// Accelerated Waveform Plot
//////////////////////////////////////////////////////////////////
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();
painter.drawLine(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);
2012-01-05 11:35:23 +00:00
for (int i = idx; i < siz; i += sam) {
ptr += sam;
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-07 19:52:59 +00:00
painter.drawLine(lastpx, lastpy, px, py);
lastpx = px;
lastpy = py;
2012-01-05 11:35:23 +00:00
if (time > maxx) {
done = true;
break;
}
2011-07-27 09:21:53 +00:00
}
}
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) {
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--;
//tptr--;
}
}
// 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
2014-05-07 19:52:59 +00:00
painter.drawLine(lastpx, lastpy, px, lastpy);
painter.drawLine(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
painter.drawLine(lastpx, lastpy, px, lastpy);
painter.drawLine(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
2014-05-07 19:52:59 +00:00
painter.drawLine(lastpx, lastpy, px, py);
2012-01-05 11:35:23 +00:00
} else {
// 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
painter.drawLine(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
}
}
if (done) { break; }
2011-06-26 08:30:44 +00:00
}
}
2012-01-05 11:35:23 +00:00
////////////////////////////////////////////////////////////////////
// Draw Legends on the top line
////////////////////////////////////////////////////////////////////
2013-10-19 08:00:52 +00:00
QFontMetrics fm(*defaultfont);
int bw = fm.width('X');
int bh = fm.height() / 1.8;
2014-05-07 19:52:59 +00:00
if ((codepoints > 0)) {
QString text = schema::channel[code].label();
int wid, hi;
GetTextExtent(text, wid, hi);
legendx -= wid;
2014-05-07 19:52:59 +00:00
painter.setClipping(false);
w.renderText(text, legendx, top - 4);
2014-05-07 19:52:59 +00:00
legendx -= bw /2;
painter.fillRect(legendx - bw, top - w.marginTop()-2, bh, w.marginTop()+1, QBrush(m_colors[gi]));
painter.setClipping(true);
2014-05-07 19:52:59 +00:00
legendx -= bw*2;
}
2011-06-26 08:30:44 +00:00
}
2011-07-27 09:21:53 +00:00
if (!total_points) { // No Data?
2011-06-26 08:30:44 +00:00
2011-07-27 09:21:53 +00:00
if (m_report_empty) {
QString msg = "No Waveform Available";
int x, y;
GetTextExtent(msg, x, y, bigfont);
//DrawText(w,msg,left+(width/2.0)-(x/2.0),scry-w.GetBottomMargin()-height/2.0+y/2.0,0,Qt::gray,bigfont);
2011-07-27 09:21:53 +00:00
}
2011-06-26 08:30:44 +00:00
}
2014-05-07 19:52:59 +00:00
painter.setClipping(false);
2011-06-26 08:30:44 +00:00
}
AHIChart::AHIChart(QColor col)
: Layer(NoChannel), m_color(col)
{
m_miny = m_maxy = 0;
}
AHIChart::~AHIChart()
{
}
2014-05-07 19:52:59 +00:00
void AHIChart::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height)
{
if (!m_visible) {
return;
}
if (!m_day) {
return;
}
// 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);
width--;
height -= 2;
EventDataType miny, maxy;
double minx, maxx;
maxx = w.max_x, minx = w.min_x;
// hmmm.. subtract_offset..
if (w.zoomY() == 0 && PROFILE.appearance->allowYAxisScaling()) {
miny = w.physMinY();
maxy = w.physMaxY();
} else {
miny = w.min_y, maxy = w.max_y;
}
w.roundY(miny, maxy);
double xx = maxx - minx;
double xmult = double(width) / xx;
EventDataType yy = maxy - miny;
EventDataType ymult = EventDataType(height - 3) / yy; // time to pixel conversion multiplier
bool first = true;
double px, py;
double lastpx, lastpy;
double top1 = top + height;
bool done = false;
2014-05-07 19:52:59 +00:00
painter.setPen(QPen(m_color,1.5));
painter.setClipRect(left, top, width, height);
painter.setClipping(true);
for (int i = 0; i < m_time.size(); i++) {
qint64 ti = m_time[i];
EventDataType v = m_data[i];
if (ti < minx) { continue; }
if (ti > maxx) { done = true; }
if (first) {
if (i > 0) {
ti = m_time[i - 1];
v = m_data[i - 1];
i--;
}
px = left + (double(ti - minx) * xmult);
py = top1 - (double(v - miny) * ymult);
first = false;
} else {
px = left + (double(ti - minx) * xmult);
py = top1 - (double(v - miny) * ymult);
2014-05-07 19:52:59 +00:00
painter.drawLine(px, py, lastpx, lastpy);
}
lastpx = px;
lastpy = py;
if (done) { break; }
}
2014-05-07 19:52:59 +00:00
painter.setClipping(false);
}
void AHIChart::SetDay(Day *d)
{
m_day = d;
m_data.clear();
m_time.clear();
m_maxy = 0;
m_miny = 0;
if (!d) { return; }
m_miny = 9999;
QList<Session *>::iterator s;
qint64 first = d->first();
qint64 last = d->last();
qint64 f;
qint64 winsize = 30000; // 30 second windows
for (qint64 ti = first; ti < last; ti += winsize) {
f = ti - 3600000L;
//if (f<first) f=first;
EventList *el[6];
EventDataType ahi = 0;
int cnt = 0;
qint64 clockdrift = (qint64(PROFILE.cpap->clockDrift()) * 1000L), drift = 0;
bool fnd = false;
for (s = d->begin(); s != d->end(); s++) {
if (!(*s)->enabled()) { continue; }
Session *sess = *s;
if ((ti < sess->first()) || (f > sess->last())) { continue; }
drift = (sess->machine()->GetType() == MT_CPAP) ? clockdrift : 0;
// Drop off suddenly outside of sessions
//if (ti>sess->last()) continue;
fnd = true;
if (sess->eventlist.contains(CPAP_Obstructive)) {
el[0] = sess->eventlist[CPAP_Obstructive][0];
} else { el[0] = nullptr; }
if (sess->eventlist.contains(CPAP_Apnea)) {
el[1] = sess->eventlist[CPAP_Apnea][0];
} else { el[1] = nullptr; }
if (sess->eventlist.contains(CPAP_Hypopnea)) {
el[2] = sess->eventlist[CPAP_Hypopnea][0];
} else { el[2] = nullptr; }
if (sess->eventlist.contains(CPAP_ClearAirway)) {
el[3] = sess->eventlist[CPAP_ClearAirway][0];
} else { el[3] = nullptr; }
if (sess->eventlist.contains(CPAP_NRI)) {
el[4] = sess->eventlist[CPAP_NRI][0];
} else { el[4] = nullptr; }
int znt = 5;
if (PROFILE.general->calculateRDI()) {
if (sess->eventlist.contains(CPAP_RERA)) {// What about ExP??
el[5] = sess->eventlist[CPAP_RERA][0];
znt++;
} else { el[5] = nullptr; }
}
qint64 t;
for (int i = 0; i < znt; i++) {
if (!el[i]) { continue; }
for (quint32 j = 0; j < el[i]->count(); j++) {
t = el[i]->time(j) + drift;
if ((t >= f) && (t < ti)) {
cnt++;
}
}
}
}
if (!fnd) { cnt = 0; }
double g = double(ti - f) / 3600000.0;
if (g > 0) { ahi = cnt / g; }
if (ahi < m_miny) { m_miny = ahi; }
if (ahi > m_maxy) { m_maxy = ahi; }
m_time.append(ti);
m_data.append(ahi);
}
m_minx = first;
m_maxx = last;
}