2014-04-09 21:01:57 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
* vim: set ts=8 sts=4 et sw=4 tw=99:
|
|
|
|
*
|
|
|
|
* gFlagsLine 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
|
|
|
|
2011-07-11 04:54:53 +00:00
|
|
|
#include <cmath>
|
2011-07-31 20:24:43 +00:00
|
|
|
#include <QVector>
|
2011-06-26 08:30:44 +00:00
|
|
|
#include "SleepLib/profiles.h"
|
|
|
|
#include "gFlagsLine.h"
|
2011-08-25 06:11:44 +00:00
|
|
|
#include "gYAxis.h"
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
gFlagsLabelArea::gFlagsLabelArea(gFlagsGroup *group)
|
|
|
|
: gSpacer(20)
|
2013-10-25 10:39:30 +00:00
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
m_group = group;
|
2013-10-25 10:39:30 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
bool gFlagsLabelArea::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
|
2013-10-25 10:39:30 +00:00
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
if (m_group) {
|
|
|
|
return m_group->mouseMoveEvent(event, graph);
|
|
|
|
}
|
|
|
|
|
2013-10-25 15:44:14 +00:00
|
|
|
return false;
|
2013-10-25 10:39:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-11 04:54:53 +00:00
|
|
|
gFlagsGroup::gFlagsGroup()
|
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
addVertexBuffer(quads = new gVertexBuffer(512, GL_QUADS));
|
|
|
|
addVertexBuffer(lines = new gVertexBuffer(20, GL_LINE_LOOP));
|
2011-09-28 11:46:32 +00:00
|
|
|
quads->setAntiAlias(true);
|
2011-08-31 11:44:53 +00:00
|
|
|
lines->setAntiAlias(false);
|
2014-04-17 05:55:38 +00:00
|
|
|
m_barh = 0;
|
|
|
|
m_empty = true;
|
2011-07-11 04:54:53 +00:00
|
|
|
}
|
|
|
|
gFlagsGroup::~gFlagsGroup()
|
|
|
|
{
|
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
qint64 gFlagsGroup::Minx()
|
|
|
|
{
|
|
|
|
if (m_day) {
|
|
|
|
return m_day->first();
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-07-27 09:21:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
qint64 gFlagsGroup::Maxx()
|
|
|
|
{
|
|
|
|
if (m_day) {
|
|
|
|
return m_day->last();
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-07-27 09:21:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
void gFlagsGroup::SetDay(Day *d)
|
2011-07-11 04:54:53 +00:00
|
|
|
{
|
2011-09-01 03:37:25 +00:00
|
|
|
LayerGroup::SetDay(d);
|
|
|
|
lvisible.clear();
|
2014-04-17 05:55:38 +00:00
|
|
|
int cnt = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < layers.size(); i++) {
|
|
|
|
gFlagsLine *f = dynamic_cast<gFlagsLine *>(layers[i]);
|
|
|
|
|
|
|
|
if (!f) { continue; }
|
|
|
|
|
|
|
|
bool e = f->isEmpty();
|
2011-07-20 16:01:31 +00:00
|
|
|
|
2011-12-10 12:14:48 +00:00
|
|
|
if (!e || f->isAlwaysVisible()) {
|
2011-07-27 09:21:53 +00:00
|
|
|
lvisible.push_back(f);
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
if (!e) {
|
|
|
|
cnt++;
|
|
|
|
}
|
2011-07-11 04:54:53 +00:00
|
|
|
}
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
m_empty = (cnt == 0);
|
|
|
|
|
2012-01-10 02:34:44 +00:00
|
|
|
if (m_empty) {
|
|
|
|
if (d) {
|
2014-04-17 05:55:38 +00:00
|
|
|
m_empty = !d->channelExists(CPAP_Pressure);
|
2012-01-10 02:34:44 +00:00
|
|
|
}
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
m_barh = 0;
|
2011-09-01 03:37:25 +00:00
|
|
|
}
|
|
|
|
|
2013-10-25 10:39:30 +00:00
|
|
|
void gFlagsGroup::paint(gGraph &g, int left, int top, int width, int height)
|
2011-09-01 03:37:25 +00:00
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
if (!m_visible) { return; }
|
|
|
|
|
|
|
|
if (!m_day) { return; }
|
2011-09-01 03:37:25 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
int vis = lvisible.size();
|
|
|
|
m_barh = float(height) / float(vis);
|
|
|
|
float linetop = top;
|
2011-08-25 06:11:44 +00:00
|
|
|
|
2013-09-10 15:09:09 +00:00
|
|
|
QColor barcol;
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < lvisible.size(); i++) {
|
2011-08-25 06:11:44 +00:00
|
|
|
// Alternating box color
|
2014-04-17 05:55:38 +00:00
|
|
|
if (i & 1) { barcol = COLOR_ALT_BG1; }
|
|
|
|
else { barcol = COLOR_ALT_BG2; }
|
|
|
|
|
|
|
|
quads->add(left, linetop, left, linetop + m_barh, left + width - 1, linetop + m_barh,
|
|
|
|
left + width - 1, linetop, barcol.rgba());
|
2011-08-31 11:44:53 +00:00
|
|
|
|
2011-08-25 06:11:44 +00:00
|
|
|
// Paint the actual flags
|
2014-04-17 05:55:38 +00:00
|
|
|
lvisible[i]->m_rect = QRect(left, linetop, width, m_barh);
|
|
|
|
lvisible[i]->paint(g, left, linetop, width, m_barh);
|
|
|
|
linetop += m_barh;
|
2011-07-11 04:54:53 +00:00
|
|
|
}
|
2011-08-25 06:11:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
gVertexBuffer *outlines = g.lines();
|
|
|
|
outlines->add(left - 1, top, left - 1, top + height, COLOR_Outline.rgba());
|
|
|
|
outlines->add(left - 1, top + height, left + width, top + height, COLOR_Outline.rgba());
|
|
|
|
outlines->add(left + width, top + height, left + width, top, COLOR_Outline.rgba());
|
|
|
|
outlines->add(left + width, top, left - 1, top, COLOR_Outline.rgba());
|
2011-09-01 07:12:25 +00:00
|
|
|
|
|
|
|
//lines->add(left-1, top, left-1, top+height);
|
|
|
|
//lines->add(left+width, top+height, left+width, top);
|
2011-07-11 04:54:53 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
bool gFlagsGroup::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
|
2013-10-25 10:39:30 +00:00
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
if (!p_profile->appearance->graphTooltips()) {
|
2013-10-26 13:59:37 +00:00
|
|
|
return false;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < lvisible.size(); i++) {
|
|
|
|
gFlagsLine *fl = lvisible[i];
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (fl->m_rect.contains(event->x(), event->y())) {
|
|
|
|
if (fl->mouseMoveEvent(event, graph)) { return true; }
|
2013-10-25 10:39:30 +00:00
|
|
|
} else {
|
|
|
|
// Inside main graph area?
|
2014-04-17 05:55:38 +00:00
|
|
|
if ((event->y() > fl->m_rect.y()) && (event->y()) < (fl->m_rect.y() + fl->m_rect.height())) {
|
2013-10-25 10:39:30 +00:00
|
|
|
if (event->x() < lvisible[i]->m_rect.x()) {
|
|
|
|
// Display tooltip
|
2014-04-17 05:55:38 +00:00
|
|
|
QString ttip = schema::channel[fl->code()].fullname() + "\n" +
|
|
|
|
schema::channel[fl->code()].description();
|
|
|
|
graph->ToolTip(ttip, event->x(), event->y() - 15);
|
2013-10-25 10:39:30 +00:00
|
|
|
graph->redraw();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2013-10-25 10:39:30 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
gFlagsLine::gFlagsLine(ChannelID code, QColor flag_color, QString label, bool always_visible,
|
|
|
|
FlagType flt)
|
|
|
|
: Layer(code), m_label(label), m_always_visible(always_visible), m_flt(flt),
|
|
|
|
m_flag_color(flag_color)
|
2011-06-26 08:30:44 +00:00
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
addVertexBuffer(quads = new gVertexBuffer(2048, GL_QUADS));
|
2011-09-01 07:12:25 +00:00
|
|
|
//addGLBuf(lines=new GLBuffer(flag_color,1024,GL_LINES));
|
2011-08-30 17:22:54 +00:00
|
|
|
quads->setAntiAlias(true);
|
2011-09-01 07:12:25 +00:00
|
|
|
//lines->setAntiAlias(true);
|
2011-12-06 14:39:14 +00:00
|
|
|
//GetTextExtent(m_label,m_lx,m_ly);
|
2011-09-01 03:37:25 +00:00
|
|
|
//m_static.setText(m_label);;
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
|
|
|
gFlagsLine::~gFlagsLine()
|
|
|
|
{
|
2011-09-01 07:12:25 +00:00
|
|
|
//delete lines;
|
2011-09-21 14:10:10 +00:00
|
|
|
//delete quads;
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
void gFlagsLine::paint(gGraph &w, int left, int top, int width, int height)
|
2011-06-26 08:30:44 +00:00
|
|
|
{
|
2014-04-17 05:55:38 +00:00
|
|
|
if (!m_visible) { return; }
|
|
|
|
|
|
|
|
if (!m_day) { return; }
|
|
|
|
|
|
|
|
lines = w.lines();
|
2011-06-26 08:30:44 +00:00
|
|
|
double minx;
|
|
|
|
double maxx;
|
|
|
|
|
2011-08-25 06:11:44 +00:00
|
|
|
if (w.blockZoom()) {
|
2014-04-17 05:55:38 +00:00
|
|
|
minx = w.rmin_x;
|
|
|
|
maxx = w.rmax_x;
|
2011-06-26 08:30:44 +00:00
|
|
|
} else {
|
2014-04-17 05:55:38 +00:00
|
|
|
minx = w.min_x;
|
|
|
|
maxx = w.max_x;
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
double xx = maxx - minx;
|
|
|
|
|
|
|
|
if (xx <= 0) { return; }
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
double xmult = width / xx;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
GetTextExtent(m_label, m_lx, m_ly);
|
2011-06-26 08:30:44 +00:00
|
|
|
|
|
|
|
// Draw text label
|
2014-04-17 05:55:38 +00:00
|
|
|
w.renderText(m_label, left - m_lx - 10, top + (height / 2) + (m_ly / 2));
|
2011-09-01 13:07:26 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
float x1, x2;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
float bartop = top + 2;
|
|
|
|
float bottom = top + height - 2;
|
|
|
|
bool verts_exceeded = false;
|
|
|
|
qint64 X, X2, L;
|
2012-01-05 11:35:23 +00:00
|
|
|
lines->setColor(schema::channel[m_code].defaultColor());
|
|
|
|
|
|
|
|
qint64 start;
|
2014-04-17 05:55:38 +00:00
|
|
|
quint32 *tptr;
|
2012-01-10 02:34:44 +00:00
|
|
|
EventStoreType *dptr, * eptr;
|
2012-01-05 11:35:23 +00:00
|
|
|
int idx;
|
2014-04-17 05:55:38 +00:00
|
|
|
QHash<ChannelID, QVector<EventList *> >::iterator cei;
|
2012-01-10 02:34:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
qint64 clockdrift = qint64(PROFILE.cpap->clockDrift()) * 1000L;
|
|
|
|
qint64 drift = 0;
|
2012-01-19 15:18:34 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
for (QList<Session *>::iterator s = m_day->begin(); s != m_day->end(); s++) {
|
|
|
|
if (!(*s)->enabled()) {
|
2012-01-10 02:34:44 +00:00
|
|
|
continue;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
drift = ((*s)->machine()->GetType() == MT_CPAP) ? clockdrift : 0;
|
|
|
|
|
|
|
|
cei = (*s)->eventlist.find(m_code);
|
2012-01-10 02:34:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (cei == (*s)->eventlist.end()) {
|
2012-01-10 02:34:44 +00:00
|
|
|
continue;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVector<EventList *> &evlist = cei.value();
|
|
|
|
|
|
|
|
for (int k = 0; k < evlist.size(); k++) {
|
|
|
|
EventList &el = *(evlist[k]);
|
|
|
|
start = el.first() + drift;
|
|
|
|
tptr = el.rawTime();
|
|
|
|
dptr = el.rawData();
|
|
|
|
int np = el.count();
|
|
|
|
eptr = dptr + np;
|
|
|
|
|
|
|
|
for (idx = 0; dptr < eptr; dptr++, tptr++, idx++) {
|
|
|
|
X = start + *tptr;
|
|
|
|
L = *dptr * 1000;
|
2012-01-10 02:34:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (X >= minx) {
|
2012-01-10 02:34:44 +00:00
|
|
|
break;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
X2 = X - L;
|
|
|
|
|
|
|
|
if (X2 >= minx) {
|
2012-01-10 02:34:44 +00:00
|
|
|
break;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2012-01-10 02:34:44 +00:00
|
|
|
|
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
np -= idx;
|
|
|
|
|
|
|
|
if (m_flt == FT_Bar) {
|
2012-01-10 02:34:44 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Draw Event Flag Bars
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2012-01-10 02:34:44 +00:00
|
|
|
// Check bounds outside of loop is faster..
|
|
|
|
// This will have to be reverted if multithreaded drawing is ever brought back
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
int rem = lines->Max() - lines->cnt();
|
|
|
|
|
|
|
|
if ((np << 1) > rem) {
|
2012-01-10 02:34:44 +00:00
|
|
|
qDebug() << "gFlagsLine would overfill lines for" << schema::channel[m_code].label();
|
2014-04-17 05:55:38 +00:00
|
|
|
np = rem >> 1;
|
|
|
|
verts_exceeded = true;
|
2012-01-10 02:34:44 +00:00
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
for (int i = 0; i < np; i++) {
|
|
|
|
X = start + *tptr++;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (X > maxx) {
|
2012-01-10 02:34:44 +00:00
|
|
|
break;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
x1 = (X - minx) * xmult + left;
|
|
|
|
lines->add(x1, bartop, x1, bottom);
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2012-01-10 02:34:44 +00:00
|
|
|
//if (lines->full()) { verts_exceeded=true; break; }
|
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
} else if (m_flt == FT_Span) {
|
2012-01-10 02:34:44 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Draw Event Flag Spans
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
quads->setColor(m_flag_color);
|
2014-04-17 05:55:38 +00:00
|
|
|
int rem = quads->Max() - quads->cnt();
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if ((np << 2) > rem) {
|
2012-01-10 02:34:44 +00:00
|
|
|
qDebug() << "gFlagsLine would overfill quads for" << schema::channel[m_code].label();
|
2014-04-17 05:55:38 +00:00
|
|
|
np = rem >> 2;
|
|
|
|
verts_exceeded = true;
|
2012-01-10 02:34:44 +00:00
|
|
|
}
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2012-01-10 02:34:44 +00:00
|
|
|
for (; dptr < eptr; dptr++) {
|
2014-04-17 05:55:38 +00:00
|
|
|
X = start + * tptr++;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
if (X > maxx) {
|
2012-01-10 02:34:44 +00:00
|
|
|
break;
|
2014-04-17 05:55:38 +00:00
|
|
|
}
|
2012-01-10 02:34:44 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
L = *dptr * 1000L;
|
|
|
|
X2 = X - L;
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
x1 = double(X - minx) * xmult + left;
|
|
|
|
x2 = double(X2 - minx) * xmult + left;
|
2012-01-05 12:12:42 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
quads->add(x2, bartop, x1, bartop, x1, bottom, x2, bottom);
|
2012-01-10 02:34:44 +00:00
|
|
|
//if (quads->full()) { verts_exceeded=true; break; }
|
2012-01-05 11:35:23 +00:00
|
|
|
|
2012-01-10 02:34:44 +00:00
|
|
|
}
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
if (verts_exceeded) { break; }
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
|
|
|
if (verts_exceeded) { break; }
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-04-17 05:55:38 +00:00
|
|
|
|
2011-08-02 04:51:49 +00:00
|
|
|
if (verts_exceeded) {
|
|
|
|
qWarning() << "maxverts exceeded in gFlagsLine::plot()";
|
|
|
|
}
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:55:38 +00:00
|
|
|
bool gFlagsLine::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
|
2013-10-25 10:39:30 +00:00
|
|
|
{
|
2013-10-25 15:44:14 +00:00
|
|
|
Q_UNUSED(event)
|
|
|
|
Q_UNUSED(graph)
|
2014-04-17 05:55:38 +00:00
|
|
|
// qDebug() << code() << event->x() << event->y() << graph->rect();
|
2013-10-25 10:39:30 +00:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|