OSCAR-code/sleepyhead/Graphs/MinutesAtPressure.cpp

516 lines
15 KiB
C++
Raw Normal View History

2014-08-11 18:29:44 +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:
*
* MinutesAtPressure Graph 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. */
#include <cmath>
#include <QApplication>
#include <QThreadPool>
#include <QMutexLocker>
#include "MinutesAtPressure.h"
#include "Graphs/gGraph.h"
#include "Graphs/gGraphView.h"
#include "SleepLib/profiles.h"
#include "Graphs/gXAxis.h"
#include "Graphs/gYAxis.h"
2014-08-11 18:29:44 +00:00
MinutesAtPressure::MinutesAtPressure() :Layer(NoChannel)
{
m_remap = nullptr;
m_minpressure = 3;
m_maxpressure = 30;
2014-08-11 18:29:44 +00:00
}
MinutesAtPressure::~MinutesAtPressure()
{
while (recalculating()) {};
}
RecalcMAP::~RecalcMAP()
{
}
void RecalcMAP::quit() {
m_quit = true;
map->mutex.lock();
map->mutex.unlock();
}
void MinutesAtPressure::SetDay(Day *day)
{
Layer::SetDay(day);
// look at session summaryValues.
if (day) {
QList<Session *>::iterator sit;
EventDataType minpressure = 40;
EventDataType maxpressure = 0;
Machine * mach = day->machine;
QMap<QDate, Day *>::iterator it;
QMap<QDate, Day *>::iterator day_end = mach->day.end();
// look at overall pressure ranges and find the max
for (it = mach->day.begin(); it != day_end; ++it) {
Day * d = it.value();
QList<Session *>::iterator sess_end = d->end();
for (sit = d->begin(); sit != sess_end; ++sit) {
Session * sess = (*sit);
if (sess->channelExists(CPAP_Pressure)) {
minpressure = qMin(sess->Min(CPAP_Pressure), minpressure);
maxpressure = qMax(sess->Max(CPAP_Pressure), maxpressure);
}
if (sess->channelExists(CPAP_EPAP)) {
minpressure = qMin(sess->Min(CPAP_EPAP), minpressure);
maxpressure = qMax(sess->Max(CPAP_EPAP), maxpressure);
}
if (sess->channelExists(CPAP_IPAP)) {
minpressure = qMin(sess->Min(CPAP_IPAP), minpressure);
maxpressure = qMax(sess->Max(CPAP_IPAP), maxpressure);
}
}
}
m_minpressure = qMax(float(4), floor(minpressure));
m_maxpressure = ceil(maxpressure);
const int minimum_cells = 16;
int c = m_maxpressure - m_minpressure;
if (c < minimum_cells) {
int v = minimum_cells - c;
m_minpressure -= v/2.0;
m_minpressure = qMax((EventStoreType)4, m_minpressure);
m_maxpressure = m_minpressure + minimum_cells;
}
}
2014-08-11 18:29:44 +00:00
m_empty = false;
m_recalculating = false;
m_lastminx = 0;
m_lastmaxx = 0;
m_empty = !m_day || !(m_day->channelExists(CPAP_Pressure) || m_day->channelExists(CPAP_EPAP));
2014-08-11 18:29:44 +00:00
}
bool MinutesAtPressure::isEmpty()
{
2014-08-11 18:36:02 +00:00
2014-08-11 18:29:44 +00:00
return m_empty;
}
2014-08-11 18:29:44 +00:00
void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion &region)
{
QRect rect = region.boundingRect();
int cells = m_maxpressure-m_minpressure+1;
2014-08-11 18:29:44 +00:00
float width = rect.width();
float left = rect.left();
float pix = width / float(cells);
2014-08-11 18:29:44 +00:00
m_minx = graph.min_x;
m_maxx = graph.max_x;
if ((m_lastminx != m_minx) || (m_lastmaxx != m_maxx)) {
recalculate(&graph);
}
m_lastminx = m_minx;
m_lastmaxx = m_maxx;
QMap<EventStoreType, int>::iterator it;
int top = rect.top()+1;
2014-08-11 18:29:44 +00:00
painter.setFont(*defaultfont);
painter.setPen(Qt::black);
// Lock the stuff we need to draw
timelock.lock();
QMap<EventStoreType, int>::iterator times_end = times.end();
QPoint mouse = graph.graphView()->currentMousePos();
2014-08-11 18:29:44 +00:00
float ypos = top;
QString text = schema::channel[m_presChannel].label();
QRectF rec(left - gYAxis::Margin, top, gYAxis::Margin,0);
2014-08-11 18:29:44 +00:00
rec = painter.boundingRect(rec, Qt::AlignTop | Qt::AlignRight, text);
rec.moveRight(left-4);
graph.renderText(text, rec, Qt::AlignRight | Qt::AlignVCenter);
//painter.drawText(rec, Qt::AlignRight | Qt::AlignVCenter, text);
if (rec.contains(mouse)) {
QString text = schema::channel[m_presChannel].description();
graph.ToolTip(text, mouse.x() + 10, mouse.y(), TT_AlignLeft);
}
2014-08-11 18:29:44 +00:00
double tmph = rec.height();
2014-08-11 18:29:44 +00:00
text = STR_UNIT_Minutes;
QRectF rec2(left - gYAxis::Margin, top + rec.height(), gYAxis::Margin, 0);
2014-08-11 18:29:44 +00:00
rec2 = painter.boundingRect(rec2, Qt::AlignTop | Qt::AlignRight, text);
rec2.moveRight(left-4);
//painter.drawText(rec2, Qt::AlignRight | Qt::AlignVCenter, text);
graph.renderText(text, rec2, Qt::AlignRight | Qt::AlignVCenter);
tmph += rec2.height();
2014-08-11 18:29:44 +00:00
float xpos = left;
2014-08-11 18:29:44 +00:00
for (it = times.begin(); it != times_end; ++it) {
QString text = QString::number(it.key());
QString value = QString("%1").arg(float(it.value()) / 60.0, 5, 'f', 1);
QRectF rec(xpos, top, pix-1, 0);
2014-08-11 18:29:44 +00:00
rec = painter.boundingRect(rec, Qt::AlignTop | Qt::AlignLeft, text);
rec = painter.boundingRect(rec, Qt::AlignTop | Qt::AlignLeft, value);
rec.setWidth(pix - 1);
painter.fillRect(rec, QColor("orange"));
graph.renderText(text, rec, Qt::AlignCenter);
//painter.drawText(rec, Qt::AlignCenter, text);
2014-08-11 18:29:44 +00:00
rec.moveTop(top + rec.height());
graph.renderText(value, rec, Qt::AlignCenter);
//painter.drawText(rec, Qt::AlignCenter, value);
2014-08-11 18:29:44 +00:00
xpos += pix;
}
ypos = top + tmph;
left = rect.left();
2014-08-11 18:29:44 +00:00
float hh = rec.height();
2014-08-11 18:29:44 +00:00
QHash<ChannelID, QMap<EventStoreType, EventDataType> >::iterator eit;
QHash<ChannelID, QMap<EventStoreType, EventDataType> >::iterator ev_end = events.end();
QMap<EventStoreType, EventDataType>::iterator vit;
2014-08-11 18:29:44 +00:00
int row = 0;
2014-08-11 18:36:02 +00:00
int numchans = chans.size();
for (int i=0; i< numchans; ++i) {
ChannelID code = chans.at(i);
2014-08-11 18:29:44 +00:00
schema::Channel & chan = schema::channel[code];
if (!chan.enabled())
continue;
schema::ChanType type = chan.type();
2014-08-11 18:36:02 +00:00
eit = events.find(code);
2014-08-11 18:29:44 +00:00
xpos = left;
QMap<EventStoreType, EventDataType>::iterator eit_end = eit.value().end();
QString text = chan.label();
QRectF rec2(xpos - gYAxis::Margin, ypos, gYAxis::Margin, hh);
rec2 = painter.boundingRect(rec2, Qt::AlignRight | Qt::AlignVCenter, text);
2014-08-11 18:29:44 +00:00
rec2.moveRight(left-4);
if (rec2.contains(mouse)) {
QString text = chan.fullname();
if (type == schema::SPAN) {
text += "\n"+QObject::tr("(% of time)");
}
graph.ToolTip(text, mouse.x() + 10, mouse.y(), TT_AlignLeft);
}
graph.renderText(text, rec2, Qt::AlignRight | Qt::AlignVCenter);
//painter.drawText(rec2, Qt::AlignRight | Qt::AlignVCenter, text);
2014-08-11 18:29:44 +00:00
for (it = times.begin(), vit = eit.value().begin(); vit != eit_end; ++vit, ++it) {
float minutes = float(it.value()) / 60.0;
float value = vit.value();
QString fmt = "%1";
if (type != schema::SPAN) {
//fmt = "%1";
value = (minutes > 0.000001) ? (value * 60.0) / minutes : 0;
} else {
//fmt = "%1%";
value = (minutes > 0.000001) ? (100/minutes) * (value / 60.0) : 0;
}
2014-08-11 18:29:44 +00:00
QRectF rec(xpos, ypos, pix-1, hh);
if ((row & 1) == 0) {
painter.fillRect(rec, QColor(245,245,255,240));
2014-08-11 18:29:44 +00:00
}
graph.renderText(QString(fmt).arg(value,5,'f',2), rec, Qt::AlignCenter);
// painter.drawText(rec, Qt::AlignCenter, QString(fmt).arg(value,5,'f',2));
2014-08-11 18:29:44 +00:00
xpos += pix;
}
ypos += hh;
row++;
}
timelock.unlock();
if (m_recalculating) {
2014-08-11 18:36:02 +00:00
// painter.setFont(*defaultfont);
// painter.setPen(QColor(0,0,0,125));
// painter.drawText(region.boundingRect(), Qt::AlignCenter, QObject::tr("Recalculating..."));
2014-08-11 18:29:44 +00:00
}
// painter.setPen(QPen(Qt::black,1));
// painter.drawRect(rect);
2014-08-11 18:29:44 +00:00
// Draw the goodies...
}
void RecalcMAP::run()
{
QMutexLocker locker(&map->mutex);
map->m_recalculating = true;
Day * day = map->m_day;
if (!day) return;
QList<Session *>::iterator sit;
QList<Session *>::iterator sess_end = day->end();
QMap<EventStoreType, int> times;
QHash<ChannelID, QMap<EventStoreType, EventDataType> > events;
QList<ChannelID> chans = day->getSortedMachineChannels(schema::SPAN | schema::FLAG | schema::MINOR_FLAG);
ChannelID code;
QList<ChannelID> badchans;
for (int i=0 ; i < chans.size(); ++i) {
code = chans.at(i);
// if (!day->channelExists(code)) badchans.push_back(code);
2014-08-11 18:29:44 +00:00
}
for (int i=0; i < badchans.size(); ++i) {
code = badchans.at(i);
chans.removeAll(code);
}
int numchans = chans.size();
// Zero the pressure counts
for (int i=map->m_minpressure; i <= map->m_maxpressure; i++) {
2014-08-11 18:29:44 +00:00
times[i] = 0;
for (int c = 0; c < numchans; ++c) {
code = chans.at(c);
events[code].insert(i, 0);
}
}
ChannelID prescode = CPAP_Pressure;
2014-08-11 18:29:44 +00:00
if (day->channelExists(CPAP_IPAP)) {
2014-08-11 18:29:44 +00:00
prescode = CPAP_IPAP;
} else if (day->channelExists(CPAP_EPAP)) {
prescode = CPAP_EPAP;
}
qint64 minx, maxx;
map->m_graph->graphView()->GetXBounds(minx, maxx);
for (sit = day->begin(); sit != sess_end; ++sit) {
Session * sess = (*sit);
QHash<ChannelID, QVector<EventList *> >::iterator ei = sess->eventlist.find(prescode);
if (ei == sess->eventlist.end())
2014-08-11 19:50:21 +00:00
continue;
2014-08-11 18:29:44 +00:00
const QVector<EventList *> & evec = ei.value();
int esize = evec.size();
for (int ei = 0; ei < esize; ++ei) {
EventList *EL = evec[ei];
EventDataType gain = EL->gain();
quint32 ELsize = EL->count();
if (ELsize < 1) return;
qint64 lasttime = 0; //EL->time(0);
EventStoreType lastdata = 0; // EL->raw(0);
bool first = true;
if ((EL->first() > maxx) || (EL->last() < minx)) {
continue;
}
for (quint32 e = 0; e < ELsize; ++e) {
qint64 time = EL->time(e);
EventStoreType data = EL->raw(e);
if ((time < minx)) {
2014-08-12 06:56:17 +00:00
lasttime = time;
lastdata = data;
first = false;
2014-08-11 18:29:44 +00:00
goto skip;
}
if (first) {
lasttime = time;
lastdata = data;
first = false;
}
if ((lastdata != data) || (time > maxx)) {
2014-08-11 18:42:26 +00:00
qint64 d1 = qMax(minx, lasttime);
qint64 d2 = qMin(maxx, time);
2014-08-11 18:29:44 +00:00
2014-08-11 18:42:26 +00:00
int duration = (d2 - d1) / 1000L;
2014-08-11 18:29:44 +00:00
EventStoreType key = floor(lastdata * gain);
if (key <= 30) {
times[key] += duration;
for (int c = 0; c < chans.size(); ++c) {
ChannelID code = chans.at(c);
schema::Channel & chan = schema::channel[code];
if (chan.type() == schema::SPAN) {
2014-08-11 18:42:26 +00:00
events[code][key] += sess->rangeSum(code, d1, d2);
2014-08-11 18:29:44 +00:00
} else {
2014-08-11 18:42:26 +00:00
events[code][key] += sess->rangeCount(code, d1, d2);
2014-08-11 18:29:44 +00:00
}
}
}
lasttime = time;
lastdata = data;
}
if (time > maxx) break;
skip:
if (m_quit) {
m_done = true;
return;
}
}
}
}
QMap<EventStoreType, int>::iterator it;
QMap<EventStoreType, int>::iterator times_end = times.end();
int maxtime = 0;
QList<EventStoreType> trash;
2014-08-11 18:29:44 +00:00
for (it = times.begin(); it != times_end; ++it) {
EventStoreType key = it.key();
int value = it.value();
// if (value == 0) {
// trash.append(key);
// } else {
maxtime = qMax(value, maxtime);
// }
2014-08-11 18:29:44 +00:00
}
chans.push_front(CPAP_AHI);
for (int i = map->m_minpressure; i <= map->m_maxpressure; i++) {
events[CPAP_AHI].insert(i,events[CPAP_Obstructive][i] + events[CPAP_Hypopnea][i] + events[CPAP_Apnea][i] + events[CPAP_ClearAirway][i]);
}
2014-08-11 18:29:44 +00:00
QHash<ChannelID, QMap<EventStoreType, EventDataType> >::iterator eit;
// for (int i=0; i< trash.size(); ++i) {
// EventStoreType key = trash.at(i);
// times.remove(key);
// for (eit = events.begin(); eit != events.end(); ++eit) {
// eit.value().remove(key);
// }
// }
2014-08-11 18:29:44 +00:00
QMutexLocker timelock(&map->timelock);
map->times = times;
map->events = events;
map->maxtime = maxtime;
map->chans = chans;
map->m_presChannel = prescode;
2014-08-11 18:29:44 +00:00
timelock.unlock();
map->recalcFinished();
m_done = true;
}
void MinutesAtPressure::recalculate(gGraph * graph)
{
while (recalculating())
m_remap->quit();
m_remap = new RecalcMAP(this);
m_remap->setAutoDelete(true);
m_graph = graph;
QThreadPool * tp = QThreadPool::globalInstance();
// tp->reserveThread();
while(!tp->tryStart(m_remap));
// Start recalculating in another thread, organize a callback to redraw when done..
}
void MinutesAtPressure::recalcFinished()
{
if (m_graph) {
m_graph->timedRedraw(0);
}
m_recalculating = false;
m_remap = nullptr;
// QThreadPool * tp = QThreadPool::globalInstance();
// tp->releaseThread();
}
bool MinutesAtPressure::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
{
// int y = event->y() - m_rect.top();
// int x = event->x() - graph->graphView()->titleWidth;
// double w = m_rect.width() - gYAxis::Margin;
// double xmult = (graph->blockZoom() ? double(graph->rmax_x - graph->rmin_x) : double(graph->max_x - graph->min_x)) / w;
// double a = x - gYAxis::Margin;
// if (a < 0) a = 0;
// if (a > w) a = w;
// double b = a * xmult;
// double c= b + (graph->blockZoom() ? graph->rmin_x : graph->min_x);
// graph->graphView()->setCurrentTime(c);
graph->timedRedraw(0);
return false;
2014-08-11 18:29:44 +00:00
}
bool MinutesAtPressure::mousePressEvent(QMouseEvent *event, gGraph *graph)
{
Q_UNUSED(event);
Q_UNUSED(graph);
return false;
2014-08-11 18:29:44 +00:00
}
bool MinutesAtPressure::mouseReleaseEvent(QMouseEvent *event, gGraph *graph)
{
Q_UNUSED(event);
Q_UNUSED(graph);
return false;
2014-08-11 18:29:44 +00:00
}