OSCAR-code/sleepyhead/Graphs/gLineOverlay.cpp

424 lines
14 KiB
C++
Raw Normal View History

2014-08-17 12:56:05 +00:00
/* gLineOverlayBar Implementation
*
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.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 <math.h>
2011-07-18 03:36:26 +00:00
#include "SleepLib/profiles.h"
2011-06-26 08:30:44 +00:00
#include "gLineOverlay.h"
gLineOverlayBar::gLineOverlayBar(ChannelID code, QColor color, QString label, FlagType flt)
2014-08-05 11:17:03 +00:00
: Layer(code), m_flag_color(color), m_label(label), m_flt(flt), m_odt(ODT_TopAndBottom)
2011-06-26 08:30:44 +00:00
{
m_hover = false;
m_blockhover = false;
2011-06-26 08:30:44 +00:00
}
gLineOverlayBar::~gLineOverlayBar()
{
}
2011-07-27 09:21:53 +00:00
2014-08-21 06:03:30 +00:00
QColor brighten(QColor, float);
void gLineOverlayBar::paint(QPainter &painter, gGraph &w, const QRegion &region)
2011-06-26 08:30:44 +00:00
{
m_hover = false;
2014-08-05 11:17:03 +00:00
if (!schema::channel[m_code].enabled())
return;
int left = region.boundingRect().left();
int topp = region.boundingRect().top(); // FIXME: Misspelling intentional.
int width = region.boundingRect().width();
int height = region.boundingRect().height();
if (!m_visible) { return; }
2011-06-26 08:30:44 +00:00
if (!m_day) { return; }
2011-06-26 08:30:44 +00:00
int start_py = topp;
2011-06-26 08:30:44 +00:00
double xx = w.max_x - w.min_x;
2014-10-02 07:56:57 +00:00
//double yy = w.max_y - w.min_y;
double jj = double(width) / double(xx);
2011-06-26 08:30:44 +00:00
if (xx <= 0) { return; }
2011-06-26 08:30:44 +00:00
double x1, x2;
int x, y;
float bottom = start_py + height - 25 * w.printScaleY(), top = start_py + 25 * w.printScaleY();
2011-07-27 09:21:53 +00:00
double X;
double Y;
QPoint mouse=w.graphView()->currentMousePos();
m_count = 0;
m_sum = 0;
m_flag_color = schema::channel[m_code].defaultColor();
if (m_flt == FT_Span) {
m_flag_color.setAlpha(128);
}
2014-05-07 19:52:59 +00:00
painter.setPen(m_flag_color);
EventStoreType raw;
quint32 *tptr;
EventStoreType *dptr, *eptr;
2012-01-05 11:35:23 +00:00
qint64 stime;
2014-08-05 11:17:03 +00:00
OverlayDisplayType odt = m_odt;
QHash<ChannelID, QVector<EventList *> >::iterator cei;
int count;
2014-07-11 12:09:38 +00:00
qint64 clockdrift = qint64(p_profile->cpap->clockDrift()) * 1000L;
qint64 drift = 0;
//bool hover = false;
2012-01-05 11:35:23 +00:00
// For each session, process it's eventlist
for (QList<Session *>::iterator s = m_day->begin(); s != m_day->end(); s++) {
if (!(*s)->enabled()) { continue; }
cei = (*s)->eventlist.find(m_code);
if (cei == (*s)->eventlist.end()) { continue; }
QVector<EventList *> &evlist = cei.value();
if (evlist.size() == 0) { continue; }
drift = ((*s)->type() == MT_CPAP) ? clockdrift : 0;
2011-07-27 09:21:53 +00:00
2012-01-05 11:35:23 +00:00
// Could loop through here, but nowhere uses more than one yet..
for (int k = 0; k < evlist.size(); k++) {
EventList &el = *(evlist[k]);
count = el.count();
stime = el.first() + drift;
dptr = el.rawData();
eptr = dptr + count;
tptr = el.rawTime();
2011-06-26 08:30:44 +00:00
2012-01-05 11:35:23 +00:00
////////////////////////////////////////////////////////////////////////////
// Skip data previous to minx bounds
2012-01-05 11:35:23 +00:00
////////////////////////////////////////////////////////////////////////////
for (; dptr < eptr; dptr++) {
X = stime + *tptr;
if (X >= w.min_x) {
break;
}
tptr++;
2012-01-05 11:35:23 +00:00
}
if (m_flt == FT_Span) {
////////////////////////////////////////////////////////////////////////////
// FT_Span
////////////////////////////////////////////////////////////////////////////
for (; dptr < eptr; dptr++) {
//hover = false;
X = stime + *tptr++;
raw = *dptr;
Y = X - (qint64(raw) * 1000.0L); // duration
if (Y > w.max_x) {
break;
}
x1 = jj * double(X - w.min_x) + left;
m_count++;
m_sum += raw;
x2 = jj * double(Y - w.min_x) + left;
if (int(x1) == int(x2)) {
x2 += 1;
}
if (x2 < left) {
x2 = left;
}
if (x1 > width + left) {
x1 = width + left;
}
QRect rect(x2, start_py, x1-x2, height);
QColor col = m_flag_color;
// if (rect.contains(mouse)) {
// hover = true;
// }
painter.fillRect(rect, QBrush(col));
2011-06-26 08:30:44 +00:00
}
}/* else if (m_flt == FT_Dot) {
////////////////////////////////////////////////////////////////////////////
// FT_Dot
////////////////////////////////////////////////////////////////////////////
for (; dptr < eptr; dptr++) {
hover = false;
X = stime + *tptr++; //el.time(i);
raw = *dptr; //el.data(i);
if (X > w.max_x) {
break;
}
x1 = jj * double(X - w.min_x) + left;
m_count++;
m_sum += raw;
// if ((odt == ODT_Bars) || (xx < 3600000)) {
// show the fat dots in the middle
2014-05-07 19:52:59 +00:00
painter.setPen(QPen(m_flag_color,4));
painter.drawPoint(x1, double(height) / double(yy)*double(-20 - w.min_y) + topp);
painter.setPen(QPen(m_flag_color,1));
// } else {
// // thin lines down the bottom
// painter.drawLine(x1, start_py + 1, x1, start_py + 1 + 12);
// }
2012-01-05 11:35:23 +00:00
}
} */else if ((m_flt == FT_Bar) || (m_flt == FT_Dot)) {
////////////////////////////////////////////////////////////////////////////
// FT_Bar
////////////////////////////////////////////////////////////////////////////
for (; dptr < eptr; dptr++) {
// hover = false;
X = stime + *tptr++;
raw = *dptr;
if (X > w.max_x) {
break;
}
x1 = jj * (double(X) - double(w.min_x)) + left;
m_count++;
m_sum += raw;
int z = start_py + height;
double d1 = jj * double(raw) * 1000.0;
if ((m_flt == FT_Bar) && (odt == ODT_Bars)) { // || (xx < 3600000)) {
QRect rect(x1-d1, top, d1+4, height);
QColor col = m_flag_color;
painter.setPen(QPen(col,4));
2014-05-07 19:52:59 +00:00
painter.drawPoint(x1, top);
if (!w.selectingArea() && !m_blockhover && rect.contains(mouse) && !m_hover) {
m_hover = true;
QColor col2(230,230,230,128);
QRect rect((x1-d1), start_py+2, d1, height-2);
if (rect.x() < left) {
rect.setX(left);
}
painter.fillRect(rect, QBrush(col2));
painter.setPen(col);
painter.drawRect(rect);
// Draw text label
2014-08-06 07:08:34 +00:00
QString lab = QString("%1 (%2)").arg(schema::channel[m_code].fullname()).arg(raw);
GetTextExtent(lab, x, y);
w.ToolTip(lab, x1 - 10, start_py + 24 + (3 * w.printScaleY()), TT_AlignRight, p_profile->general->tooltipTimeout());
//painter.fillRect(x1 - (x / 2) - x, start_py + 14 + (3 * w.printScaleY()), x+4,y+4, QBrush(QColor(255,255,255,245)));
// painter.setPen(QPen(Qt::gray,1));
// painter.drawRect(x1 - (x / 2) - x, start_py + 14 + (3 * w.printScaleY()), x+4,y+4);
// w.renderText(lab, x1 - (x / 2)+2 - x, start_py + 14 + y + (3 * w.printScaleY()),0);
// painter.drawLine(rect.x(), top, rect.x()+d1, top);
// painter.drawLine(rect.x(), bottom, rect.x()+d1, bottom);
// painter.drawLine(rect.x(), top, rect.x(), bottom);
// col = COLOR_Gold;
// hover = true;
painter.setPen(QPen(col,3));
} else {
painter.setPen(QPen(col,1));
}
painter.drawLine(x1, top, x1, bottom);
if (xx < (3600000)) {
2014-08-06 07:08:34 +00:00
QString lab = QString("%1").arg(m_label);
GetTextExtent(lab, x, y);
2014-08-29 04:02:16 +00:00
w.renderText(lab, x1 - (x / 2), top - y + (5 * w.printScaleY()),0);
}
} else {
//////////////////////////////////////////////////////////////////////////////////////
// Top and bottom markers
//////////////////////////////////////////////////////////////////////////////////////
//bool b = false;
if (!w.selectingArea() && !m_blockhover && QRect(x1-2, topp, 6, height).contains(mouse) && !m_hover) {
// only want to draw the highlight/label once per frame
m_hover = true;
//b = true;
// Draw text label
2014-08-06 07:08:34 +00:00
QString lab = QString("%1 (%2)").arg(schema::channel[m_code].fullname()).arg(raw);
GetTextExtent(lab, x, y, defaultfont);
w.ToolTip(lab, x1 - 10, start_py + 24 + (3 * w.printScaleY()), TT_AlignRight, p_profile->general->tooltipTimeout());
// painter.fillRect(x1 - (x / 2) - x, start_py + 14 + (3 * w.printScaleY()), x+4,y+4, QBrush(QColor(255,255,255,245)));
// painter.setPen(QPen(Qt::gray,1));
// painter.drawRect(x1 - (x / 2) - x, start_py + 14 + (3 * w.printScaleY()), x+4,y+4);
// w.renderText(lab, x1 - (x / 2)+2 - x, start_py + 14 + y + (3 * w.printScaleY()),0);
//x1-=1;
QColor col = m_flag_color;
col.setAlpha(60);
painter.setPen(QPen(col, 4));
painter.drawLine(x1, start_py+14, x1, z - 12);
painter.setPen(QPen(m_flag_color,4));
painter.drawLine(x1, z, x1, z - 14);
painter.drawLine(x1, start_py+2, x1, start_py + 16);
} else {
QColor col = m_flag_color;
col.setAlpha(10);
painter.setPen(QPen(col,1));
painter.drawLine(x1, start_py+14, x1, z);
painter.setPen(QPen(m_flag_color,1));
// painter.drawLine(x1, z, x1, z - 12);
painter.drawLine(x1, start_py+2, x1, start_py + 14);
}
}
2012-01-05 11:35:23 +00:00
}
}
}
2011-06-26 08:30:44 +00:00
}
}
bool gLineOverlayBar::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
{
Q_UNUSED(event)
Q_UNUSED(graph)
return false;
}
2011-06-26 08:30:44 +00:00
gLineOverlaySummary::gLineOverlaySummary(QString text, int x, int y)
: Layer(CPAP_Obstructive), m_text(text), m_x(x), m_y(y) // The Layer code is a dummy here.
{
}
gLineOverlaySummary::~gLineOverlaySummary()
{
}
void gLineOverlaySummary::paint(QPainter &painter, gGraph &w, const QRegion &region)
{
2014-05-08 04:07:23 +00:00
Q_UNUSED(painter)
int left = region.boundingRect().left();
int top = region.boundingRect().top();
int width = region.boundingRect().width();
int height = region.boundingRect().height();
if (!m_visible) { return; }
if (!m_day) { return; }
Q_UNUSED(width);
Q_UNUSED(height);
float cnt = 0;
double sum = 0;
bool isSpan = false;
for (int i = 0; i < m_overlays.size(); i++) {
cnt += m_overlays[i]->count();
sum += m_overlays[i]->sum();
if (m_overlays[i]->flagtype() == FT_Span) { isSpan = true; }
}
double val, 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;
}
val = 0;
time /= 1000;
int h = time / 3600;
int m = int(time / 60) % 60;
int s = int(time) % 60;
time /= 3600;
//if (time<1) time=1;
if (time > 0) { val = cnt / time; }
QString a;
if (0) { //(w.graphView()->selectionInProgress())) { // || w.graphView()->metaSelect()) && (!w.selDurString().isEmpty())) {
a = QObject::tr("Duration")+": "+w.selDurString();
} else {
a = QObject::tr("Events") + ": " + QString::number(cnt) + ", " +
QObject::tr("Duration") + " " + QString().sprintf("%02i:%02i:%02i", h, m, s) + ", " +
m_text + ": " + QString::number(val, 'f', 2);
}
if (isSpan) {
float sph;
if (!time) { sph = 0; }
else {
sph = (100.0 / float(time)) * (sum / 3600.0);
if (sph > 100) { sph = 100; }
}
// eg: %num of time in a span, like Periodic Breathing
a += " " + QObject::tr("(% %1 in events)").arg(sph, 0, 'f', 2);
}
w.renderText(a, left + m_x, top + m_y);
}