From 0eccfb6d2187f641b4bba69df3718b88d8257425 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Mon, 7 Mar 2016 19:10:58 +1000 Subject: [PATCH] Bye bye ugly By Pressure text, start of proper Time At Pressure chart --- sleepyhead/Graphs/MinutesAtPressure.cpp | 447 ++++++++++++++++++++---- sleepyhead/Graphs/MinutesAtPressure.h | 54 +++ sleepyhead/Graphs/gXAxis.cpp | 32 ++ sleepyhead/Graphs/gXAxis.h | 45 +++ sleepyhead/daily.cpp | 3 +- sleepyhead/main.cpp | 12 +- 6 files changed, 516 insertions(+), 77 deletions(-) diff --git a/sleepyhead/Graphs/MinutesAtPressure.cpp b/sleepyhead/Graphs/MinutesAtPressure.cpp index ea90ab86..80191fa0 100644 --- a/sleepyhead/Graphs/MinutesAtPressure.cpp +++ b/sleepyhead/Graphs/MinutesAtPressure.cpp @@ -20,6 +20,17 @@ #include "Graphs/gYAxis.h" +// Calculate Catmull-Rom Spline of given 4 samples, with t between 0-1; +float CatmullRomSpline(float p0, float p1, float p2, float p3, float t = 0.5) +{ + float t2 = t*t; + float t3 = t2 * t; + + return (float)0.5 * ((2 * p1) + + (-p0 + p2) * t + + (2*p0 - 5*p1 + 4*p2 - p3) * t2 + + (-p0 + 3*p1- 3*p2 + p3) * t3); +} MinutesAtPressure::MinutesAtPressure() :Layer(NoChannel) { @@ -94,7 +105,8 @@ void MinutesAtPressure::SetDay(Day *day) m_maxpressure = m_minpressure + minimum_cells; } QFontMetrics FM(*defaultfont); - QList chans = day->getSortedMachineChannels(schema::SPAN | schema::FLAG | schema::MINOR_FLAG); + quint32 chantype = schema::SPAN | schema::FLAG | schema::MINOR_FLAG; + QList chans = day->getSortedMachineChannels(chantype); m_minimum_height = (chans.size()+3) * FM.height() - 5; } @@ -119,17 +131,6 @@ bool MinutesAtPressure::isEmpty() return m_empty; } -// Calculate Catmull-Rom Spline of given 4 samples, with t between 0-1; -float CatmullRomSpline(float p0, float p1, float p2, float p3, float t = 0.5) -{ - float t2 = t*t; - float t3 = t2 * t; - - return (float)0.5 * ((2 * p1) + - (-p0 + p2) * t + - (2*p0 - 5*p1 + 4*p2 - p3) * t2 + - (-p0 + 3*p1- 3*p2 + p3) * t3); -} void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion ®ion) { @@ -144,6 +145,7 @@ void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion &r float height = rect.height(); float left = rect.left(); float pix = width / float(cells); + float bottom = rect.bottom(); int numchans = chans.size(); @@ -177,10 +179,115 @@ void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion &r // Lock the stuff we need to draw timelock.lock(); + painter.setFont(*defaultfont); painter.setPen(Qt::black); + painter.drawRect(rect.left(),rect.top(), rect.width(), height); - QMap::iterator times_end = times.end(); + + + int min = 40; + int max = 240; + int tot = max - min; + float xstep = float(width) / float(tot); + height -= 2; + float peak = float(qMax(ipap.peaktime, epap.peaktime)); + float ystep = float(height) / peak; + + int p0, p1, p2, p3; + + if (ipap.min_pressure > 0) { + float xp,yp; + + float pstep = xstep*10.0; + + xp = left + pstep/2.0; + int w, h; + for (int i = 0; i<=20; ++i) { + yp = bottom; + painter.drawLine(xp, yp, xp, yp+6); + + QString label = QString("%1").arg(i+4); + GetTextExtent(label, w, h); + graph.renderText(label, xp-w/2, yp+h+4); + xp+= pstep; + } + + xstep /= 4.0; + painter.setPen(Qt::red); + + xp=left; + float lastyp = bottom - (float(ipap.times[min-1]) * ystep); + for (int i=min; i::iterator times_end = times.end(); QPoint mouse = graph.graphView()->currentMousePos(); float ypos = top; @@ -362,19 +469,19 @@ void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion &r xpos += pix / 10.0; } - /* float minutes = float(it.value()) / 60.0; - y1 = minutes * ymult; +// float minutes = float(it.value()) / 60.0; +// y1 = minutes * ymult; - it=times.begin(); - it++; - for (; it != times_end; ++it) { - float minutes = float(it.value()) / 60.0; - y2 = minutes * ymult; +// it=times.begin(); +// it++; +// for (; it != times_end; ++it) { +// float minutes = float(it.value()) / 60.0; +// y2 = minutes * ymult; - painter.drawLine(xpos, bottom-y1, xpos+pix, bottom-y2); - y1 = y2; - xpos += pix; - }*/ +// painter.drawLine(xpos, bottom-y1, xpos+pix, bottom-y2); +// y1 = y2; +// xpos += pix; +// } float maxev = 0; @@ -436,7 +543,7 @@ void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion &r // QString txt=QString("%1 %2").arg(maxmins).arg(float(maxevents * 60.0) / maxmins); // graph.renderText(txt, rect.left(), rect.top()-10); - +*/ timelock.unlock(); if (m_recalculating) { @@ -454,6 +561,187 @@ void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion &r } +//! \brief Updates Time At Pressure from session *sess +void RecalcMAP::updateTimes(PressureInfo & info, Session * sess) +{ + qint64 d1,d2; + EventDataType gain; + EventStoreType data, lastdata; + qint64 time, lasttime; + int ELsize, duration; + bool first; + int key; + EventDataType val = 0; + + ChannelID code = info.code; + qint64 minx = info.minx; + qint64 maxx = info.maxx; + + if (code == 0) return; + + + // Find pressure channel + QHash >::iterator ei = sess->eventlist.find(code); + + // Done already if no channel + if (ei == sess->eventlist.end()) + return; + + const QVector & evec = ei.value(); + int esize = evec.size(); + + // Loop through event lists + for (int ei = 0; ei < esize; ++ei) { + const EventList *EL = evec.at(ei); + gain = EL->gain(); + + // Don't bother with short sessions + ELsize = EL->count(); + if (ELsize < 1) continue; + + lasttime = 0; + lastdata = 0; + + first = true; + + // Skip if outside of range + if ((EL->first() > maxx) || (EL->last() < minx)) { + continue; + } + + // Scan through pressure samples + for (int e = 0; e < ELsize; ++e) { + if (m_quit) { + m_done = true; + return; + } + + time = EL->time(e); + data = floor(float(EL->raw(e)) * gain * 10.0); // pressure times ten, so can look at .1 intervals in an integer + + Q_ASSERT(data < 300); + + if ((time < minx) || first) { + lasttime = time; + lastdata = data; + + first = false; + continue; + } + + if (lastdata != data) { + d1 = qMax(minx, lasttime); + d2 = qMin(maxx, time); + + duration = (d2 - d1) / 1000L; + info.times[lastdata] += duration; + + key = lastdata; + + int cs = info.chans.size(); + for (int c = 0; c < cs; ++c) { + ChannelID cod = info.chans.at(c); + schema::Channel & chan = schema::channel[cod]; + if (chan.type() == schema::SPAN) { + info.events[cod][key] += val = sess->rangeSum(cod, d1, d2); + } else { + info.events[cod][key] += val = sess->rangeCount(cod, d1, d2); + } + } + + lasttime = time; + lastdata = data; + + } + if (time > maxx) { + break; + } + } + if (lasttime < maxx) { + d1 = qMax(lasttime, minx); + d2 = qMin(maxx, EL->last()); + + duration = (d2 - d1) / 1000L; + info.times[lastdata] += duration; + key = lastdata; + int cs = info.chans.size(); + for (int c = 0; c < cs; ++c) { + ChannelID cod = info.chans.at(c); + schema::Channel & chan = schema::channel[cod]; + if (chan.type() == schema::SPAN) { + info.events[cod][key] += sess->rangeSum(cod, d1, d2); + } else { + info.events[cod][key] += sess->rangeCount(cod, d1, d2); + } + } + + } + + } +} + + +void PressureInfo::finishCalcs() +{ + peaktime = peakevents = 0; + min_pressure = max_pressure = 0; + + int val; + + for (int i=0; i 0) { + if (min_pressure == 0) { + min_pressure = i; + } + max_pressure = i; + } + } + + ChannelID cod; + chans.push_front(CPAP_AHI); + + int size = events[CPAP_Obstructive].size(); + + events[CPAP_AHI].resize(size); + + + QHash >::iterator OB = events.find(CPAP_Obstructive); + QHash >::iterator HY = events.find(CPAP_Hypopnea); + QHash >::iterator A = events.find(CPAP_Apnea); + QHash >::iterator CA = events.find(CPAP_ClearAirway); + + + for (int i = 0; i < size; i++) { + + val = 0; + + if (OB != events.end()) + val += OB.value()[i]; + if (HY != events.end()) + val += HY.value()[i]; + if (A != events.end()) + val += A.value()[i]; + if (CA != events.end()) + val += CA.value()[i]; + + events[CPAP_AHI][i] = val; + } + + for (int i = 0; i < size; i++) { + + for (int j=0 ; j < chans.size(); ++j) { + cod = chans.at(j); + if ((cod == CPAP_AHI) || (schema::channel[cod].type() == schema::SPAN)) continue; + val = events[cod][i]; + peakevents = qMax(val, peakevents); + } + } +} + + void RecalcMAP::run() { QMutexLocker locker(&map->mutex); @@ -469,48 +757,59 @@ void RecalcMAP::run() QHash > events; + // Get the channels for specified Channel types 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=map->m_minpressure; i <= map->m_maxpressure; i++) { - times[i] = 0; - - for (int c = 0; c < numchans; ++c) { - code = chans.at(c); - events[code].insert(i, 0); - } - } - - ChannelID prescode = CPAP_Pressure; - -// if (day->channelExists(CPAP_IPAP)) { -// prescode = CPAP_IPAP; -// } else - if (day->channelExists(CPAP_EPAP)) { - prescode = CPAP_EPAP; - } + ChannelID ipapcode = (day->channelExists(CPAP_IPAP)) ? CPAP_IPAP : CPAP_Pressure; + ChannelID epapcode = (day->channelExists(CPAP_EPAP)) ? CPAP_EPAP : 0; qint64 minx, maxx; map->m_graph->graphView()->GetXBounds(minx, maxx); + PressureInfo IPAP(ipapcode, minx, maxx), EPAP(epapcode, minx, maxx); + + IPAP.AddChannels(chans); + EPAP.AddChannels(chans); + + 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=map->m_minpressure; i <= map->m_maxpressure; i++) { +// times[i] = 0; + +// for (int c = 0; c < numchans; ++c) { +// code = chans.at(c); +// events[code].insert(i, 0); +// } +// } + + for (sit = day->begin(); sit != sess_end; ++sit) { Session * sess = (*sit); - QHash >::iterator ei = sess->eventlist.find(prescode); + + updateTimes(EPAP, sess); + updateTimes(IPAP, sess); + + if (m_quit) { + m_done = true; + return; + } + + +/* QHash >::iterator ei = sess->eventlist.find(ipapcode); if (ei == sess->eventlist.end()) continue; @@ -600,11 +899,14 @@ skip: } - } + } */ } - QMap::iterator it; + EPAP.finishCalcs(); + IPAP.finishCalcs(); + +/* QMap::iterator it; QMap::iterator times_end = times.end(); int maxtime = 0; @@ -623,7 +925,11 @@ skip: int maxevents = 0, val; for (int i = map->m_minpressure; i <= map->m_maxpressure; i++) { - val = events[CPAP_Obstructive][i] + events[CPAP_Hypopnea][i] + events[CPAP_Apnea][i] + events[CPAP_ClearAirway][i]; + val = events[CPAP_Obstructive][i] + + events[CPAP_Hypopnea][i] + + events[CPAP_Apnea][i] + + events[CPAP_ClearAirway][i]; + events[CPAP_AHI].insert(i, val); // maxevents = qMax(val, maxevents); } @@ -646,15 +952,15 @@ skip: // eit.value().remove(key); // } // } - +*/ QMutexLocker timelock(&map->timelock); - map->times = times; - map->events = events; - map->maxtime = maxtime; - map->maxevents = maxevents; - map->chans = chans; - map->m_presChannel = prescode; +// map->times = times; +// map->events = events; + map->epap = EPAP; + map->ipap = IPAP; +// map->chans = chans; + // map->m_presChannel = ipapcode; timelock.unlock(); map->recalcFinished(); @@ -707,7 +1013,8 @@ bool MinutesAtPressure::mouseMoveEvent(QMouseEvent *, gGraph *graph) // 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 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; diff --git a/sleepyhead/Graphs/MinutesAtPressure.h b/sleepyhead/Graphs/MinutesAtPressure.h index 5a9c4540..21f3c268 100644 --- a/sleepyhead/Graphs/MinutesAtPressure.h +++ b/sleepyhead/Graphs/MinutesAtPressure.h @@ -13,6 +13,55 @@ #include "SleepLib/day.h" class MinutesAtPressure; +struct PressureInfo +{ + PressureInfo() + { + code = 0; + minx = maxx = 0; + peaktime = peakevents = 0; + min_pressure = max_pressure = 0; + } + PressureInfo(PressureInfo ©) { + code = copy.code; + minx = copy.minx; + maxx = copy.maxx; + peaktime = copy.peaktime; + peakevents = copy.peakevents; + min_pressure = copy.min_pressure; + max_pressure = copy.max_pressure; + times = copy.times; + events = copy.events; + chans = copy.chans; + } + + PressureInfo(ChannelID code, qint64 minx, qint64 maxx) : code(code), minx(minx), maxx(maxx) + { + times.resize(300); + } + void AddChannel(ChannelID c) + { + chans.append(c); + events[c].resize(300); + } + void AddChannels(QList & chans) + { + for (int i=0; i times; + int peaktime, peakevents; + int min_pressure, max_pressure; + + QHash > events; + QList chans; +}; + class RecalcMAP:public QRunnable { friend class MinutesAtPressure; @@ -20,8 +69,10 @@ public: explicit RecalcMAP(MinutesAtPressure * map) :map(map), m_quit(false), m_done(false) {} virtual ~RecalcMAP(); virtual void run(); + void quit(); protected: + void updateTimes(PressureInfo & info, Session * sess); MinutesAtPressure * map; volatile bool m_quit; volatile bool m_done; @@ -91,6 +142,7 @@ protected: gGraph * m_graph; RecalcMAP * m_remap; QMap times; + QMap epap_times; QList chans; QHash > events; int maxtime; @@ -99,6 +151,8 @@ protected: EventStoreType m_minpressure; EventStoreType m_maxpressure; + PressureInfo epap, ipap; + EventDataType max_mins; QMap ahis; diff --git a/sleepyhead/Graphs/gXAxis.cpp b/sleepyhead/Graphs/gXAxis.cpp index 822f8624..000830eb 100644 --- a/sleepyhead/Graphs/gXAxis.cpp +++ b/sleepyhead/Graphs/gXAxis.cpp @@ -453,3 +453,35 @@ void gXAxisDay::paint(QPainter &painter, gGraph &graph, const QRegion ®ion) painter.drawLines(lines); } + + +gXAxisPressure::gXAxisPressure(QColor col) + :Layer(NoChannel) +{ + m_line_color = col; + m_text_color = col; + m_major_color = Qt::darkGray; + m_minor_color = Qt::lightGray; + m_show_major_lines = false; + m_show_minor_lines = false; + m_show_minor_ticks = true; + m_show_major_ticks = true; +} +gXAxisPressure::~gXAxisPressure() +{ +} + +int gXAxisPressure::minimumHeight() +{ + QFontMetrics fm(*defaultfont); + int h = fm.height(); +#if defined(Q_OS_MAC) + return 9+h; +#else + return 11+h; +#endif +} + +void gXAxisPressure::paint(QPainter &painter, gGraph &graph, const QRegion ®ion) +{ +} diff --git a/sleepyhead/Graphs/gXAxis.h b/sleepyhead/Graphs/gXAxis.h index f372dfc1..80371252 100644 --- a/sleepyhead/Graphs/gXAxis.h +++ b/sleepyhead/Graphs/gXAxis.h @@ -147,4 +147,49 @@ class gXAxisDay: public Layer }; +class gXAxisPressure: public Layer +{ +public: + static const int Margin = 30; // How much room does this take up. (Bottom margin) + +public: + gXAxisPressure(QColor col = Qt::black); + virtual ~gXAxisPressure(); + + virtual void paint(QPainter &painter, gGraph &w, const QRegion ®ion); + + virtual int minimumHeight(); + + virtual Layer * Clone() { + gXAxisPressure * xaxis = new gXAxisPressure(); + Layer::CloneInto(xaxis); + CloneInto(xaxis); + return xaxis; + } + + void CloneInto(gXAxisPressure * layer) { + layer->m_show_major_ticks = m_show_major_ticks; + layer->m_show_minor_ticks = m_show_minor_ticks; + layer->m_show_major_lines = m_show_major_lines; + layer->m_show_minor_lines = m_show_minor_lines; + layer->m_major_color = m_major_color; + layer->m_minor_color = m_minor_color; + layer->m_line_color = m_line_color; + layer->m_text_color = m_text_color; + + //layer->m_image = m_image; + } + +protected: + bool m_show_major_lines; + bool m_show_minor_lines; + bool m_show_minor_ticks; + bool m_show_major_ticks; + + QColor m_line_color; + QColor m_text_color; + QColor m_major_color; + QColor m_minor_color; +}; + #endif // GXAXIS_H diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index 7c2913fd..6d2fba57 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -269,10 +269,11 @@ Daily::Daily(QWidget *parent,gGraphView * shared) pc->addPlot(CPAP_IPAPHi, square); gGraph * TAP2; - graphlist[STR_GRAPH_TAP] = TAP2 = new gGraph(STR_GRAPH_TAP, GraphView, QObject::tr("By Pressure"), QObject::tr("Statistics at Pressure"), default_height); + graphlist[STR_GRAPH_TAP] = TAP2 = new gGraph(STR_GRAPH_TAP, GraphView, QObject::tr("Time at Pressure"), QObject::tr("Time at Pressure"), default_height); MinutesAtPressure * map; TAP2->AddLayer(map = new MinutesAtPressure()); TAP2->AddLayer(new gLabelArea(map),LayerLeft,gYAxis::Margin); + TAP2->AddLayer(new gXAxisPressure(),LayerBottom,gXAxisPressure::Margin); TAP2->setBlockSelect(true); if (p_profile->general->calculateRDI()) { diff --git a/sleepyhead/main.cpp b/sleepyhead/main.cpp index 16c48baa..885dde77 100644 --- a/sleepyhead/main.cpp +++ b/sleepyhead/main.cpp @@ -85,11 +85,11 @@ void release_notes() QVBoxLayout * layout = new QVBoxLayout(&relnotes); QWebView * web = new QWebView(&relnotes); - QString welcomeMessage = ""+ - QObject::tr("

After four years in the making, this build brings SleepyHead into the final beta phase.

")+ - QObject::tr("

Things are not perfect yet, but the focus from now is putting on the finishing touches. ")+ - QObject::tr("This version brings support for the new Philips Respironics DreamStation, and older PRS1 1060P models.

")+ - "
"; + QString welcomeMessage = "" + "

"+QObject::tr("After four years in the making, this build brings SleepyHead into the final beta phase.")+"

" + "

"+QObject::tr("Things are not perfect yet, but the focus from now is putting on the finishing touches. ")+ + QObject::tr("This version brings support for the new Philips Respironics DreamStation, and older PRS1 1060P models.")+ + "

"; QFile clfile(":/docs/release_notes.html"); QString changeLog = QObject::tr("Sorry, could not locate changelog."); @@ -112,7 +112,7 @@ void release_notes() } html += "

"+QObject::tr("Sleep Well, and good luck!")+"

" - "

"+"JediMark"+"


Change log


"; + "

"+"JediMark"+"


"+QObject::tr("Change log")+"


"; html += changeLog; html += "";