/* -*- 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 * * 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 #include #include #include #include "MinutesAtPressure.h" #include "Graphs/gGraph.h" #include "Graphs/gGraphView.h" #include "SleepLib/profiles.h" #include "Graphs/gXAxis.h" MinutesAtPressure::MinutesAtPressure() :Layer(NoChannel) { m_remap = nullptr; } 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); 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_IPAP)); } bool MinutesAtPressure::isEmpty() { return m_empty; } void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion ®ion) { QRect rect = region.boundingRect(); float width = rect.width(); float cells = 28; float pix = width / cells; float left = rect.left(); 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::iterator it; int top = rect.top(); painter.setFont(*defaultfont); painter.setPen(Qt::black); // Lock the stuff we need to draw timelock.lock(); QMap::iterator times_end = times.end(); QString text = STR_TR_Pressure; QRect rec(left,top, pix * 3,0); rec = painter.boundingRect(rec, Qt::AlignTop | Qt::AlignRight, text); rec.moveRight(left-4); painter.drawText(rec, Qt::AlignRight | Qt::AlignVCenter, text); text = STR_UNIT_Minutes; QRect rec2(left, top + rec.height(),pix * 3, 0); rec2 = painter.boundingRect(rec2, Qt::AlignTop | Qt::AlignRight, text); rec2.moveRight(left-4); painter.drawText(rec2, Qt::AlignRight | Qt::AlignVCenter, text); int xpos = left; 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); QRect rec(xpos, top, pix-1, 0); 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")); painter.drawText(rec, Qt::AlignCenter, text); rec.moveTop(top + rec.height()); painter.drawText(rec, Qt::AlignCenter, value); xpos += pix; } float hh = rec.height(); int ypos = top + hh * 2; QHash >::iterator eit; QHash >::iterator ev_end = events.end(); QMap::iterator vit; int row = 0; int numchans = chans.size(); for (int i=0; i< numchans; ++i) { ChannelID code = chans.at(i); schema::Channel & chan = schema::channel[code]; eit = events.find(code); xpos = left; QMap::iterator eit_end = eit.value().end(); QString text = chan.label(); QRect rec2(xpos, ypos, pix * 3, hh); rec2 = painter.boundingRect(rec2, Qt::AlignTop | Qt::AlignRight, text); rec2.moveRight(left-4); painter.drawText(rec2, Qt::AlignRight | Qt::AlignVCenter, text); for (it = times.begin(), vit = eit.value().begin(); vit != eit_end; ++vit, ++it) { float duration = float(it.value()) * 60.0; float value = (vit.value()) ; QRect rec(xpos, ypos, pix-1, hh); if (row & 1) { painter.fillRect(rec, QColor(240,240,240,240)); } painter.drawText(rec, Qt::AlignCenter, QString("%1").arg(value,5,'f',1)); xpos += pix; } ypos += hh; row++; } timelock.unlock(); if (m_recalculating) { // painter.setFont(*defaultfont); // painter.setPen(QColor(0,0,0,125)); // painter.drawText(region.boundingRect(), Qt::AlignCenter, QObject::tr("Recalculating...")); } // Draw the goodies... } void RecalcMAP::run() { QMutexLocker locker(&map->mutex); map->m_recalculating = true; Day * day = map->m_day; if (!day) return; QList::iterator sit; QList::iterator sess_end = day->end(); QMap times; QHash > events; QList chans = day->getSortedMachineChannels(schema::SPAN | schema::FLAG | schema::MINOR_FLAG); ChannelID code; QList badchans; for (int i=0 ; i < chans.size(); ++i) { code = chans.at(i); if (!day->channelExists(code)) badchans.push_back(code); } 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=3; i<=30; i++) { times[i] = 0; for (int c = 0; c < numchans; ++c) { code = chans.at(c); events[code].insert(i, 0); } } ChannelID prescode; if (day->channelExists(CPAP_Pressure)) { prescode = CPAP_Pressure; } else if (day->channelExists(CPAP_IPAP)) { 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 >::iterator ei = sess->eventlist.find(prescode); if (ei == sess->eventlist.end()) break; const QVector & 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)) { if (first) { lasttime = time; lastdata = data; first = false; } goto skip; } if (first) { lasttime = time; lastdata = data; first = false; } if ((lastdata != data) || (time > maxx)) { qint64 d1 = qMax(minx, lasttime); qint64 d2 = qMin(maxx, time); int duration = (d2 - d1) / 1000L; 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) { events[code][key] += sess->rangeSum(code, d1, d2); } else { events[code][key] += sess->rangeCount(code, d1, d2); } } } lasttime = time; lastdata = data; } if (time > maxx) break; skip: if (m_quit) { m_done = true; return; } } } } QMap::iterator it; QMap::iterator times_end = times.end(); int maxtime = 0; for (it = times.begin(); it != times_end; ++it) { maxtime = qMax(it.value(), maxtime); } QMutexLocker timelock(&map->timelock); map->times = times; map->events = events; map->maxtime = maxtime; map->chans = chans; 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) { Q_UNUSED(event); Q_UNUSED(graph); return true; } bool MinutesAtPressure::mousePressEvent(QMouseEvent *event, gGraph *graph) { Q_UNUSED(event); Q_UNUSED(graph); return true; } bool MinutesAtPressure::mouseReleaseEvent(QMouseEvent *event, gGraph *graph) { Q_UNUSED(event); Q_UNUSED(graph); return true; }