Mega update: Summary demand loading, Overview summarychart rework, rxchanges caching

This commit is contained in:
Mark Watkins 2014-09-12 00:23:08 +10:00
parent 04b06a9f6d
commit 4c0b4908bc
35 changed files with 2377 additions and 435 deletions

23
README
View File

@ -3,9 +3,21 @@ SleepyHead QT port v0.9 branch
SleepyHead is cross platform, opensource sleep tracking program for reviewing CPAP and Oximetry data, SleepyHead is cross platform, opensource sleep tracking program for reviewing CPAP and Oximetry data,
which are devices used in the treatment of Sleep Disorders like Obstructive Sleep Apnea. which are devices used in the treatment of Sleep Disorders like Obstructive Sleep Apnea.
To Build: Requirements:
-------------
Qt5 SDK with webkit (opengl stuff recommended)
Linux needs libudev-dev for qserialport to compile
qmake
Building:
--------
Recommend shadow building to not cruft up the source code folder:
cd ..
mkdir build_sleepyhead
cd build_sleepyhead
qmake ../sleepyhead-code/SleepyHeadQT.pro
make make
You may need to add a -spec option to qmake to suit your platform. You may need to add a -spec option to qmake to suit your platform.
@ -13,15 +25,12 @@ Adding -j3 speeds up the make command on a dual core or greater system.
Author: Mark Watkins <jedimark@users.sourceforge.net> Author: Mark Watkins <jedimark@users.sourceforge.net>
Copyright (C)2011 Mark Watkins Copyright (C)2011-2014 Mark Watkins
Licence Stuff Licence Stuff
------------- -------------
This software is released under the GNU Public License, at a GPL version of my choosing at a later date. This software is released under the GNU Public License version 3.0
Exceptions and 3rd Party Libraries: Exceptions and 3rd Party Libraries:
Incorporates QextSerialPort. Insert New BSD license here? (Apparently PD.. need to verify)
http://code.google.com/p/qextserialport/
It uses QuaZip, by Sergey A. Tachenov, which is a C++ wrapper over Gilles Vollant's ZIP/UNZIP package.. It uses QuaZip, by Sergey A. Tachenov, which is a C++ wrapper over Gilles Vollant's ZIP/UNZIP package..
http://sourceforge.net/projects/quazip/ http://sourceforge.net/projects/quazip/

View File

@ -145,10 +145,11 @@ void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion &r
m_minx = graph.min_x; m_minx = graph.min_x;
m_maxx = graph.max_x; m_maxx = graph.max_x;
if ((m_lastminx != m_minx) || (m_lastmaxx != m_maxx)) { if (graph.printing() || ((m_lastminx != m_minx) || (m_lastmaxx != m_maxx))) {
recalculate(&graph); recalculate(&graph);
} }
m_lastminx = m_minx; m_lastminx = m_minx;
m_lastmaxx = m_maxx; m_lastmaxx = m_maxx;
@ -158,11 +159,12 @@ void MinutesAtPressure::paint(QPainter &painter, gGraph &graph, const QRegion &r
if (graph.printing()) { if (graph.printing()) {
// lock the other mutex... // lock the other mutex...
while (recalculating()) {}; // while (recalculating()) {};
recalculate(&graph); // recalculate(&graph);
while (recalculating()) {}; while (recalculating()) {};
} }
if (!painter.isActive()) return;
// Lock the stuff we need to draw // Lock the stuff we need to draw

View File

@ -152,7 +152,7 @@ void gToolTip::paint(QPainter &painter) //actually paints it.
if (xx < 0) { xx = 0; } if (xx < 0) { xx = 0; }
rect.setLeft(xx); rect.setLeft(xx);
rect.setTop(rect.y() - rect.height() / 2); rect.setTop(rect.y() - 15);
rect.setWidth(w); rect.setWidth(w);
int z = rect.x() + rect.width(); int z = rect.x() + rect.width();
@ -256,8 +256,13 @@ void gGraphView::queGraph(gGraph *g, int left, int top, int width, int height)
#endif #endif
} }
void gGraphView::trashGraphs() void gGraphView::trashGraphs(bool destroy)
{ {
if (destroy) {
for (int i=0; i< m_graphs.size(); ++i) {
delete m_graphs[i];
}
}
// Don't actually want to delete them here.. we are just borrowing the graphs // Don't actually want to delete them here.. we are just borrowing the graphs
m_graphs.clear(); m_graphs.clear();
m_graphsbyname.clear(); m_graphsbyname.clear();
@ -300,6 +305,7 @@ gGraphView::gGraphView(QWidget *parent, gGraphView *shared)
m_selected_graph = nullptr; m_selected_graph = nullptr;
m_scrollbar = nullptr; m_scrollbar = nullptr;
m_point_released = m_point_clicked = QPoint(0,0); m_point_released = m_point_clicked = QPoint(0,0);
m_showAuthorMessage = true;
horizScrollTime.start(); horizScrollTime.start();
vertScrollTime.start(); vertScrollTime.start();
@ -889,7 +895,8 @@ void gGraphView::updateScale()
if (th < h) { if (th < h) {
th -= visibleGraphs() * graphSpacer; // compensate for spacer height th -= graphSpacer;
// th -= visibleGraphs() * graphSpacer; // compensate for spacer height
m_scaleY = h / th; // less graphs than fits on screen, so scale to fit m_scaleY = h / th; // less graphs than fits on screen, so scale to fit
} else { } else {
m_scaleY = 1.0; m_scaleY = 1.0;
@ -941,6 +948,7 @@ void gGraphView::GetRXBounds(qint64 &st, qint64 &et)
void gGraphView::ResetBounds(bool refresh) //short group) void gGraphView::ResetBounds(bool refresh) //short group)
{ {
if (m_graphs.size() == 0) return;
Q_UNUSED(refresh) Q_UNUSED(refresh)
qint64 m1 = 0, m2 = 0; qint64 m1 = 0, m2 = 0;
gGraph *g = nullptr; gGraph *g = nullptr;
@ -964,7 +972,9 @@ void gGraphView::ResetBounds(bool refresh) //short group)
// } // }
// } // }
if (!g) { g = m_graphs[0]; } if (!g) {
g = m_graphs[0];
}
m_minx = g->min_x; m_minx = g->min_x;
m_maxx = g->max_x; m_maxx = g->max_x;
@ -989,7 +999,7 @@ void gGraphView::SetXBounds(qint64 minx, qint64 maxx, short group, bool refresh)
m_minx = minx; m_minx = minx;
m_maxx = maxx; m_maxx = maxx;
if (refresh) { redraw(); } if (refresh) { timedRedraw(0); }
} }
void gGraphView::updateScrollBar() void gGraphView::updateScrollBar()
@ -1269,11 +1279,15 @@ void gGraphView::paintGL()
graphs_drawn = renderGraphs(painter); graphs_drawn = renderGraphs(painter);
if (!graphs_drawn) { // No graphs drawn? show something useful :) if (!graphs_drawn) { // No graphs drawn? show something useful :)
QString txt = QObject::tr("SleepyHead is proudly brought to you by JediMark."); QString txt;
if (emptyText() == STR_Empty_Brick) { if (m_showAuthorMessage) {
txt += "\nI'm very sorry your machine doesn't record useful data to graph in Daily View :("; if (emptyText() == STR_Empty_Brick) {
txt = "\nI'm very sorry your machine doesn't record useful data to graph in Daily View :(";
} else {
// not proud of telling them their machine is a Brick.. ;)
txt = QObject::tr("SleepyHead is proudly brought to you by JediMark.");
}
} }
// int x2, y2; // int x2, y2;
// GetTextExtent(m_emptytext, x2, y2, bigfont); // GetTextExtent(m_emptytext, x2, y2, bigfont);
// int tp2, tp1; // int tp2, tp1;
@ -1872,7 +1886,7 @@ void gGraphView::populateMenu(gGraph * graph)
gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(graph,LT_LineChart)); gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(graph,LT_LineChart));
SummaryChart * sc = dynamic_cast<SummaryChart *>(findLayer(graph,LT_SummaryChart)); SummaryChart * sc = dynamic_cast<SummaryChart *>(findLayer(graph,LT_SummaryChart));
gSessionTimesChart * stg = dynamic_cast<gSessionTimesChart *>(findLayer(graph,LT_SessionTimes)); gSummaryChart * stg = dynamic_cast<gSummaryChart *>(findLayer(graph,LT_Overview));
limits_menu->clear(); limits_menu->clear();
@ -3212,7 +3226,7 @@ bool gGraphView::LoadSettings(QString title)
in.setByteOrder(QDataStream::LittleEndian); in.setByteOrder(QDataStream::LittleEndian);
quint32 t1; quint32 t1;
quint16 t2; quint16 version;
in >> t1; in >> t1;
@ -3221,9 +3235,9 @@ bool gGraphView::LoadSettings(QString title)
return false; return false;
} }
in >> t2; in >> version;
if (t2 < gvversion) { if (version < gvversion) {
qDebug() << "gGraphView" << title << "settings will be upgraded."; qDebug() << "gGraphView" << title << "settings will be upgraded.";
} }
@ -3272,15 +3286,16 @@ bool gGraphView::LoadSettings(QString title)
gGraph *g = nullptr; gGraph *g = nullptr;
if (t2 <= 2) { if (version <= 2) {
// Names were stored as translated strings, so look up title instead. continue;
g = nullptr; // // Names were stored as translated strings, so look up title instead.
for (int z=0; z<m_graphs.size(); ++z) { // g = nullptr;
if (m_graphs[z]->title() == name) { // for (int z=0; z<m_graphs.size(); ++z) {
g = m_graphs[z]; // if (m_graphs[z]->title() == name) {
break; // g = m_graphs[z];
} // break;
} // }
// }
} else { } else {
gi = m_graphsbyname.find(name); gi = m_graphsbyname.find(name);
if (gi == m_graphsbyname.end()) { if (gi == m_graphsbyname.end()) {

View File

@ -470,7 +470,7 @@ class gGraphView
void showSplitter() { m_showsplitter = true; } void showSplitter() { m_showsplitter = true; }
//! \brief Trash all graph objects listed (without destroying Graph contents) //! \brief Trash all graph objects listed (without destroying Graph contents)
void trashGraphs(); void trashGraphs(bool destroy);
//! \brief Enable or disable the Text Pixmap Caching system preference overide //! \brief Enable or disable the Text Pixmap Caching system preference overide
void setUsePixmapCache(bool b) { use_pixmap_cache = b; } void setUsePixmapCache(bool b) { use_pixmap_cache = b; }
@ -512,6 +512,9 @@ class gGraphView
void getSelectionTimes(qint64 & start, qint64 & end); void getSelectionTimes(qint64 & start, qint64 & end);
//! \brief Whether to show a little authorship message down the bottom of empty graphs.
void setShowAuthorMessage(bool b) { m_showAuthorMessage = b; }
// for profiling purposes, a count of lines drawn in a single frame // for profiling purposes, a count of lines drawn in a single frame
int lines_drawn_this_frame; int lines_drawn_this_frame;
int quads_drawn_this_frame; int quads_drawn_this_frame;
@ -563,6 +566,7 @@ class gGraphView
//! \brief Add Graph to drawing queue, mainly for the benefit of multithreaded drawing code //! \brief Add Graph to drawing queue, mainly for the benefit of multithreaded drawing code
void queGraph(gGraph *, int originX, int originY, int width, int height); void queGraph(gGraph *, int originX, int originY, int width, int height);
Day *m_day; Day *m_day;
//! \brief the list of graphs to draw this frame //! \brief the list of graphs to draw this frame
@ -648,6 +652,8 @@ class gGraphView
QAction * zoom100_action; QAction * zoom100_action;
bool m_showAuthorMessage;
signals: signals:
void updateCurrentTime(double); void updateCurrentTime(double);
void updateRange(double,double); void updateRange(double,double);
@ -670,6 +676,7 @@ class gGraphView
ResetBounds(true); ResetBounds(true);
} }
bool hasSnapshots(); bool hasSnapshots();
void togglePin(); void togglePin();
@ -678,6 +685,7 @@ protected slots:
void onPlotsClicked(QAction *); void onPlotsClicked(QAction *);
void onOverlaysClicked(QAction *); void onOverlaysClicked(QAction *);
void onSnapshotGraphToggle(); void onSnapshotGraphToggle();
}; };
#endif // GGRAPHVIEW_H #endif // GGRAPHVIEW_H

View File

@ -248,7 +248,7 @@ void gLineOverlayBar::paint(QPainter &painter, gGraph &w, const QRegion &region)
// painter.drawLine(rect.x(), bottom, rect.x()+d1, bottom); // painter.drawLine(rect.x(), bottom, rect.x()+d1, bottom);
// painter.drawLine(rect.x(), top, rect.x(), bottom); // painter.drawLine(rect.x(), top, rect.x(), bottom);
// col = QColor("gold"); // col = COLOR_Gold;
hover = true; hover = true;
painter.setPen(QPen(col,3)); painter.setPen(QPen(col,3));
} else { } else {

File diff suppressed because it is too large Load Diff

View File

@ -9,13 +9,15 @@
#ifndef GSESSIONTIMESCHART_H #ifndef GSESSIONTIMESCHART_H
#define GSESSIONTIMESCHART_H #define GSESSIONTIMESCHART_H
#include <SleepLib/day.h> #include "SleepLib/day.h"
#include "SleepLib/profiles.h"
#include "gGraphView.h" #include "gGraphView.h"
struct TimeSpan struct TimeSpan
{ {
public: public:
TimeSpan(): begin(0), end(0) {} TimeSpan():begin(0), end(0) {}
TimeSpan(float b, float e) : begin(b), end(e) {} TimeSpan(float b, float e) : begin(b), end(e) {}
TimeSpan(const TimeSpan & copy) { TimeSpan(const TimeSpan & copy) {
begin = copy.begin; begin = copy.begin;
@ -26,31 +28,94 @@ public:
float end; float end;
}; };
/*! \class gSessionTimesChart struct SummaryCalcItem {
\brief Displays a summary of session times SummaryCalcItem() {
*/ code = 0;
class gSessionTimesChart : public Layer type = ST_CNT;
}
SummaryCalcItem(const SummaryCalcItem & copy) {
code = copy.code;
type = copy.type;
}
SummaryCalcItem(ChannelID code, SummaryType type)
:code(code), type(type) {}
ChannelID code;
SummaryType type;
};
struct SummaryChartSlice {
SummaryChartSlice() {
code = 0;
height = 0;
value = 0;
name = ST_CNT;
}
SummaryChartSlice(const SummaryChartSlice & copy) {
code = copy.code;
value = copy.value;
height = copy.height;
name = copy.name;
color = copy.color;
}
SummaryChartSlice(ChannelID code, EventDataType value, EventDataType height, QString name, QColor color)
:code(code), value(value), height(height), name(name), color(color) {}
ChannelID code;
EventDataType value;
EventDataType height;
QString name;
QColor color;
};
class gSummaryChart : public Layer
{ {
public: public:
gSessionTimesChart(QString label, MachineType machtype); gSummaryChart(QString label, MachineType machtype);
~gSessionTimesChart(); gSummaryChart(ChannelID code, MachineType machtype);
virtual ~gSummaryChart();
//! \brief Renders the graph to the QPainter object //! \brief Renders the graph to the QPainter object
virtual void paint(QPainter &painter, gGraph &w, const QRegion &region); virtual void paint(QPainter &, gGraph &, const QRegion &);
//! \brief Precalculation code prior to drawing. Day object is not needed here, it's just here for Layer compatability. //! \brief Called whenever data model changes underneath. Day object is not needed here, it's just here for Layer compatability.
virtual void SetDay(Day *day = nullptr); virtual void SetDay(Day *day = nullptr);
//! \brief Returns true if no data was found for this day during SetDay //! \brief Returns true if no data was found for this day during SetDay
virtual bool isEmpty() { return m_empty; } virtual bool isEmpty() { return m_empty; }
//! \brief Deselect highlighting (the gold bar) virtual void populate(Day *, int idx);
virtual void deselect() {
hl_day = -1; //! \brief Override to setup custom stuff before main loop
virtual void preCalc() {}
//! \brief Override to call stuff in main loop
virtual void customCalc(Day *, QList<SummaryChartSlice> &) {}
//! \brief Override to call stuff after draw is complete
virtual void afterDraw(QPainter &, gGraph &, QRect) {}
//! \brief Return any extra data to show beneath the date in the hover over tooltip
virtual QString tooltipData(Day *, int);
void addCalc(ChannelID code, SummaryType type) { calcitems.append(SummaryCalcItem(code, type)); }
virtual Layer * Clone() {
gSummaryChart * sc = new gSummaryChart(m_label, m_machtype);
Layer::CloneInto(sc);
CloneInto(sc);
return sc;
} }
//! \brief Returns true if currently selected.. void CloneInto(gSummaryChart * layer) {
virtual bool isSelected() { return hl_day >= 0; } layer->m_empty = m_empty;
layer->firstday = firstday;
layer->lastday = lastday;
layer->cache = cache;
layer->calcitems = calcitems;
layer->expected_slices = expected_slices;
layer->nousedays = nousedays;
layer->totaldays = totaldays;
layer->peak_value = peak_value;
}
protected: protected:
//! \brief Key was pressed that effects this layer //! \brief Key was pressed that effects this layer
@ -73,7 +138,192 @@ protected:
float tz_hours; float tz_hours;
QDate firstday; QDate firstday;
QDate lastday; QDate lastday;
QMap<quint32, QList<TimeSpan> > sessiontimes;
static QMap<QDate, int> dayindex;
static QList<Day *> daylist;
QHash<int, QList<SummaryChartSlice> > cache;
QList<SummaryCalcItem> calcitems;
int expected_slices;
int nousedays;
int totaldays;
EventDataType peak_value;
EventDataType min_value;
};
/*! \class gSessionTimesChart
\brief Displays a summary of session times
*/
class gSessionTimesChart : public gSummaryChart
{
public:
gSessionTimesChart()
:gSummaryChart("SessionTimes", MT_CPAP) {}
virtual ~gSessionTimesChart() {}
virtual void SetDay(Day * day = nullptr) {
gSummaryChart::SetDay(day);
split = p_profile->session->daySplitTime();
m_miny = 0;
m_maxy = 28;
}
virtual void preCalc() {
num_slices = 0;
num_days = 0;
total_length = 0;
}
virtual void customCalc(Day *, QList<SummaryChartSlice> & slices) {
int size = slices.size();
num_slices += size;
for (int i=0; i<size; ++i) {
const SummaryChartSlice & slice = slices.at(i);
total_length += slice.height;
}
num_days++;
}
virtual void afterDraw(QPainter &, gGraph &, QRect);
//! \brief Renders the graph to the QPainter object
virtual void paint(QPainter &painter, gGraph &graph, const QRegion &region);
virtual Layer * Clone() {
gSessionTimesChart * sc = new gSessionTimesChart();
gSummaryChart::CloneInto(sc);
CloneInto(sc);
return sc;
}
void CloneInto(gSessionTimesChart * layer) {
layer->split = split;
}
QTime split;
int num_slices;
int num_days;
int total_slices;
double total_length;
};
class gUsageChart : public gSummaryChart
{
public:
gUsageChart()
:gSummaryChart("Usage", MT_CPAP) {
addCalc(NoChannel, ST_HOURS);
}
virtual ~gUsageChart() {}
virtual void preCalc();
virtual void customCalc(Day *, QList<SummaryChartSlice> &);
virtual void afterDraw(QPainter &, gGraph &, QRect);
virtual void populate(Day *day, int idx);
virtual QString tooltipData(Day * day, int);
virtual Layer * Clone() {
gUsageChart * sc = new gUsageChart();
gSummaryChart::CloneInto(sc);
CloneInto(sc);
return sc;
}
void CloneInto(gUsageChart * layer) {
layer->incompdays = incompdays;
layer->compliance_threshold = compliance_threshold;
}
private:
int incompdays;
EventDataType compliance_threshold;
double totalhours;
int totaldays;
};
class gAHIChart : public gSummaryChart
{
public:
gAHIChart()
:gSummaryChart("AHIChart", MT_CPAP) {
channels.append(CPAP_ClearAirway);
channels.append(CPAP_Obstructive);
channels.append(CPAP_Apnea);
channels.append(CPAP_Hypopnea);
if (p_profile->general->calculateRDI())
channels.append(CPAP_RERA);
num_channels = channels.size();
}
virtual ~gAHIChart() {}
virtual void preCalc();
virtual void customCalc(Day *, QList<SummaryChartSlice> &);
virtual void afterDraw(QPainter &, gGraph &, QRect);
virtual void populate(Day *, int idx);
virtual QString tooltipData(Day * day, int);
virtual Layer * Clone() {
gAHIChart * sc = new gAHIChart();
gSummaryChart::CloneInto(sc);
CloneInto(sc);
return sc;
}
void CloneInto(gAHIChart * layer) {
layer->channels = channels;
layer->num_channels = num_channels;
layer->indices = indices;
layer->ahi_total = ahi_total;
layer->calc_cnt = calc_cnt;
}
QList<ChannelID> channels;
int num_channels;
QHash<ChannelID, double> indices;
double ahi_total;
double total_hours;
int calc_cnt;
};
class gPressureChart : public gSummaryChart
{
public:
gPressureChart()
:gSummaryChart("Pressure", MT_CPAP) {
}
virtual ~gPressureChart() {}
virtual void SetDay(Day * day = nullptr) {
gSummaryChart::SetDay(day);
m_miny = 0;
m_maxy = 24;
}
virtual Layer * Clone() {
gPressureChart * sc = new gPressureChart();
gSummaryChart::CloneInto(sc);
return sc;
}
virtual void populate(Day * day, int idx);
virtual QString tooltipData(Day * day, int idx) {
return day->getCPAPMode() + "\n" + day->getPressureSettings() + gSummaryChart::tooltipData(day, idx);
}
}; };
#endif // GSESSIONTIMESCHART_H #endif // GSESSIONTIMESCHART_H

View File

@ -651,7 +651,7 @@ void SummaryChart::paint(QPainter &painter, gGraph &w, const QRegion &region)
col = summaryColor; col = summaryColor;
} }
if (zd == hl_day) { if (zd == hl_day) {
col = QColor("gold"); col = COLOR_Gold;
} }
QColor col1 = col; QColor col1 = col;
@ -749,7 +749,7 @@ void SummaryChart::paint(QPainter &painter, gGraph &w, const QRegion &region)
} }
if (zd == hl_day) { if (zd == hl_day) {
col = QColor("gold"); col = COLOR_Gold;
} }
//if (!tmp) continue; //if (!tmp) continue;

View File

@ -117,21 +117,18 @@ void gXAxis::paint(QPainter &painter, gGraph &w, const QRegion &region)
// Allow zoom // Allow zoom
minx = w.min_x; minx = w.min_x;
maxx = w.max_x; maxx = w.max_x;
} }
int days = ceil(double(maxx-minx) / 86400000.0); int days = ceil(double(maxx-minx) / 86400000.0);
bool buttuglydaysteps = !p_profile->appearance->animations(); if (m_roundDays) {
if (buttuglydaysteps) { minx = floor(double(minx)/86400000.0);
if (m_roundDays && (days >= 1)) { minx *= 86400000L;
minx = floor(double(minx)/86400000.0);
minx *= 86400000L;
maxx = minx + 86400000L * qint64(days); maxx = minx + 86400000L * qint64(days);
}
} }
// duration of graph display window in milliseconds. // duration of graph display window in milliseconds.
qint64 xx = maxx - minx; qint64 xx = maxx - minx;
@ -357,3 +354,96 @@ void gXAxis::paint(QPainter &painter, gGraph &w, const QRegion &region)
} }
} }
gXAxisDay::gXAxisDay(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;
}
gXAxisDay::~gXAxisDay()
{
}
int gXAxisDay::minimumHeight()
{
QFontMetrics fm(*defaultfont);
int h = fm.height();
#if defined(Q_OS_MAC)
return 9+h;
#else
return 11+h;
#endif
}
void gXAxisDay::paint(QPainter &painter, gGraph &graph, const QRegion &region)
{
float left = region.boundingRect().left();
float top = region.boundingRect().top();
float width = region.boundingRect().width();
float height = region.boundingRect().height();
QString months[] = {
QObject::tr("Jan"), QObject::tr("Feb"), QObject::tr("Mar"), QObject::tr("Apr"), QObject::tr("May"), QObject::tr("Jun"),
QObject::tr("Jul"), QObject::tr("Aug"), QObject::tr("Sep"), QObject::tr("Oct"), QObject::tr("Nov"), QObject::tr("Dec")
};
qint64 minx;
qint64 maxx;
minx = graph.min_x;
maxx = graph.max_x;
QDateTime date2 = QDateTime::fromMSecsSinceEpoch(minx);
QDateTime enddate2 = QDateTime::fromMSecsSinceEpoch(maxx);
QDate date = date2.date();
// QDate enddate = enddate2.date();
int days = ceil(double(maxx - minx) / 86400000.0);
float barw = width / float(days);
qint64 xx = maxx - minx;
// shouldn't really be negative, but this is safer than an assert
if (xx <= 0) {
return;
}
float lastx = left;
float y1 = top;
QString fd = "Mjj 00";
int x,y;
GetTextExtent(fd, x, y);
float xpos = (barw / 2.0) - (float(x) / 2.0);
float lastxpos = 0;
QVector<QLine> lines;
for (int i=0; i < days; i++) {
if ((lastx + barw) > (left + width + 1))
break;
QString tmpstr = QString("%1 %2").arg(months[date.month() - 1]).arg(date.day(), 2, 10, QChar('0'));
float x1 = lastx + xpos;
//lines.append(QLine(lastx, top, lastx, top+6));
if (x1 > (lastxpos + x + 8*graph.printScaleX())) {
graph.renderText(tmpstr, x1, y1 + y + 8);
lastxpos = x1;
lines.append(QLine(lastx+barw/2, top, lastx+barw/2, top+6));
}
lastx = lastx + barw;
date = date.addDays(1);
}
painter.setPen(QPen(Qt::black,1));
painter.drawLines(lines);
}

View File

@ -89,4 +89,62 @@ class gXAxis: public Layer
bool m_roundDays; bool m_roundDays;
}; };
class gXAxisDay: public Layer
{
public:
static const int Margin = 30; // How much room does this take up. (Bottom margin)
public:
gXAxisDay(QColor col = Qt::black);
virtual ~gXAxisDay();
virtual void paint(QPainter &painter, gGraph &w, const QRegion &region);
void SetShowMinorLines(bool b) { m_show_minor_lines = b; }
void SetShowMajorLines(bool b) { m_show_major_lines = b; }
bool ShowMinorLines() { return m_show_minor_lines; }
bool ShowMajorLines() { return m_show_major_lines; }
void SetShowMinorTicks(bool b) { m_show_minor_ticks = b; }
void SetShowMajorTicks(bool b) { m_show_major_ticks = b; }
bool ShowMinorTicks() { return m_show_minor_ticks; }
bool ShowMajorTicks() { return m_show_major_ticks; }
//! \brief Returns the minimum height needed to fit
virtual int minimumHeight();
virtual Layer * Clone() {
gXAxisDay * xaxis = new gXAxisDay();
Layer::CloneInto(xaxis);
CloneInto(xaxis);
return xaxis;
}
void CloneInto(gXAxisDay * 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;
QImage m_image;
};
#endif // GXAXIS_H #endif // GXAXIS_H

View File

@ -26,7 +26,7 @@ enum LayerPosition { LayerLeft, LayerRight, LayerTop, LayerBottom, LayerCenter,
enum ToolTipAlignment { TT_AlignCenter, TT_AlignLeft, TT_AlignRight }; enum ToolTipAlignment { TT_AlignCenter, TT_AlignLeft, TT_AlignRight };
enum LayerType { LT_Other = 0, LT_LineChart, LT_SummaryChart, LT_EventFlags, LT_Spacer, LT_SessionTimes }; enum LayerType { LT_Other = 0, LT_LineChart, LT_SummaryChart, LT_EventFlags, LT_Spacer, LT_Overview };
/*! \class Layer /*! \class Layer
\brief The base component for all individual Graph layers \brief The base component for all individual Graph layers

View File

@ -18,6 +18,10 @@
Day::Day() Day::Day()
{ {
d_firstsession = true; d_firstsession = true;
d_summaries_open = false;
d_events_open = false;
d_invalidate = true;
} }
Day::~Day() Day::~Day()
{ {
@ -26,6 +30,22 @@ Day::~Day()
} }
} }
void Day::updateCPAPCache()
{
d_count.clear();
d_sum.clear();
OpenSummary();
QList<ChannelID> channels = getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN);
int num_channels = channels.size();
for (int i=0; i< num_channels; ++i) {
ChannelID code = channels.at(i);
d_count[code] = count(code);
d_sum[code] = count(code);
d_machhours[MT_CPAP] = hours(MT_CPAP);
}
}
Session * Day::firstSession(MachineType type) Session * Day::firstSession(MachineType type)
{ {
for (int i=0; i<sessions.size(); i++) { for (int i=0; i<sessions.size(); i++) {
@ -40,6 +60,7 @@ Session * Day::firstSession(MachineType type)
bool Day::addMachine(Machine *mach) bool Day::addMachine(Machine *mach)
{ {
invalidate();
if (!machines.contains(mach->type())) { if (!machines.contains(mach->type())) {
machines[mach->type()] = mach; machines[mach->type()] = mach;
return true; return true;
@ -86,6 +107,7 @@ Session *Day::find(SessionID sessid)
void Day::addSession(Session *s) void Day::addSession(Session *s)
{ {
invalidate();
Q_ASSERT(s!=nullptr); Q_ASSERT(s!=nullptr);
QHash<MachineType, Machine *>::iterator mi = machines.find(s->machine()->type()); QHash<MachineType, Machine *>::iterator mi = machines.find(s->machine()->type());
@ -435,6 +457,9 @@ EventDataType Day::percentile(ChannelID code, EventDataType percentile)
return v1; return v1;
} }
if (valcnt.size() == 1) {
return valcnt[0].value;
}
v2 = valcnt[k + 1].value; v2 = valcnt[k + 1].value;
w2 = valcnt[k + 1].count; w2 = valcnt[k + 1].count;
sum2 = sum1 + w2; sum2 = sum1 + w2;
@ -1211,17 +1236,31 @@ bool Day::channelHasData(ChannelID id)
void Day::OpenEvents() void Day::OpenEvents()
{ {
if (d_events_open)
return;
Q_FOREACH(Session * session, sessions) { Q_FOREACH(Session * session, sessions) {
if (session->machine()->type() != MT_JOURNAL) if (session->machine()->type() != MT_JOURNAL)
session->OpenEvents(); session->OpenEvents();
} }
d_events_open = true;
} }
void Day::OpenSummary()
{
if (d_summaries_open) return;
Q_FOREACH(Session * session, sessions) {
session->LoadSummary();
}
d_summaries_open = true;
}
void Day::CloseEvents() void Day::CloseEvents()
{ {
Q_FOREACH(Session * session, sessions) { Q_FOREACH(Session * session, sessions) {
session->TrashEvents(); session->TrashEvents();
} }
d_events_open = false;
} }
QList<ChannelID> Day::getSortedMachineChannels(MachineType type, quint32 chantype) QList<ChannelID> Day::getSortedMachineChannels(MachineType type, quint32 chantype)

View File

@ -159,8 +159,18 @@ class Day
bool hasEnabledSessions(); bool hasEnabledSessions();
//! \brief Return the total time in decimal hours for this day //! \brief Return the total time in decimal hours for this day
EventDataType hours() { return double(total_time()) / 3600000.0; } EventDataType hours() {
EventDataType hours(MachineType type) { return double(total_time(type)) / 3600000.0; } if (!d_invalidate) return d_hours;
d_invalidate = false;
return d_hours = double(total_time()) / 3600000.0;
}
EventDataType hours(MachineType type) {
QHash<MachineType, EventDataType>::iterator it = d_machhours.find(type);
if (it == d_machhours.end()) {
return d_machhours[type] = double(total_time(type)) / 3600000.0;
}
return it.value();
}
//! \brief Return the session indexed by i //! \brief Return the session indexed by i
Session *operator [](int i) { return sessions[i]; } Session *operator [](int i) { return sessions[i]; }
@ -183,6 +193,8 @@ class Day
//! \brief Loads all Events files for this Days Sessions //! \brief Loads all Events files for this Days Sessions
void OpenEvents(); void OpenEvents();
void OpenSummary();
//! \brief Closes all Events files for this Days Sessions //! \brief Closes all Events files for this Days Sessions
void CloseEvents(); void CloseEvents();
@ -218,21 +230,21 @@ class Day
//! \brief Calculate AHI (Apnea Hypopnea Index) //! \brief Calculate AHI (Apnea Hypopnea Index)
EventDataType calcAHI() { EventDataType calcAHI() {
EventDataType c = count(CPAP_Hypopnea) + count(CPAP_Obstructive) + count(CPAP_Apnea) + count(CPAP_ClearAirway); EventDataType c = count(CPAP_Hypopnea) + count(CPAP_Obstructive) + count(CPAP_Apnea) + count(CPAP_ClearAirway);
EventDataType minutes = hours() * 60.0; EventDataType minutes = hours(MT_CPAP) * 60.0;
return (c * 60.0) / minutes; return (c * 60.0) / minutes;
} }
//! \brief Calculate RDI (Respiratory Disturbance Index) //! \brief Calculate RDI (Respiratory Disturbance Index)
EventDataType calcRDI() { EventDataType calcRDI() {
EventDataType c = count(CPAP_Hypopnea) + count(CPAP_Obstructive) + count(CPAP_Apnea) + count(CPAP_ClearAirway) + count(CPAP_RERA); EventDataType c = count(CPAP_Hypopnea) + count(CPAP_Obstructive) + count(CPAP_Apnea) + count(CPAP_ClearAirway) + count(CPAP_RERA);
EventDataType minutes = hours() * 60.0; EventDataType minutes = hours(MT_CPAP) * 60.0;
return (c * 60.0) / minutes; return (c * 60.0) / minutes;
} }
//! \brief Percent of night for specified channel //! \brief Percent of night for specified channel
EventDataType calcPON(ChannelID code) { EventDataType calcPON(ChannelID code) {
EventDataType c = sum(code); EventDataType c = sum(code);
EventDataType minutes = hours() * 60.0; EventDataType minutes = hours(MT_CPAP) * 60.0;
return (100.0 / minutes) * (c / 60.0); return (100.0 / minutes) * (c / 60.0);
} }
@ -240,7 +252,7 @@ class Day
//! \brief Calculate index (count per hour) for specified channel //! \brief Calculate index (count per hour) for specified channel
EventDataType calcIdx(ChannelID code) { EventDataType calcIdx(ChannelID code) {
EventDataType c = count(code); EventDataType c = count(code);
EventDataType minutes = hours() * 60.0; EventDataType minutes = hours(MT_CPAP) * 60.0;
return (c * 60.0) / minutes; return (c * 60.0) / minutes;
} }
@ -248,7 +260,7 @@ class Day
//! \brief SleepyyHead Events Index, AHI combined with SleepyHead detected events.. :) //! \brief SleepyyHead Events Index, AHI combined with SleepyHead detected events.. :)
EventDataType calcSHEI() { EventDataType calcSHEI() {
EventDataType c = count(CPAP_Hypopnea) + count(CPAP_Obstructive) + count(CPAP_Apnea) + count(CPAP_ClearAirway) + count(CPAP_UserFlag1) + count(CPAP_UserFlag2); EventDataType c = count(CPAP_Hypopnea) + count(CPAP_Obstructive) + count(CPAP_Apnea) + count(CPAP_ClearAirway) + count(CPAP_UserFlag1) + count(CPAP_UserFlag2);
EventDataType minutes = hours() * 60.0; EventDataType minutes = hours(MT_CPAP) * 60.0;
return (c * 60.0) / minutes; return (c * 60.0) / minutes;
} }
//! \brief Total duration of all Apnea/Hypopnea events in seconds, //! \brief Total duration of all Apnea/Hypopnea events in seconds,
@ -279,6 +291,13 @@ class Day
void decUseCounter() { d_useCounter--; if (d_useCounter<0) d_useCounter = 0; } void decUseCounter() { d_useCounter--; if (d_useCounter<0) d_useCounter = 0; }
int useCounter() { return d_useCounter; } int useCounter() { return d_useCounter; }
void invalidate() {
d_invalidate = true;
d_machhours.clear();
}
void updateCPAPCache();
protected: protected:
@ -288,6 +307,13 @@ class Day
private: private:
bool d_firstsession; bool d_firstsession;
int d_useCounter; int d_useCounter;
bool d_summaries_open;
bool d_events_open;
float d_hours;
QHash<MachineType, EventDataType> d_machhours;
QHash<ChannelID, long> d_count;
QHash<ChannelID, double> d_sum;
bool d_invalidate;
}; };

View File

@ -2827,8 +2827,9 @@ void ResInitModelMap()
resmed_codes[CPAP_PSMin].push_back("Min PS"); resmed_codes[CPAP_PSMin].push_back("Min PS");
resmed_codes[CPAP_PSMax].push_back("Max PS"); resmed_codes[CPAP_PSMax].push_back("Max PS");
resmed_codes[CPAP_Leak].push_back("Leak"); resmed_codes[CPAP_Leak].push_back("Leak"); // Leak Leck Lekk Läck Fuites
resmed_codes[CPAP_Leak].push_back("Leck"); resmed_codes[CPAP_Leak].push_back("Leck");
resmed_codes[CPAP_Leak].push_back("Fuites");
resmed_codes[CPAP_Leak].push_back("\xE6\xBC\x8F\xE6\xB0\x94"); resmed_codes[CPAP_Leak].push_back("\xE6\xBC\x8F\xE6\xB0\x94");
resmed_codes[CPAP_Leak].push_back("Lekk"); resmed_codes[CPAP_Leak].push_back("Lekk");
@ -2849,6 +2850,7 @@ void ResInitModelMap()
resmed_codes[CPAP_TgMV].push_back("TgMV"); resmed_codes[CPAP_TgMV].push_back("TgMV");
resmed_codes[OXI_Pulse].push_back("Pulse"); resmed_codes[OXI_Pulse].push_back("Pulse");
resmed_codes[OXI_Pulse].push_back("Puls"); resmed_codes[OXI_Pulse].push_back("Puls");
resmed_codes[OXI_Pulse].push_back("Pouls");
resmed_codes[OXI_Pulse].push_back("Pols"); resmed_codes[OXI_Pulse].push_back("Pols");
resmed_codes[OXI_SPO2].push_back("SpO2"); resmed_codes[OXI_SPO2].push_back("SpO2");
resmed_codes[CPAP_Obstructive].push_back("Obstructive apnea"); resmed_codes[CPAP_Obstructive].push_back("Obstructive apnea");

View File

@ -75,7 +75,7 @@ Session *Machine::SessionExists(SessionID session)
} }
} }
const quint16 sessinfo_version = 0; const quint16 sessinfo_version = 1;
bool Machine::saveSessionInfo() bool Machine::saveSessionInfo()
{ {
@ -90,6 +90,8 @@ bool Machine::saveSessionInfo()
out << filetype_sessenabled; out << filetype_sessenabled;
out << sessinfo_version; out << sessinfo_version;
out << m_availableChannels;
QHash<SessionID, Session *>::iterator s; QHash<SessionID, Session *>::iterator s;
out << (int)sessionlist.size(); out << (int)sessionlist.size();
@ -131,6 +133,10 @@ bool Machine::loadSessionInfo()
in >> ft16; in >> ft16;
in >> version; in >> version;
if (version >= 1) {
in >> m_availableChannels;
}
int size; int size;
in >> size; in >> size;
@ -184,8 +190,6 @@ QDate Machine::pickDate(qint64 first)
bool Machine::AddSession(Session *s) bool Machine::AddSession(Session *s)
{ {
invalidateCache();
Q_ASSERT(s != nullptr); Q_ASSERT(s != nullptr);
Q_ASSERT(p_profile); Q_ASSERT(p_profile);
Q_ASSERT(p_profile->isOpen()); Q_ASSERT(p_profile->isOpen());
@ -388,7 +392,10 @@ bool Machine::Purge(int secret)
QFile impfile(getDataPath()+"/imported_files.csv"); QFile impfile(getDataPath()+"/imported_files.csv");
impfile.remove(); impfile.remove();
QFile sumfile(getDataPath()+"Summaries.dat"); QFile rxcache(p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" ));
rxcache.remove();
QFile sumfile(getDataPath()+"Summaries.xml");
sumfile.remove(); sumfile.remove();
// Create a copy of the list so the hash can be manipulated // Create a copy of the list so the hash can be manipulated
@ -413,6 +420,10 @@ bool Machine::Purge(int secret)
QDir evdir(eventspath); QDir evdir(eventspath);
evdir.removeRecursively(); evdir.removeRecursively();
QString summariespath = getSummariesPath();
QDir sumdir(summariespath);
sumdir.removeRecursively();
// Clean up any straggling files (like from short sessions not being loaded...) // Clean up any straggling files (like from short sessions not being loaded...)
dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks); dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks);
@ -595,10 +606,10 @@ bool Machine::Load()
if (!ok) { continue; } if (!ok) { continue; }
QString str = summarypath+filename;
Session *sess = new Session(this, sessid); Session *sess = new Session(this, sessid);
if (sess->LoadSummary(str)) { // Forced to load it, because know nothing about this session..
if (sess->LoadSummary()) {
AddSession(sess); AddSession(sess);
} else { } else {
qWarning() << "Error loading summary file" << filename; qWarning() << "Error loading summary file" << filename;
@ -854,26 +865,25 @@ bool Machine::LoadSummary(bool everything)
bool s_ok; bool s_ok;
QString sumpath = getSummariesPath();
QDomNodeList sessionlist = root.childNodes(); QDomNodeList sessionlist = root.childNodes();
int size = sessionlist.size(); int size = sessionlist.size();
for (int s=0; s < size; ++s) { for (int s=0; s < size; ++s) {
node = sessionlist.at(s); node = sessionlist.at(s);
QDomElement e = node.toElement(); QDomElement e = node.toElement();
SessionID sessid = e.attribute("id", "0").toLong(&s_ok); SessionID sessid = e.attribute("id", "0").toLong(&s_ok);
qint64 first = e.attribute("first", 0).toLongLong(); qint64 first = e.attribute("first", "0").toLongLong();
qint64 last = e.attribute("last", 0).toLongLong(); qint64 last = e.attribute("last", "0").toLongLong();
bool enabled = e.attribute("enabled", "1").toInt() == 1;
bool events = e.attribute("events", "1").toInt() == 1;
if (s_ok) { if (s_ok) {
Session * sess = new Session(this, sessid); Session * sess = new Session(this, sessid);
QString filename = sumpath + QString().sprintf("%08lx.000", sessid); sess->really_set_first(first);
if (sess->LoadSummary(filename)) { sess->really_set_last(last);
AddSession(sess); sess->setEnabled(enabled);
} else { sess->setSummaryOnly(!events);
delete sess; AddSession(sess);
} // sess->LoadSummary();
} }
} }
@ -960,43 +970,29 @@ bool Machine::Save()
return true; return true;
} }
void Machine::invalidateCache() void Machine::updateChannels(Session * sess)
{ {
availableCache.clear(); int size = sess->m_availableChannels.size();
for (int i=0; i < size; ++i) {
ChannelID code = sess->m_availableChannels.at(i);
m_availableChannels[code] = true;
}
} }
QList<ChannelID> & Machine::availableChannels(quint32 chantype) QList<ChannelID> Machine::availableChannels(quint32 chantype)
{ {
QHash<quint32, QList<ChannelID> >::iterator ac = availableCache.find(chantype); QList<ChannelID> list;
if (ac != availableCache.end()) {
return ac.value();
} QHash<ChannelID, bool>::iterator end = m_availableChannels.end();
QHash<ChannelID, bool>::iterator it;
QHash<ChannelID, int> chanhash; for (it = m_availableChannels.begin(); it != end; ++it) {
ChannelID code = it.key();
// look through the daylist and return a list of available channels for this machine const schema::Channel & chan = schema::channel[code];
QMap<QDate, Day *>::iterator dit; if (chan.type() & chantype) {
QMap<QDate, Day *>::iterator day_end = day.end(); list.push_back(code);
for (dit = day.begin(); dit != day_end; ++dit) {
QList<Session *>::iterator sess_end = dit.value()->end();
for (QList<Session *>::iterator sit = dit.value()->begin(); sit != sess_end; ++sit) {
Session * sess = (*sit);
if (sess->machine() != this) continue;
int size = sess->m_availableChannels.size();
for (int i=0; i < size; ++i) {
ChannelID code = sess->m_availableChannels.at(i);
QHash<ChannelID, schema::Channel *>::iterator ch = schema::channel.channels.find(code);
schema::Channel * chan = ch.value();
if (chan->type() & chantype) {
chanhash[code]++;
}
}
} }
} }
return availableCache[chantype] = chanhash.keys(); return list;
} }

View File

@ -103,6 +103,10 @@ class Machine
bool unlinkDay(Day * day); bool unlinkDay(Day * day);
inline bool hasChannel(ChannelID code) {
return m_availableChannels.contains(code);
}
//! \brief Contains a secondary index of day data, containing just this machines sessions //! \brief Contains a secondary index of day data, containing just this machines sessions
QMap<QDate, Day *> day; QMap<QDate, Day *> day;
@ -202,8 +206,7 @@ class Machine
void setLoaderName(QString value); void setLoaderName(QString value);
QHash<quint32, QList<ChannelID> > availableCache; QList<ChannelID> availableChannels(quint32 chantype);
QList<ChannelID> & availableChannels(quint32 chantype);
MachineLoader * loader() { return m_loader; } MachineLoader * loader() { return m_loader; }
@ -215,6 +218,8 @@ class Machine
void setInfo(MachineInfo inf); void setInfo(MachineInfo inf);
const MachineInfo getInfo() { return info; } const MachineInfo getInfo() { return info; }
void updateChannels(Session * sess);
protected: protected:
MachineInfo info; MachineInfo info;
QDate firstday, lastday; QDate firstday, lastday;
@ -235,7 +240,7 @@ class Machine
QList<ImportTask *> m_tasklist; QList<ImportTask *> m_tasklist;
void invalidateCache(); QHash<ChannelID, bool> m_availableChannels;
}; };

View File

@ -43,7 +43,7 @@ qint64 timezoneOffset();
/*! \enum SummaryType /*! \enum SummaryType
\brief Calculation/Display method to select from dealing with summary information \brief Calculation/Display method to select from dealing with summary information
*/ */
enum SummaryType { ST_CNT, ST_SUM, ST_AVG, ST_WAVG, ST_PERC, ST_90P, ST_MIN, ST_MAX, ST_CPH, ST_SPH, ST_FIRST, ST_LAST, ST_HOURS, ST_SESSIONS, ST_SETMIN, ST_SETAVG, ST_SETMAX, ST_SETWAVG, ST_SETSUM, ST_SESSIONID, ST_DATE }; enum SummaryType { ST_CNT, ST_SUM, ST_AVG, ST_WAVG, ST_PERC, ST_90P, ST_MIN, ST_MAX, ST_MID, ST_CPH, ST_SPH, ST_FIRST, ST_LAST, ST_HOURS, ST_SESSIONS, ST_SETMIN, ST_SETAVG, ST_SETMAX, ST_SETWAVG, ST_SETSUM, ST_SESSIONID, ST_DATE };
/*! \enum MachineType /*! \enum MachineType
\brief Generalized type of a machine \brief Generalized type of a machine

View File

@ -72,6 +72,19 @@ void MachineLoader::removeMachine(Machine * m)
} }
Machine * MachineLoader::lookupMachine(QString serial)
{
QHash<QString, QHash<QString, Machine *> >::iterator mlit = MachineList.find(loaderName());
if (mlit != MachineList.end()) {
QHash<QString, Machine *>::iterator mit = mlit.value().find(serial);
if (mit != mlit.value().end()) {
return mit.value();
}
}
return nullptr;
}
Machine * MachineLoader::CreateMachine(MachineInfo info, MachineID id) Machine * MachineLoader::CreateMachine(MachineInfo info, MachineID id)
{ {
Q_ASSERT(p_profile != nullptr); Q_ASSERT(p_profile != nullptr);

View File

@ -48,6 +48,7 @@ class MachineLoader: public QObject
virtual int Version() = 0; virtual int Version() = 0;
static Machine * CreateMachine(MachineInfo info, MachineID id = 0); static Machine * CreateMachine(MachineInfo info, MachineID id = 0);
Machine * lookupMachine(QString serial);
// !\\brief Used internally by loaders, override to return base MachineInfo record // !\\brief Used internally by loaders, override to return base MachineInfo record
virtual MachineInfo newInfo() { return MachineInfo(); } virtual MachineInfo newInfo() { return MachineInfo(); }

View File

@ -614,14 +614,36 @@ Day *Profile::GetGoodDay(QDate date, MachineType type)
// For a machine match, find at least one enabled Session. // For a machine match, find at least one enabled Session.
for (int i = 0; i < day->size(); ++i) { for (int i = 0; i < day->size(); ++i) {
Session * sess = (*day)[i]; Session * sess = (*day)[i];
if (((type == MT_UNKNOWN) || (sess->machine()->type() == type)) && sess->enabled()) if (((type == MT_UNKNOWN) || (sess->machine()->type() == type)) && sess->enabled()) {
day->OpenSummary();
return day; return day;
}
} }
// No enabled Sessions were found. // No enabled Sessions were found.
return nullptr; return nullptr;
} }
Day *Profile::FindGoodDay(QDate date, MachineType type)
{
Day *day = FindDay(date, type);
if (!day)
return nullptr;
// For a machine match, find at least one enabled Session.
for (int i = 0; i < day->size(); ++i) {
Session * sess = (*day)[i];
if (((type == MT_UNKNOWN) || (sess->machine()->type() == type)) && sess->enabled()) {
return day;
}
}
// No enabled Sessions were found.
return nullptr;
}
Day *Profile::GetDay(QDate date, MachineType type) Day *Profile::GetDay(QDate date, MachineType type)
{ {
QMap<QDate, Day *>::iterator di = daylist.find(date); QMap<QDate, Day *>::iterator di = daylist.find(date);
@ -629,9 +651,33 @@ Day *Profile::GetDay(QDate date, MachineType type)
Day * day = di.value(); Day * day = di.value();
if (type == MT_UNKNOWN) return day; // just want the day record if (type == MT_UNKNOWN) {
day->OpenSummary();
return day; // just want the day record
}
if (day->machines.contains(type)) return day; if (day->machines.contains(type)) {
day->OpenSummary();
return day;
}
return nullptr;
}
Day *Profile::FindDay(QDate date, MachineType type)
{
QMap<QDate, Day *>::iterator di = daylist.find(date);
if (di == daylist.end()) return nullptr;
Day * day = di.value();
if (type == MT_UNKNOWN) {
return day; // just want the day record
}
if (day->machines.contains(type)) {
return day;
}
return nullptr; return nullptr;
} }
@ -923,7 +969,7 @@ int Profile::countDays(MachineType mt, QDate start, QDate end)
int days = 0; int days = 0;
do { do {
Day *day = GetGoodDay(date, mt); Day *day = FindGoodDay(date, mt);
if (day) { if (day) {
days++; days++;
@ -1552,7 +1598,7 @@ QDate Profile::FirstDay(MachineType mt)
QDate d = m_first; QDate d = m_first;
do { do {
if (GetDay(d, mt) != nullptr) { if (FindDay(d, mt) != nullptr) {
return d; return d;
} }
@ -1572,7 +1618,7 @@ QDate Profile::LastDay(MachineType mt)
QDate d = m_last; QDate d = m_last;
do { do {
if (GetDay(d, mt) != nullptr) { if (FindDay(d, mt) != nullptr) {
return d; return d;
} }
@ -1597,7 +1643,7 @@ QDate Profile::FirstGoodDay(MachineType mt)
} }
do { do {
if (GetGoodDay(d, mt) != nullptr) { if (FindGoodDay(d, mt) != nullptr) {
return d; return d;
} }
@ -1620,7 +1666,7 @@ QDate Profile::LastGoodDay(MachineType mt)
} }
do { do {
if (GetGoodDay(d, mt) != nullptr) { if (FindGoodDay(d, mt) != nullptr) {
return d; return d;
} }
@ -1629,6 +1675,20 @@ QDate Profile::LastGoodDay(MachineType mt)
return f; return f;
} }
bool Profile::channelAvailable(ChannelID code)
{
QHash<MachineID, Machine *>::iterator it;
QHash<MachineID, Machine *>::iterator machlist_end=machlist.end();
for (it = machlist.begin(); it != machlist_end; it++) {
Machine * mach = it.value();
if (mach->hasChannel(code))
return true;
}
return false;
}
bool Profile::hasChannel(ChannelID code) bool Profile::hasChannel(ChannelID code)
{ {
QDate d = LastDay(); QDate d = LastDay();
@ -1648,6 +1708,7 @@ bool Profile::hasChannel(ChannelID code)
if (dit != daylist.end()) { if (dit != daylist.end()) {
Day *day = dit.value(); Day *day = dit.value();
if (day->channelHasData(code)) { if (day->channelHasData(code)) {
found = true; found = true;
break; break;

View File

@ -87,10 +87,16 @@ class Profile : public Preferences
//! \brief Get Day record if data available for date and machine type, else return nullptr //! \brief Get Day record if data available for date and machine type, else return nullptr
Day *GetDay(QDate date, MachineType type = MT_UNKNOWN); Day *GetDay(QDate date, MachineType type = MT_UNKNOWN);
//! \brief Same as GetDay but does not open the summaries
Day *FindDay(QDate date, MachineType type = MT_UNKNOWN);
//! \brief Get Day record if data available for date and machine type, //! \brief Get Day record if data available for date and machine type,
// and has enabled session data, else return nullptr // and has enabled session data, else return nullptr
Day *GetGoodDay(QDate date, MachineType type); Day *GetGoodDay(QDate date, MachineType type);
//! \breif Same as GetGoodDay but does not open the summaries
Day *FindGoodDay(QDate date, MachineType type);
//! \brief Returns a list of all machines of type t //! \brief Returns a list of all machines of type t
QList<Machine *> GetMachines(MachineType t = MT_UNKNOWN); QList<Machine *> GetMachines(MachineType t = MT_UNKNOWN);
@ -147,6 +153,11 @@ class Profile : public Preferences
//! \brief Tests if Channel code is available in all day sets //! \brief Tests if Channel code is available in all day sets
bool hasChannel(ChannelID code); bool hasChannel(ChannelID code);
//! \brief Looks up if any machines report channel is available
bool channelAvailable(ChannelID code);
//! \brief Calculates the minimum session settings value for channel code, between start and end dates //! \brief Calculates the minimum session settings value for channel code, between start and end dates
EventDataType calcSettingsMin(ChannelID code, MachineType mt = MT_CPAP, QDate start = QDate(), EventDataType calcSettingsMin(ChannelID code, MachineType mt = MT_CPAP, QDate start = QDate(),
QDate end = QDate()); QDate end = QDate());

View File

@ -142,11 +142,11 @@ void init()
schema::channel.add(GRP_CPAP, new Channel(CPAP_PressureMin = 0x1020, SETTING, MT_CPAP, SESSION, schema::channel.add(GRP_CPAP, new Channel(CPAP_PressureMin = 0x1020, SETTING, MT_CPAP, SESSION,
"PressureMin", QObject::tr("Min Pressure") , QObject::tr("Minimum Therapy Pressure"), "PressureMin", QObject::tr("Min Pressure") , QObject::tr("Minimum Therapy Pressure"),
QObject::tr("Pressure Min"), STR_UNIT_CMH2O, DEFAULT, QColor("black"))); QObject::tr("Pressure Min"), STR_UNIT_CMH2O, DEFAULT, QColor("orange")));
schema::channel.add(GRP_CPAP, new Channel(CPAP_PressureMax = 0x1021, SETTING, MT_CPAP, SESSION, schema::channel.add(GRP_CPAP, new Channel(CPAP_PressureMax = 0x1021, SETTING, MT_CPAP, SESSION,
"PressureMax", QObject::tr("Max Pressure"), QObject::tr("Maximum Therapy Pressure"), "PressureMax", QObject::tr("Max Pressure"), QObject::tr("Maximum Therapy Pressure"),
QObject::tr("Pressure Max"), STR_UNIT_CMH2O, DEFAULT, QColor("black"))); QObject::tr("Pressure Max"), STR_UNIT_CMH2O, DEFAULT, QColor("light blue")));
schema::channel.add(GRP_CPAP, new Channel(CPAP_RampTime = 0x1022, SETTING, MT_CPAP, SESSION, schema::channel.add(GRP_CPAP, new Channel(CPAP_RampTime = 0x1022, SETTING, MT_CPAP, SESSION,
"RampTime", QObject::tr("Ramp Time") , QObject::tr("Ramp Delay Period"), "RampTime", QObject::tr("Ramp Time") , QObject::tr("Ramp Delay Period"),
@ -170,13 +170,13 @@ void init()
schema::channel.add(GRP_CPAP, new Channel(CPAP_ClearAirway = 0x1001, FLAG, MT_CPAP, SESSION, schema::channel.add(GRP_CPAP, new Channel(CPAP_ClearAirway = 0x1001, FLAG, MT_CPAP, SESSION,
"ClearAirway", QObject::tr("Clear Airway Apnea"), "ClearAirway", QObject::tr("Clear Airway"),
QObject::tr("An apnea where the airway is open"), QObject::tr("An apnea where the airway is open"),
QObject::tr("CA"), STR_UNIT_EventsPerHour, DEFAULT, QColor("purple"))); QObject::tr("CA"), STR_UNIT_EventsPerHour, DEFAULT, QColor("purple")));
schema::channel.add(GRP_CPAP, new Channel(CPAP_Obstructive = 0x1002, FLAG, MT_CPAP, SESSION, schema::channel.add(GRP_CPAP, new Channel(CPAP_Obstructive = 0x1002, FLAG, MT_CPAP, SESSION,
"Obstructive", QObject::tr("Obstructive Apnea"), "Obstructive", QObject::tr("Obstructive"),
QObject::tr("An apnea caused by airway obstruction"), QObject::tr("An apnea caused by airway obstruction"),
QObject::tr("OA"), STR_UNIT_EventsPerHour, DEFAULT, QColor("#40c0ff"))); QObject::tr("OA"), STR_UNIT_EventsPerHour, DEFAULT, QColor("#40c0ff")));
@ -197,9 +197,9 @@ void init()
QObject::tr("FL"), STR_UNIT_EventsPerHour, DEFAULT, QColor("#404040"))); QObject::tr("FL"), STR_UNIT_EventsPerHour, DEFAULT, QColor("#404040")));
schema::channel.add(GRP_CPAP, new Channel(CPAP_RERA = 0x1006, FLAG, MT_CPAP, SESSION, "RERA", schema::channel.add(GRP_CPAP, new Channel(CPAP_RERA = 0x1006, FLAG, MT_CPAP, SESSION, "RERA",
QObject::tr("Respiratory Effort Related Arousal"), QObject::tr("RERA"),
QObject::tr("An restriction in breathing that causes an either an awakening or sleep disturbance."), QObject::tr("Respiratory Effort Related Arousal: An restriction in breathing that causes an either an awakening or sleep disturbance."),
QObject::tr("RE"), STR_UNIT_EventsPerHour, DEFAULT, QColor("gold"))); QObject::tr("RE"), STR_UNIT_EventsPerHour, DEFAULT, COLOR_Gold));
schema::channel.add(GRP_CPAP, new Channel(CPAP_VSnore = 0x1007, FLAG, MT_CPAP, SESSION, "VSnore", schema::channel.add(GRP_CPAP, new Channel(CPAP_VSnore = 0x1007, FLAG, MT_CPAP, SESSION, "VSnore",
QObject::tr("Vibratory Snore"), QObject::tr("A vibratory snore"), QObject::tr("Vibratory Snore"), QObject::tr("A vibratory snore"),
@ -235,7 +235,7 @@ void init()
schema::channel.add(GRP_CPAP, new Channel(CPAP_SensAwake = 0x100d, FLAG, MT_CPAP, SESSION, schema::channel.add(GRP_CPAP, new Channel(CPAP_SensAwake = 0x100d, FLAG, MT_CPAP, SESSION,
"SensAwake", QObject::tr("SensAwake"), "SensAwake", QObject::tr("SensAwake"),
QObject::tr("SensAwake feature will reduce pressure when waking is detected."), QObject::tr("SensAwake feature will reduce pressure when waking is detected."),
QObject::tr("SA"), STR_UNIT_EventsPerHour, DEFAULT, QColor("gold"))); QObject::tr("SA"), STR_UNIT_EventsPerHour, DEFAULT, COLOR_Gold));
schema::channel.add(GRP_CPAP, new Channel(CPAP_UserFlag1 = 0x101e, FLAG, MT_CPAP, SESSION, schema::channel.add(GRP_CPAP, new Channel(CPAP_UserFlag1 = 0x101e, FLAG, MT_CPAP, SESSION,
"UserFlag1", QObject::tr("User Flag #1"), "UserFlag1", QObject::tr("User Flag #1"),
@ -592,6 +592,7 @@ Channel::Channel(ChannelID id, ChanType type, MachineType machtype, ScopeType sc
calc[Calc_LowerThresh] = ChannelCalc(id, Calc_LowerThresh, Qt::blue, false); calc[Calc_LowerThresh] = ChannelCalc(id, Calc_LowerThresh, Qt::blue, false);
calc[Calc_UpperThresh] = ChannelCalc(id, Calc_UpperThresh, Qt::red, false); calc[Calc_UpperThresh] = ChannelCalc(id, Calc_UpperThresh, Qt::red, false);
} }
m_showInOverview = false;
} }
bool Channel::isNull() bool Channel::isNull()
{ {
@ -871,6 +872,7 @@ bool ChannelList::Save(QString filename)
cn.setAttribute("order", chan->order()); cn.setAttribute("order", chan->order());
cn.setAttribute("type", chan->type()); cn.setAttribute("type", chan->type());
cn.setAttribute("datatype", chan->datatype()); cn.setAttribute("datatype", chan->datatype());
cn.setAttribute("overview", chan->showInOverview());
QHash<int, QString>::iterator op; QHash<int, QString>::iterator op;
for (op = chan->m_options.begin(); op!=chan->m_options.end(); ++op) { for (op = chan->m_options.begin(); op!=chan->m_options.end(); ++op) {
QDomElement c2 = doc.createElement("option"); QDomElement c2 = doc.createElement("option");

View File

@ -87,7 +87,7 @@ extern Channel EmptyChannel;
class Channel class Channel
{ {
public: public:
Channel() { m_id = 0; m_upperThreshold = 0; m_lowerThreshold = 0; m_enabled = true; m_order = 255; m_machtype = MT_UNKNOWN; } Channel() { m_id = 0; m_upperThreshold = 0; m_lowerThreshold = 0; m_enabled = true; m_order = 255; m_machtype = MT_UNKNOWN; m_showInOverview = false; }
Channel(ChannelID id, ChanType type, MachineType machtype, ScopeType scope, QString code, QString fullname, Channel(ChannelID id, ChanType type, MachineType machtype, ScopeType scope, QString code, QString fullname,
QString description, QString label, QString unit, DataType datatype = DEFAULT, QColor = Qt::black, QString description, QString label, QString unit, DataType datatype = DEFAULT, QColor = Qt::black,
int link = 0); int link = 0);
@ -105,6 +105,8 @@ class Channel
const QString &units() { return m_unit; } const QString &units() { return m_unit; }
inline short order() const { return m_order; } inline short order() const { return m_order; }
bool showInOverview() { return m_showInOverview; }
inline EventDataType upperThreshold() const { return m_upperThreshold; } inline EventDataType upperThreshold() const { return m_upperThreshold; }
inline EventDataType lowerThreshold() const { return m_lowerThreshold; } inline EventDataType lowerThreshold() const { return m_lowerThreshold; }
inline QColor upperThresholdColor() const { return m_upperThresholdColor; } inline QColor upperThresholdColor() const { return m_upperThresholdColor; }
@ -124,6 +126,8 @@ class Channel
void setLowerThresholdColor(QColor color) { m_lowerThresholdColor = color; } void setLowerThresholdColor(QColor color) { m_lowerThresholdColor = color; }
void setOrder(short order) { m_order = order; } void setOrder(short order) { m_order = order; }
void setShowInOverview(bool b) { m_showInOverview = b; }
QString option(int i) { QString option(int i) {
if (m_options.contains(i)) { if (m_options.contains(i)) {
return m_options[i]; return m_options[i];
@ -170,6 +174,8 @@ class Channel
bool m_enabled; bool m_enabled;
short m_order; short m_order;
bool m_showInOverview;
}; };
/*! \class ChannelList /*! \class ChannelList

View File

@ -38,6 +38,7 @@ Session::Session(Machine *m, SessionID session)
s_session = session; s_session = session;
s_changed = false; s_changed = false;
s_events_loaded = false; s_events_loaded = false;
s_summary_loaded = false;
_first_session = true; _first_session = true;
s_enabled = -1; s_enabled = -1;
@ -110,7 +111,7 @@ bool Session::OpenEvents()
// qWarning() << "Error Loading Events" << filename; // qWarning() << "Error Loading Events" << filename;
return false; return false;
} }
qDebug() << "opening" << filename; qDebug() << "Loading" << s_machine->loaderName() << "Events" << filename;
return s_events_loaded = true; return s_events_loaded = true;
} }
@ -263,7 +264,15 @@ bool Session::StoreSummary()
QString filename = s_machine->getSummariesPath() + QString().sprintf("%08lx.000", s_session); QString filename = s_machine->getSummariesPath() + QString().sprintf("%08lx.000", s_session);
QFile file(filename); QFile file(filename);
file.open(QIODevice::WriteOnly); if (!file.open(QIODevice::WriteOnly)) {
QDir dir;
dir.mkpath(s_machine->getSummariesPath());
if (!file.open(QIODevice::WriteOnly)) {
qDebug() << "Summary open for writing failed";
return false;
}
}
QDataStream out(&file); QDataStream out(&file);
out.setVersion(QDataStream::Qt_4_6); out.setVersion(QDataStream::Qt_4_6);
@ -319,9 +328,13 @@ bool Session::StoreSummary()
} }
bool Session::LoadSummary(QString filename) bool Session::LoadSummary()
{ {
s_changed = true; static int sumcnt = 0;
if (s_summary_loaded) return true;
QString filename = s_machine->getSummariesPath() + QString().sprintf("%08lx.000", s_session);
if (filename.isEmpty()) { if (filename.isEmpty()) {
qDebug() << "Empty summary filename"; qDebug() << "Empty summary filename";
@ -335,6 +348,9 @@ bool Session::LoadSummary(QString filename)
return false; return false;
} }
qDebug() << "Loading" << s_machine->loaderName() << "Summary" << filename << sumcnt++;
QDataStream in(&file); QDataStream in(&file);
in.setVersion(QDataStream::Qt_4_6); in.setVersion(QDataStream::Qt_4_6);
in.setByteOrder(QDataStream::LittleEndian); in.setByteOrder(QDataStream::LittleEndian);
@ -564,9 +580,10 @@ bool Session::LoadSummary(QString filename)
} else { } else {
// summary only upgrades go here. // summary only upgrades go here.
} }
SetChanged(true); StoreSummary();
} }
s_summary_loaded = true;
return true; return true;
} }
@ -1083,6 +1100,8 @@ void Session::UpdateSummaries()
} }
} }
timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline()); timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline());
s_machine->updateChannels(this);
} }
EventDataType Session::SearchValue(ChannelID code, qint64 time, bool square) EventDataType Session::SearchValue(ChannelID code, qint64 time, bool square)

View File

@ -53,6 +53,8 @@ public:
*/ */
class Session class Session
{ {
friend class Day;
friend class Machine;
public: public:
/*! \fn Session(Machine *,SessionID); /*! \fn Session(Machine *,SessionID);
\brief Create a session object belonging to Machine, with supplied SessionID \brief Create a session object belonging to Machine, with supplied SessionID
@ -83,7 +85,7 @@ class Session
void LoadSummaryData(QDataStream & in); void LoadSummaryData(QDataStream & in);
//! \brief Loads the Sessions Summary Indexes from filename, from SleepLibs custom data format. //! \brief Loads the Sessions Summary Indexes from filename, from SleepLibs custom data format.
bool LoadSummary(QString filename); bool LoadSummary();
//! \brief Loads the Sessions EventLists from filename, from SleepLibs custom data format. //! \brief Loads the Sessions EventLists from filename, from SleepLibs custom data format.
bool LoadEvents(QString filename); bool LoadEvents(QString filename);
@ -389,6 +391,7 @@ protected:
bool _first_session; bool _first_session;
bool s_summaryOnly; bool s_summaryOnly;
bool s_summary_loaded;
bool s_events_loaded; bool s_events_loaded;
bool s_enabled; bool s_enabled;

View File

@ -693,11 +693,11 @@ void Daily::UpdateCalendarDay(QDate date)
nodata.setForeground(QBrush(COLOR_Black, Qt::SolidPattern)); nodata.setForeground(QBrush(COLOR_Black, Qt::SolidPattern));
nodata.setFontWeight(QFont::Normal); nodata.setFontWeight(QFont::Normal);
bool hascpap=p_profile->GetDay(date,MT_CPAP)!=nullptr; bool hascpap=p_profile->FindDay(date,MT_CPAP)!=nullptr;
bool hasoxi=p_profile->GetDay(date,MT_OXIMETER)!=nullptr; bool hasoxi=p_profile->FindDay(date,MT_OXIMETER)!=nullptr;
bool hasjournal=p_profile->GetDay(date,MT_JOURNAL)!=nullptr; bool hasjournal=p_profile->FindDay(date,MT_JOURNAL)!=nullptr;
bool hasstage=p_profile->GetDay(date,MT_SLEEPSTAGE)!=nullptr; bool hasstage=p_profile->FindDay(date,MT_SLEEPSTAGE)!=nullptr;
bool haspos=p_profile->GetDay(date,MT_POSITION)!=nullptr; bool haspos=p_profile->FindDay(date,MT_POSITION)!=nullptr;
if (hascpap) { if (hascpap) {
if (hasoxi) { if (hasoxi) {
ui->calendar->setDateTextFormat(date,oxicpap); ui->calendar->setDateTextFormat(date,oxicpap);
@ -1334,17 +1334,10 @@ void Daily::Load(QDate date)
Day * d = di.value(); Day * d = di.value();
if (d->eventsLoaded()) { if (d->eventsLoaded()) {
if (d->useCounter() == 0) { if (d->useCounter() == 0) {
for (QList<Session *>::iterator s=d->begin();s!=d->end();++s) { d->CloseEvents();
(*s)->TrashEvents();
}
} }
} }
} }
// if (lastcpapday->useCounter() == 0) {
// for (QList<Session *>::iterator s=lastcpapday->begin();s!=lastcpapday->end();++s) {
// (*s)->TrashEvents();
// }
// }
} }
} }
@ -1625,7 +1618,7 @@ void Daily::Load(QDate date)
html+="</body></html>"; html+="</body></html>";
QColor cols[]={ QColor cols[]={
QColor("gold"), COLOR_Gold,
QColor("light blue"), QColor("light blue"),
}; };
const int maxcolors=sizeof(cols)/sizeof(QColor); const int maxcolors=sizeof(cols)/sizeof(QColor);
@ -2262,7 +2255,8 @@ void Daily::on_ZombieMeter_valueChanged(int action)
journal->settings[Journal_ZombieMeter]=ui->ZombieMeter->value(); journal->settings[Journal_ZombieMeter]=ui->ZombieMeter->value();
journal->SetChanged(true); journal->SetChanged(true);
if (mainwin->getOverview()) mainwin->getOverview()->ResetGraph("Zombie"); // shouldn't be needed anymore with new overview model..
//if (mainwin->getOverview()) mainwin->getOverview()->ResetGraph("Zombie");
} }
void Daily::on_bookmarkTable_itemChanged(QTableWidgetItem *item) void Daily::on_bookmarkTable_itemChanged(QTableWidgetItem *item)
@ -2363,7 +2357,9 @@ void Daily::on_ouncesSpinBox_editingFinished()
} }
} }
journal->SetChanged(true); journal->SetChanged(true);
if (mainwin->getOverview()) mainwin->getOverview()->ResetGraph(STR_GRAPH_Weight);
// shouldn't be needed anymore with new overview model
//if (mainwin->getOverview()) mainwin->getOverview()->ResetGraph(STR_GRAPH_Weight);
} }
QString Daily::GetDetailsText() QString Daily::GetDetailsText()

View File

@ -336,7 +336,7 @@ void MainWindow::on_changeWarningMessage()
} }
quint16 chandata_version = 0; quint16 chandata_version = 1;
void saveChannels() void saveChannels()
{ {
@ -370,6 +370,7 @@ void saveChannels()
out << chan->lowerThresholdColor(); out << chan->lowerThresholdColor();
out << chan->upperThreshold(); out << chan->upperThreshold();
out << chan->upperThresholdColor(); out << chan->upperThresholdColor();
out << chan->showInOverview();
} }
f.close(); f.close();
@ -420,6 +421,7 @@ void loadChannels()
QString fullname; QString fullname;
QString label; QString label;
QString description; QString description;
bool showOverview = false;
for (int i=0; i < size; i++) { for (int i=0; i < size; i++) {
in >> code; in >> code;
@ -438,6 +440,10 @@ void loadChannels()
in >> lowerThresholdColor; in >> lowerThresholdColor;
in >> upperThreshold; in >> upperThreshold;
in >> upperThresholdColor; in >> upperThresholdColor;
if (version >= 1) {
in >> showOverview;
}
if (chan->isNull()) { if (chan->isNull()) {
qDebug() << "loadChannels has no idea about channel" << name; qDebug() << "loadChannels has no idea about channel" << name;
if (in.atEnd()) return; if (in.atEnd()) return;
@ -452,6 +458,8 @@ void loadChannels()
chan->setLowerThresholdColor(lowerThresholdColor); chan->setLowerThresholdColor(lowerThresholdColor);
chan->setUpperThreshold(upperThreshold); chan->setUpperThreshold(upperThreshold);
chan->setUpperThresholdColor(upperThresholdColor); chan->setUpperThresholdColor(upperThresholdColor);
chan->setShowInOverview(showOverview);
if (in.atEnd()) return; if (in.atEnd()) return;
} }
@ -1470,8 +1478,8 @@ void MainWindow::on_action_Preferences_triggered()
} }
if (overview) { if (overview) {
overview->ReloadGraphs(); overview->RebuildGraphs(true);
overview->RedrawGraphs(); //overview->RedrawGraphs();
} }
} }

View File

@ -114,23 +114,19 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
ui->dateLayout->addWidget(dateLabel,1); ui->dateLayout->addWidget(dateLabel,1);
// uc = new SummaryChart(STR_UNIT_Hours, GT_BAR);
// uc->addSlice(NoChannel, COLOR_Green, ST_HOURS);
// UC->AddLayer(uc);
/* return;
// TODO: Automate graph creation process // TODO: Automate graph creation process
ChannelID ahicode = p_profile->general->calculateRDI() ? CPAP_RDI : CPAP_AHI;
if (ahicode == CPAP_RDI) {
AHI = createGraph(STR_GRAPH_AHI, STR_TR_RDI, tr("Respiratory\nDisturbance\nIndex"));
} else {
AHI = createGraph(STR_GRAPH_AHI, STR_TR_AHI, tr("Apnea\nHypopnea\nIndex"));
}
STG = createGraph("New Session", tr("Session Times2"), tr("Session Times"), YT_Time);
stg = new gSessionTimesChart("STG", MT_CPAP);
STG->AddLayer(stg);
UC = createGraph(STR_GRAPH_Usage, tr("Usage"), tr("Usage\n(hours)"));
FL = createGraph(schema::channel[CPAP_FlowLimit].code(), schema::channel[CPAP_FlowLimit].label(), STR_TR_FlowLimit);
float percentile = p_profile->general->prefCalcPercentile() / 100.0; float percentile = p_profile->general->prefCalcPercentile() / 100.0;
int mididx = p_profile->general->prefCalcMiddle(); int mididx = p_profile->general->prefCalcMiddle();
@ -144,32 +140,13 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
const EventDataType maxperc = 0.995F; const EventDataType maxperc = 0.995F;
US = createGraph(STR_GRAPH_SessionTimes, tr("Session Times"), tr("Session Times\n(hours)"), YT_Time); US = createGraph(STR_GRAPH_SessionTimes, tr("Session Times"), tr("Session Times\n(hours)"), YT_Time);
PR = createGraph("Pressure", STR_TR_Pressure, STR_TR_Pressure + "\n(" + STR_UNIT_CMH2O + ")");
SET = createGraph("Settings", STR_TR_Settings, STR_TR_Settings); SET = createGraph("Settings", STR_TR_Settings, STR_TR_Settings);
LK = createGraph("Leaks", STR_TR_Leaks, STR_TR_UnintentionalLeaks + "\n(" + STR_UNIT_LPM + ")");
TOTLK = createGraph("TotalLeaks", STR_TR_TotalLeaks, STR_TR_TotalLeaks + "\n(" + STR_UNIT_LPM + ")");
NPB = createGraph("TimeInPB", tr("% in %1").arg(schema::channel[CPAP_CSR].label()), tr("%1\n(% of night)").arg(schema::channel[CPAP_LargeLeak].description()));
NLL = createGraph("TimeInLL", tr("% in %1").arg(schema::channel[CPAP_LargeLeak].label()), tr("Large Leaks\n(% of night)"));
if (ahicode == CPAP_RDI) {
AHIHR = createGraph(STR_GRAPH_PeakAHI, tr("Peak RDI"), tr("Peak RDI\nShows RDI Clusters\n(RDI/hr)"));
} else {
AHIHR = createGraph(STR_GRAPH_PeakAHI, tr("Peak AHI"), tr("Peak AHI\nShows AHI Clusters\n(AHI/hr)"));
}
RR = createGraph(schema::channel[CPAP_RespRate].code(), schema::channel[CPAP_RespRate].label(), schema::channel[CPAP_RespRate].fullname()+"\n"+schema::channel[CPAP_RespRate].units());
TV = createGraph(schema::channel[CPAP_TidalVolume].code(),schema::channel[CPAP_TidalVolume].label(), tr("Tidal\nVolume\n(ml)"));
MV = createGraph(schema::channel[CPAP_MinuteVent].code(), schema::channel[CPAP_MinuteVent].label(), tr("Minute\nVentilation\n(L/min)"));
TGMV = createGraph(schema::channel[CPAP_TgMV].code(), schema::channel[CPAP_TgMV].label(), tr("Target\nVentilation\n(L/min)")); TGMV = createGraph(schema::channel[CPAP_TgMV].code(), schema::channel[CPAP_TgMV].label(), tr("Target\nVentilation\n(L/min)"));
PTB = createGraph(schema::channel[CPAP_PTB].code(), schema::channel[CPAP_PTB].label(), tr("Patient\nTriggered\nBreaths\n(%)")); PTB = createGraph(schema::channel[CPAP_PTB].code(), schema::channel[CPAP_PTB].label(), tr("Patient\nTriggered\nBreaths\n(%)"));
SES = createGraph(STR_GRAPH_Sessions, STR_TR_Sessions, STR_TR_Sessions + tr("\n(count)")); SES = createGraph(STR_GRAPH_Sessions, STR_TR_Sessions, STR_TR_Sessions + tr("\n(count)"));
PULSE = createGraph(schema::channel[OXI_Pulse].code(), schema::channel[OXI_Pulse].label(), STR_TR_PulseRate + "\n(" + STR_UNIT_BPM + ")");
SPO2 = createGraph(schema::channel[OXI_SPO2].code(), schema::channel[OXI_SPO2].label(), tr("Oxygen Saturation\n(%)"));
SA = createGraph(schema::channel[CPAP_SensAwake].code(), schema::channel[CPAP_SensAwake].label(), tr("SensAwake\n(count)"));
WEIGHT = createGraph(STR_GRAPH_Weight, STR_TR_Weight, STR_TR_Weight, YT_Weight);
BMI = createGraph(STR_GRAPH_BMI, STR_TR_BMI, tr("Body\nMass\nIndex"));
ZOMBIE = createGraph(STR_GRAPH_Zombie, STR_TR_Zombie, tr("How you felt\n(0-10)"));
ahihr = new SummaryChart(STR_UNIT_EventsPerHour, GT_POINTS); ahihr = new SummaryChart(STR_UNIT_EventsPerHour, GT_POINTS);
ahihr->addSlice(ahicode, COLOR_Blue, ST_MAX); ahihr->addSlice(ahicode, COLOR_Blue, ST_MAX);
@ -205,9 +182,6 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
spo2->addSlice(OXI_SPO2, COLOR_Blue, ST_MIN); spo2->addSlice(OXI_SPO2, COLOR_Blue, ST_MIN);
SPO2->AddLayer(spo2); SPO2->AddLayer(spo2);
uc = new SummaryChart(STR_UNIT_Hours, GT_BAR);
uc->addSlice(NoChannel, COLOR_Green, ST_HOURS);
UC->AddLayer(uc);
fl = new SummaryChart(STR_TR_FL, GT_POINTS); fl = new SummaryChart(STR_TR_FL, GT_POINTS);
fl->addSlice(CPAP_FlowLimit, COLOR_Brown, ST_CPH); fl->addSlice(CPAP_FlowLimit, COLOR_Brown, ST_CPH);
@ -296,13 +270,6 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
// Added in summarychart.. Slightly annoying.. // Added in summarychart.. Slightly annoying..
PR->AddLayer(pr); PR->AddLayer(pr);
lk = new SummaryChart(STR_TR_Leaks, GT_POINTS);
lk->addSlice(CPAP_Leak, COLOR_LightBlue, ST_mid, 0.5);
lk->addSlice(CPAP_Leak, COLOR_DarkGray, ST_PERC, percentile);
//lk->addSlice(CPAP_Leak,COLOR_DarkBlue,ST_WAVG);
lk->addSlice(CPAP_Leak, COLOR_Gray, ST_max, maxperc);
//lk->addSlice(CPAP_Leak,COLOR_DarkYellow);
LK->AddLayer(lk);
totlk = new SummaryChart(STR_TR_TotalLeaks, GT_POINTS); totlk = new SummaryChart(STR_TR_TotalLeaks, GT_POINTS);
totlk->addSlice(CPAP_LeakTotal, COLOR_LightBlue, ST_mid, 0.5); totlk->addSlice(CPAP_LeakTotal, COLOR_LightBlue, ST_mid, 0.5);
@ -312,21 +279,25 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
//tot->addSlice(CPAP_Leak, COLOR_DarkYellow); //tot->addSlice(CPAP_Leak, COLOR_DarkYellow);
TOTLK->AddLayer(totlk); TOTLK->AddLayer(totlk);
NPB->AddLayer(npb = new SummaryChart(tr("% PB"), GT_POINTS));
npb->addSlice(CPAP_CSR, schema::channel[CPAP_CSR].defaultColor(), ST_SPH);
NLL->AddLayer(nll = new SummaryChart(tr("% %1").arg(schema::channel[CPAP_LargeLeak].fullname()), GT_POINTS)); NLL->AddLayer(nll = new SummaryChart(tr("% %1").arg(schema::channel[CPAP_LargeLeak].fullname()), GT_POINTS));
nll->addSlice(CPAP_LargeLeak, schema::channel[CPAP_LargeLeak].defaultColor(), ST_SPH); nll->addSlice(CPAP_LargeLeak, schema::channel[CPAP_LargeLeak].defaultColor(), ST_SPH);
// <--- The code to the previous marker is crap // <--- The code to the previous marker is crap
AHI->setPinned(false); AHI->setPinned(false);
ui->rangeCombo->setCurrentIndex(p_profile->general->lastOverviewRange());
icon_on = new QIcon(":/icons/session-on.png");
icon_off = new QIcon(":/icons/session-off.png");
SES->setRecMinY(1); SES->setRecMinY(1);
SET->setRecMinY(0); SET->setRecMinY(0);
//SET->setRecMaxY(5); //SET->setRecMaxY(5);
*/
RebuildGraphs(false);
ui->rangeCombo->setCurrentIndex(p_profile->general->lastOverviewRange());
icon_on = new QIcon(":/icons/session-on.png");
icon_off = new QIcon(":/icons/session-off.png");
GraphView->resetLayout(); GraphView->resetLayout();
GraphView->LoadSettings("Overview"); //no trans GraphView->LoadSettings("Overview"); //no trans
@ -340,8 +311,121 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
Overview::~Overview() Overview::~Overview()
{ {
delete ui; delete ui;
delete icon_on; // delete icon_on;
delete icon_off; // delete icon_off;
}
void Overview::RebuildGraphs(bool reset)
{
qint64 minx, maxx;
if (reset) {
GraphView->GetXBounds(minx, maxx);
}
GraphView->trashGraphs(true);
ChannelID ahicode = p_profile->general->calculateRDI() ? CPAP_RDI : CPAP_AHI;
if (ahicode == CPAP_RDI) {
AHI = createGraph("AHIBreakdown", STR_TR_RDI, tr("Respiratory\nDisturbance\nIndex"));
} else {
AHI = createGraph("AHIBreakdown", STR_TR_AHI, tr("Apnea\nHypopnea\nIndex"));
}
ahi = new gAHIChart();
AHI->AddLayer(ahi);
UC = createGraph(STR_GRAPH_Usage, tr("Usage"), tr("Usage\n(hours)"));
UC->AddLayer(uc = new gUsageChart());
STG = createGraph("New Session", tr("Session Times"), tr("Session Times"), YT_Time);
stg = new gSessionTimesChart();
STG->AddLayer(stg);
PR = createGraph("Pressure Settings", STR_TR_Pressure, STR_TR_Pressure + "\n(" + STR_UNIT_CMH2O + ")");
pres = new gPressureChart();
PR->AddLayer(pres);
// LK = createGraph("Leaks", STR_TR_Leaks, STR_TR_UnintentionalLeaks + "\n(" + STR_UNIT_LPM + ")");
// LK->AddLayer(new gSummaryChart(CPAP_Leak, MT_CPAP));
// TOTLK = createGraph("TotalLeaks", STR_TR_TotalLeaks, STR_TR_TotalLeaks + "\n(" + STR_UNIT_LPM + ")");
// TOTLK->AddLayer(new gSummaryChart(CPAP_LeakTotal, MT_CPAP));
// NLL = createGraph("TimeInLL", tr("% in %1").arg(schema::channel[CPAP_LargeLeak].label()), tr("Large Leaks\n(% of night)"));
// NLL->AddLayer(nll = new gSummaryChart("TimeInLL", MT_CPAP));
// nll->addCalc(CPAP_LargeLeak, ST_SPH);
// RR = createGraph(schema::channel[CPAP_RespRate].code(), schema::channel[CPAP_RespRate].label(), schema::channel[CPAP_RespRate].fullname()+"\n"+schema::channel[CPAP_RespRate].units());
// RR->AddLayer(new gSummaryChart(CPAP_RespRate, MT_CPAP));
// TV = createGraph(schema::channel[CPAP_TidalVolume].code(),schema::channel[CPAP_TidalVolume].label(), tr("Tidal\nVolume\n(ml)"));
// TV->AddLayer(new gSummaryChart(CPAP_TidalVolume, MT_CPAP));
// MV = createGraph(schema::channel[CPAP_MinuteVent].code(), schema::channel[CPAP_MinuteVent].label(), tr("Minute\nVentilation\n(L/min)"));
// MV->AddLayer(new gSummaryChart(CPAP_MinuteVent, MT_CPAP));
// FL = createGraph(schema::channel[CPAP_FLG].code(), schema::channel[CPAP_FLG].label(), STR_TR_FlowLimit);
// FL->AddLayer(new gSummaryChart(CPAP_FLG, MT_CPAP));
// SN = createGraph(schema::channel[CPAP_Snore].code(), schema::channel[CPAP_Snore].label(), schema::channel[CPAP_Snore].fullname()+"\n"+schema::channel[CPAP_Snore].units());
// SN->AddLayer(new gSummaryChart(CPAP_Snore, MT_CPAP));
QHash<ChannelID, schema::Channel *>::iterator chit;
QHash<ChannelID, schema::Channel *>::iterator chit_end = schema::channel.channels.end();
for (chit = schema::channel.channels.begin(); chit != chit_end; ++chit) {
schema::Channel * chan = chit.value();
if (chan->showInOverview()) {
ChannelID code = chan->id();
QString name = chan->fullname();
if (name.length() > 16) name = chan->label();
gGraph *G = createGraph(chan->code(), name, chan->description());
if ((chan->type() == schema::FLAG) || (chan->type() == schema::MINOR_FLAG)) {
gSummaryChart * sc = new gSummaryChart(chan->code(), MT_CPAP);
sc->addCalc(code, ST_CPH);
G->AddLayer(sc);
} else if (chan->type() == schema::SPAN) {
gSummaryChart * sc = new gSummaryChart(chan->code(), MT_CPAP);
sc->addCalc(code, ST_SPH);
G->AddLayer(sc);
} else if (chan->type() == schema::WAVEFORM) {
G->AddLayer(new gSummaryChart(code, chan->machtype()));
}
}
}
/* PULSE = createGraph(schema::channel[OXI_Pulse].code(), schema::channel[OXI_Pulse].label(), STR_TR_PulseRate + "\n(" + STR_UNIT_BPM + ")");
PULSE->AddLayer(new gSummaryChart(OXI_Pulse, MT_OXIMETER));
SPO2 = createGraph(schema::channel[OXI_SPO2].code(), schema::channel[OXI_SPO2].label(), tr("Oxygen Saturation\n(%)"));
SPO2->AddLayer(new gSummaryChart(OXI_SPO2, MT_OXIMETER));
NPB = createGraph("TimeInPB", tr("% in %1").arg(schema::channel[CPAP_CSR].label()), tr("%1\n(% of night)").arg(schema::channel[CPAP_LargeLeak].description()));
NPB->AddLayer(npb = new gSummaryChart(tr("% PB"), MT_CPAP));
npb->addCalc(CPAP_CSR, ST_SPH);
if (ahicode == CPAP_RDI) {
AHIHR = createGraph(STR_GRAPH_PeakAHI, tr("Peak RDI"), tr("Peak RDI\nShows RDI Clusters\n(RDI/hr)"));
AHIHR->AddLayer(new gSummaryChart(CPAP_RDI, MT_CPAP));
} else {
AHIHR = createGraph(STR_GRAPH_PeakAHI, tr("Peak AHI"), tr("Peak AHI\nShows AHI Clusters\n(AHI/hr)"));
AHIHR->AddLayer(new gSummaryChart(CPAP_AHI, MT_CPAP));
} */
WEIGHT = createGraph(STR_GRAPH_Weight, STR_TR_Weight, STR_TR_Weight, YT_Weight);
BMI = createGraph(STR_GRAPH_BMI, STR_TR_BMI, tr("Body\nMass\nIndex"));
ZOMBIE = createGraph(STR_GRAPH_Zombie, STR_TR_Zombie, tr("How you felt\n(0-10)"));
if (reset) {
// GraphView->setDay(nullptr);
GraphView->resetLayout();
GraphView->setDay(nullptr);
// GraphView->resetLayout();
GraphView->SetXBounds(minx, maxx, 0, false);
GraphView->resetLayout();
updateGraphCombo();
}
} }
void Overview::closeEvent(QCloseEvent *event) void Overview::closeEvent(QCloseEvent *event)
@ -374,10 +458,8 @@ gGraph *Overview::createGraph(QString code, QString name, QString units, YTicker
} }
g->AddLayer(yt, LayerLeft, gYAxis::Margin); g->AddLayer(yt, LayerLeft, gYAxis::Margin);
gXAxis *x = new gXAxis(); gXAxisDay *x = new gXAxisDay();
x->setUtcFix(true); g->AddLayer(x, LayerBottom, 0, gXAxisDay::Margin);
x->setRoundDays(true);
g->AddLayer(x, LayerBottom, 0, gXAxis::Margin);
g->AddLayer(new gXGrid()); g->AddLayer(new gXGrid());
return g; return g;
} }
@ -475,8 +557,8 @@ void Overview::UpdateCalendarDay(QDateEdit *dateedit, QDate date)
cpapcol.setFontWeight(QFont::Bold); cpapcol.setFontWeight(QFont::Bold);
oxiday.setForeground(QBrush(Qt::red, Qt::SolidPattern)); oxiday.setForeground(QBrush(Qt::red, Qt::SolidPattern));
oxiday.setFontWeight(QFont::Bold); oxiday.setFontWeight(QFont::Bold);
bool hascpap = p_profile->GetDay(date, MT_CPAP) != nullptr; bool hascpap = p_profile->FindDay(date, MT_CPAP) != nullptr;
bool hasoxi = p_profile->GetDay(date, MT_OXIMETER) != nullptr; bool hasoxi = p_profile->FindDay(date, MT_OXIMETER) != nullptr;
//bool hasjournal=p_profile->GetDay(date,MT_JOURNAL)!=nullptr; //bool hasjournal=p_profile->GetDay(date,MT_JOURNAL)!=nullptr;
if (hascpap) { if (hascpap) {

View File

@ -62,20 +62,21 @@ class Overview : public QWidget
\param QString units The units of measurements to show in the popup */ \param QString units The units of measurements to show in the popup */
gGraph *createGraph(QString code, QString name, QString units = "", YTickerType yttype = YT_Number); gGraph *createGraph(QString code, QString name, QString units = "", YTickerType yttype = YT_Number);
gGraph *AHI, *AHIHR, *UC, *FL, *SA, *US, *PR, *LK, *NPB, *SET, *SES, *RR, *MV, *TV, *PTB, *PULSE, *SPO2, *NLL, gGraph *AHI, *AHIHR, *UC, *FL, *SA, *US, *PR, *LK, *NPB, *SET, *SES, *RR, *MV, *TV, *PTB, *PULSE, *SPO2, *NLL,
// gGraph *AHI, *AHIHR, *UC, *FL, *US, *PR, *LK, *NPB, *SET, *SES, *RR, *MV, *TV, *PTB, *PULSE, *SPO2, *WEIGHT, *ZOMBIE, *BMI, *TGMV, *TOTLK, *STG, *SN;
*WEIGHT, *ZOMBIE, *BMI, *TGMV, *TOTLK, *STG; SummaryChart *bc, *sa, *us, *pr, *set, *ses, *ptb, *pulse, *spo2,
SummaryChart *bc, *uc, *fl, *sa, *us, *pr, *lk, *npb, *set, *ses, *rr, *mv, *tv, *ptb, *pulse, *spo2, *weight, *zombie, *bmi, *ahihr, *tgmv, *totlk;
// SummaryChart *bc, *uc, *fl, *us, *pr, *lk, *npb, *set, *ses, *rr, *mv, *tv, *ptb, *pulse, *spo2,
*weight, *zombie, *bmi, *ahihr, *tgmv, *totlk, *nll;
gSessionTimesChart * stg; gSummaryChart * stg, *uc, *ahi, * pres, *lk, *npb, *rr, *mv, *tv, *nll, *sn;
//! \breif List of SummaryCharts shown on the overview page //! \breif List of SummaryCharts shown on the overview page
QVector<SummaryChart *> OverviewCharts; QVector<SummaryChart *> OverviewCharts;
void ResetGraph(QString name); void ResetGraph(QString name);
void RebuildGraphs(bool reset = true);
public slots: public slots:
void onRebuildGraphs() { RebuildGraphs(true); }
// ! \brief Print button down the bottom, does the same as File->Print // ! \brief Print button down the bottom, does the same as File->Print
//void on_printButton_clicked(); //void on_printButton_clicked();

View File

@ -37,6 +37,7 @@ OximeterImport::OximeterImport(QWidget *parent) :
oximodule = nullptr; oximodule = nullptr;
liveView = new gGraphView(this); liveView = new gGraphView(this);
liveView->setVisible(false); liveView->setVisible(false);
liveView->setShowAuthorMessage(false);
ui->retryButton->setVisible(false); ui->retryButton->setVisible(false);
ui->stopButton->setVisible(false); ui->stopButton->setVisible(false);
ui->saveButton->setVisible(false); ui->saveButton->setVisible(false);

View File

@ -976,7 +976,7 @@ background: qlineargradient( x1:0 y1:0, x2:1 y2:0, stop:0 white, stop:1 #cccccc)
</font> </font>
</property> </property>
<property name="text"> <property name="text">
<string notr="true">placeholder</string> <string>If you can read this, you likely have your oximeter type set wrong in preferences.</string>
</property> </property>
<property name="alignment"> <property name="alignment">
<set>Qt::AlignCenter</set> <set>Qt::AlignCenter</set>

View File

@ -406,17 +406,19 @@ void PreferencesDialog::InitChanInfo()
headers.append(tr("Name")); headers.append(tr("Name"));
headers.append(tr("Color")); headers.append(tr("Color"));
headers.append(tr("Flag Type")); headers.append(tr("Flag Type"));
headers.append(tr("Overview"));
headers.append(tr("Label")); headers.append(tr("Label"));
headers.append(tr("Details")); headers.append(tr("Details"));
chanModel->setHorizontalHeaderLabels(headers); chanModel->setHorizontalHeaderLabels(headers);
ui->chanView->setColumnWidth(0, 200); ui->chanView->setColumnWidth(0, 200);
ui->chanView->setColumnWidth(1, 50); ui->chanView->setColumnWidth(1, 50);
ui->chanView->setColumnWidth(2, 100); ui->chanView->setColumnWidth(2, 100);
ui->chanView->setColumnWidth(3, 100); ui->chanView->setColumnWidth(3, 60);
ui->chanView->setColumnWidth(4, 100);
ui->chanView->setSelectionMode(QAbstractItemView::SingleSelection); ui->chanView->setSelectionMode(QAbstractItemView::SingleSelection);
ui->chanView->setSelectionBehavior(QAbstractItemView::SelectItems); ui->chanView->setSelectionBehavior(QAbstractItemView::SelectItems);
chanModel->setColumnCount(5); chanModel->setColumnCount(6);
QStandardItem *hdr = nullptr; QStandardItem *hdr = nullptr;
@ -437,7 +439,7 @@ void PreferencesDialog::InitChanInfo()
hdr->setEditable(false); hdr->setEditable(false);
QList<QStandardItem *> list; QList<QStandardItem *> list;
list.append(hdr); list.append(hdr);
for (int i=0; i<4; i++) { for (int i=0; i<5; i++) {
QStandardItem *it = new QStandardItem(); QStandardItem *it = new QStandardItem();
it->setEnabled(false); it->setEnabled(false);
list.append(it); list.append(it);
@ -485,6 +487,13 @@ void PreferencesDialog::InitChanInfo()
it->setEditable(type != schema::UNKNOWN); it->setEditable(type != schema::UNKNOWN);
items.push_back(it); items.push_back(it);
it = new QStandardItem(QString());
it->setToolTip(tr("Whether this flag has a dedicated overview chart."));
it->setCheckable(true);
it->setCheckState(chan->showInOverview() ? Qt::Checked : Qt::Unchecked);
it->setData(chan->id(), Qt::UserRole);
items.push_back(it);
it = new QStandardItem(chan->label()); it = new QStandardItem(chan->label());
it->setToolTip(tr("This is the short-form label to indicate this channel on screen.")); it->setToolTip(tr("This is the short-form label to indicate this channel on screen."));
@ -524,6 +533,7 @@ void PreferencesDialog::InitWaveInfo()
QStringList headers; QStringList headers;
headers.append(tr("Name")); headers.append(tr("Name"));
headers.append(tr("Color")); headers.append(tr("Color"));
headers.append(tr("Overview"));
headers.append(tr("Lower")); headers.append(tr("Lower"));
headers.append(tr("Upper")); headers.append(tr("Upper"));
headers.append(tr("Label")); headers.append(tr("Label"));
@ -531,13 +541,14 @@ void PreferencesDialog::InitWaveInfo()
waveModel->setHorizontalHeaderLabels(headers); waveModel->setHorizontalHeaderLabels(headers);
ui->waveView->setColumnWidth(0, 200); ui->waveView->setColumnWidth(0, 200);
ui->waveView->setColumnWidth(1, 50); ui->waveView->setColumnWidth(1, 50);
ui->waveView->setColumnWidth(2, 50); ui->waveView->setColumnWidth(2, 60);
ui->waveView->setColumnWidth(3, 50); ui->waveView->setColumnWidth(3, 50);
ui->waveView->setColumnWidth(4, 100); ui->waveView->setColumnWidth(4, 50);
ui->waveView->setColumnWidth(5, 100);
ui->waveView->setSelectionMode(QAbstractItemView::SingleSelection); ui->waveView->setSelectionMode(QAbstractItemView::SingleSelection);
ui->waveView->setSelectionBehavior(QAbstractItemView::SelectItems); ui->waveView->setSelectionBehavior(QAbstractItemView::SelectItems);
waveModel->setColumnCount(6); waveModel->setColumnCount(7);
QStandardItem *hdr = nullptr; QStandardItem *hdr = nullptr;
@ -555,7 +566,7 @@ void PreferencesDialog::InitWaveInfo()
hdr->setEditable(false); hdr->setEditable(false);
QList<QStandardItem *> list; QList<QStandardItem *> list;
list.append(hdr); list.append(hdr);
for (int i=0; i<5; i++) { for (int i=0; i<6; i++) {
QStandardItem *it = new QStandardItem(); QStandardItem *it = new QStandardItem();
it->setEnabled(false); it->setEnabled(false);
list.append(it); list.append(it);
@ -594,10 +605,18 @@ void PreferencesDialog::InitWaveInfo()
it->setEditable(false); it->setEditable(false);
it->setData(chan->defaultColor().rgba(), Qt::UserRole); it->setData(chan->defaultColor().rgba(), Qt::UserRole);
it->setToolTip(tr("Double click to change the default color for this channel plot/flag/data.")); it->setToolTip(tr("Double click to change the default color for this channel plot/flag/data."));
it->setSelectable(false); it->setSelectable(false);
items.push_back(it); items.push_back(it);
it = new QStandardItem();
it->setCheckable(true);
it->setCheckState(chan->showInOverview() ? Qt::Checked : Qt::Unchecked);
it->setEditable(true);
it->setData(chan->id(), Qt::UserRole);
it->setToolTip(tr("Whether a breakdown of this waveform displays in overview."));
items.push_back(it);
it = new QStandardItem(QString::number(chan->lowerThreshold(),'f',1)); it = new QStandardItem(QString::number(chan->lowerThreshold(),'f',1));
it->setToolTip(tr("Here you can set the <b>lower</b> threshold used for certain calculations on the %1 waveform").arg(chan->fullname())); it->setToolTip(tr("Here you can set the <b>lower</b> threshold used for certain calculations on the %1 waveform").arg(chan->fullname()));
it->setEditable(true); it->setEditable(true);
@ -894,10 +913,11 @@ bool PreferencesDialog::Save()
chan.setEnabled(item->checkState() == Qt::Checked ? true : false); chan.setEnabled(item->checkState() == Qt::Checked ? true : false);
chan.setFullname(item->text()); chan.setFullname(item->text());
chan.setDefaultColor(QColor(topitem->child(j,1)->data(Qt::UserRole).toUInt())); chan.setDefaultColor(QColor(topitem->child(j,1)->data(Qt::UserRole).toUInt()));
chan.setLowerThreshold(topitem->child(j,2)->text().toDouble()); chan.setShowInOverview(topitem->child(j,2)->checkState() == Qt::Checked);
chan.setUpperThreshold(topitem->child(j,3)->text().toDouble()); chan.setLowerThreshold(topitem->child(j,3)->text().toDouble());
chan.setLabel(topitem->child(j,4)->text()); chan.setUpperThreshold(topitem->child(j,4)->text().toDouble());
chan.setDescription(topitem->child(j,5)->text()); chan.setLabel(topitem->child(j,5)->text());
chan.setDescription(topitem->child(j,6)->text());
} }
} }
@ -927,8 +947,9 @@ bool PreferencesDialog::Save()
} }
} }
chan.setType(type); chan.setType(type);
chan.setLabel(topitem->child(j,3)->text()); chan.setShowInOverview(topitem->child(j,3)->checkState() == Qt::Checked);
chan.setDescription(topitem->child(j,4)->text()); chan.setLabel(topitem->child(j,4)->text());
chan.setDescription(topitem->child(j,5)->text());
} }
} }

View File

@ -115,7 +115,6 @@ private:
MySortFilterProxyModel * waveFilterModel; MySortFilterProxyModel * waveFilterModel;
QStandardItemModel *waveModel; QStandardItemModel *waveModel;
}; };

View File

@ -7,6 +7,8 @@
* distribution for more details. */ * distribution for more details. */
#include <QApplication> #include <QApplication>
#include <QFile>
#include <QDataStream>
#include <cmath> #include <cmath>
#include "mainwindow.h" #include "mainwindow.h"
@ -23,6 +25,363 @@ QString formatTime(float time)
return QString().sprintf("%02i:%02i", hours, minutes); //,seconds); return QString().sprintf("%02i:%02i", hours, minutes); //,seconds);
} }
QDataStream & operator>>(QDataStream & in, RXItem & rx)
{
in >> rx.start;
in >> rx.end;
in >> rx.days;
in >> rx.ahi;
in >> rx.rdi;
in >> rx.hours;
QString loadername;
in >> loadername;
QString serial;
in >> serial;
MachineLoader * loader = GetLoader(loadername);
if (loader) {
rx.machine = loader->lookupMachine(serial);
} else {
qDebug() << "Bad machine object" << loadername << serial;
rx.machine = nullptr;
}
in >> rx.relief;
in >> rx.mode;
in >> rx.pressure;
QList<QDate> list;
in >> list;
rx.dates.clear();
for (int i=0; i<list.size(); ++i) {
QDate date = list.at(i);
rx.dates[date] = p_profile->FindDay(date, MT_CPAP);
}
in >> rx.s_count;
in >> rx.s_sum;
return in;
}
QDataStream & operator<<(QDataStream & out, const RXItem & rx)
{
out << rx.start;
out << rx.end;
out << rx.days;
out << rx.ahi;
out << rx.rdi;
out << rx.hours;
out << rx.machine->loaderName();
out << rx.machine->serial();
out << rx.relief;
out << rx.mode;
out << rx.pressure;
out << rx.dates.keys();
out << rx.s_count;
out << rx.s_sum;
return out;
}
void Statistics::loadRXChanges()
{
QString path = p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" );
QFile file(path);
if (!file.open(QFile::ReadOnly)) {
return;
}
QDataStream in(&file);
in.setByteOrder(QDataStream::LittleEndian);
quint32 mag32;
if (in.version() != QDataStream::Qt_5_0) {
}
in >> mag32;
if (mag32 != magic) {
return;
}
quint16 version;
in >> version;
in >> rxitems;
}
void Statistics::saveRXChanges()
{
QString path = p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" );
QFile file(path);
if (!file.open(QFile::WriteOnly)) {
return;
}
QDataStream out(&file);
out.setByteOrder(QDataStream::LittleEndian);
out.setVersion(QDataStream::Qt_5_0);
out << magic;
out << (quint16)0;
out << rxitems;
}
bool rxAHILessThan(const RXItem * rx1, const RXItem * rx2)
{
return (double(rx1->ahi) / rx1->hours) < (double(rx2->ahi) / rx2->hours);
}
void Statistics::updateRXChanges()
{
rxitems.clear();
loadRXChanges();
QMap<QDate, Day *>::iterator di;
QMap<QDate, Day *>::iterator it;
QMap<QDate, Day *>::iterator it_end = p_profile->daylist.end();
QMap<QDate, RXItem>::iterator ri;
QMap<QDate, RXItem>::iterator ri_end = rxitems.end();
quint64 tmp;
for (it = p_profile->daylist.begin(); it != it_end; ++it) {
const QDate & date = it.key();
Day * day = it.value();
Machine * mach = day->machine(MT_CPAP);
if (mach == nullptr) continue;
bool fnd = false;
ri_end = rxitems.end();
for (ri = rxitems.begin(); ri != ri_end; ++ri) {
RXItem & rx = ri.value();
if ((date >= rx.start) && (date <= rx.end)) {
// Fits in date range
if (rx.dates.contains(date)) {
fnd = true;
break;
}
// First up, check if fits in date range, but isn't loaded for some reason
// Need summaries for this
day->OpenSummary();
QList<ChannelID> flags = day->getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN);
QString relief = day->getPressureRelief();
QString mode = day->getCPAPMode();
QString pressure = day->getPressureSettings();
if ((rx.relief == relief) && (rx.mode == mode) && (rx.pressure == pressure) && (rx.machine == mach)) {
for (int i=0; i < flags.size(); i++) {
ChannelID code = flags.at(i);
rx.s_count[code] += day->count(code);
rx.s_sum[code] += day->sum(code);
}
tmp = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Apnea) + day->count(CPAP_ClearAirway);
rx.ahi += tmp;
rx.rdi += tmp + day->count(CPAP_RERA);
rx.hours += day->hours(MT_CPAP);
rx.dates[date] = day;
rx.days = rx.dates.size();
fnd = true;
break;
} else {
// Bleh.... split the day record!
RXItem rx1, rx2;
// First create the new day..
rx1.start = date;
rx1.end = date;
rx1.days = 1;
tmp = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Apnea) + day->count(CPAP_ClearAirway);
rx1.ahi = tmp;
rx1.rdi = tmp + day->count(CPAP_RERA);
for (int i=0; i < flags.size(); i++) {
ChannelID code = flags.at(i);
rx1.s_count[code] = day->count(code);
rx1.s_sum[code] = day->sum(code);
}
rx1.hours = day->hours(MT_CPAP);
rx1.relief = relief;
rx1.mode = mode;
rx1.pressure = pressure;
rx1.machine = day->machine(MT_CPAP);
rx1.dates[date] = day;
rxitems.insert(date, rx1);
QMap<QDate, Day *> datecopy = rx.dates;
rx.dates.clear();
rx.end = rx.start;
rx2.start = rx.end;
rx2.end = rx.start;
rx2.ahi = 0;
rx2.rdi = 0;
rx2.hours = 0;
rx.ahi = 0;
rx.rdi = 0;
rx.hours = 0;
rx.s_count.clear();
rx2.s_count.clear();
rx.s_sum.clear();
rx2.s_sum.clear();
for (di = datecopy.begin(); di != datecopy.end(); ++di) {
// Split everything before
if (di.key() < date) {
Day * dy = rx.dates[di.key()] = p_profile->GetDay(di.key(), MT_CPAP);
tmp = dy->count(CPAP_Hypopnea) + dy->count(CPAP_Obstructive) + dy->count(CPAP_Apnea) + dy->count(CPAP_ClearAirway);;
rx.ahi += tmp;
rx.rdi += tmp + dy->count(CPAP_RERA);
QList<ChannelID> flags2 = dy->getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN);
for (int i=0; i < flags2.size(); i++) {
ChannelID code = flags2.at(i);
rx.s_count[code] += dy->count(code);
rx.s_sum[code] += dy->sum(code);
}
rx.hours += dy->hours(MT_CPAP);
//rx.days++;
rx.end = qMax(di.key(), rx.end);
}
// Split everything after
if (di.key() > date) {
Day * dy = rx2.dates[di.key()] = p_profile->GetDay(di.key(), MT_CPAP);
tmp = dy->count(CPAP_Hypopnea) + dy->count(CPAP_Obstructive) + dy->count(CPAP_Apnea) + dy->count(CPAP_ClearAirway);;
rx2.ahi += tmp;
rx2.rdi += tmp + dy->count(CPAP_RERA);
QList<ChannelID> flags2 = dy->getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN);
for (int i=0; i < flags2.size(); i++) {
ChannelID code = flags2.at(i);
rx2.s_count[code] += dy->count(code);
rx2.s_sum[code] += dy->sum(code);
}
rx2.hours += dy->hours(MT_CPAP);
rx2.end = qMax(di.key(), rx2.end);
rx2.start = qMin(di.key(), rx2.start);
}
}
rx.days = rx.dates.size();
rx2.pressure = rx.pressure;
rx2.mode = rx.mode;
rx2.relief = rx.relief;
rx2.machine = rx.machine;
rx2.days = rx2.dates.size();
rxitems.insert(rx2.end, rx2);
fnd = true;
break;
}
}
}
if (fnd) continue;
day->OpenSummary();
QList<ChannelID> flags3 = day->getSortedMachineChannels(MT_CPAP, schema::FLAG | schema::MINOR_FLAG | schema::SPAN);
QString relief = day->getPressureRelief();
QString mode = day->getCPAPMode();
QString pressure = day->getPressureSettings();
for (ri = rxitems.begin(); ri != ri_end; ++ri) {
RXItem & rx = ri.value();
if (rx.end.daysTo(date) == 1) {
if ((rx.relief == relief) && (rx.mode == mode) && (rx.pressure == pressure) && (rx.machine == day->machine(MT_CPAP)) ) {
tmp = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Apnea) + day->count(CPAP_ClearAirway);
rx.ahi += tmp;
rx.rdi += tmp + day->count(CPAP_RERA);
for (int i=0; i < flags3.size(); i++) {
ChannelID code = flags3.at(i);
rx.s_count[code] += day->count(code);
rx.s_sum[code] += day->sum(code);
}
rx.hours += day->hours(MT_CPAP);
rx.dates[date] = day;
rx.days = rx.dates.size();
rx.end = date;
fnd = true;
break;
}
}
}
if (!fnd) {
RXItem rx;
rx.start = date;
rx.end = date;
rx.days = 1;
tmp = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Apnea) + day->count(CPAP_ClearAirway);
rx.ahi = tmp;
rx.rdi = tmp + day->count(CPAP_RERA);
for (int i=0; i < flags3.size(); i++) {
ChannelID code = flags3.at(i);
rx.s_count[code] = day->count(code);
rx.s_sum[code] = day->sum(code);
}
rx.hours = day->hours();
rx.relief = relief;
rx.mode = mode;
rx.pressure = pressure;
rx.machine = day->machine(MT_CPAP);
rx.dates.insert(date, day);
rxitems.insert(date, rx);
}
}
saveRXChanges();
QList<RXItem *> list;
ri_end = rxitems.end();
for (ri = rxitems.begin(); ri != ri_end; ++ri) {
list.append(&ri.value());
ri.value().highlight = 0;
}
qSort(list.begin(), list.end(), rxAHILessThan);
if (list.size() >= 4) {
list[0]->highlight = 1; // best
list[1]->highlight = 2; // best
int ls = list.size() - 1;
list[ls-1]->highlight = 3; // best
list[ls]->highlight = 4;
} else if (list.size() >= 2) {
list[0]->highlight = 1; // best
int ls = list.size() - 1;
list[ls]->highlight = 4;
} else if (list.size() > 0) {
list[0]->highlight = 1; // best
}
// update record box info..
}
Statistics::Statistics(QObject *parent) : Statistics::Statistics(QObject *parent) :
QObject(parent) QObject(parent)
@ -503,11 +862,178 @@ struct Period {
QString header; QString header;
}; };
const QString heading_color="#ffffff";
const QString subheading_color="#e0e0e0";
const int rxthresh = 5;
QString Statistics::GenerateMachineList()
{
QList<Machine *> cpap_machines = p_profile->GetMachines(MT_CPAP);
QList<Machine *> oximeters = p_profile->GetMachines(MT_OXIMETER);
QList<Machine *> mach;
mach.append(cpap_machines);
mach.append(oximeters);
QString html;
if (mach.size() > 0) {
html += "<div align=center><br/>";
html += QString("<table class=curved style=\"page-break-before:auto;\" "+table_width+">");
html += "<thead>";
html += "<tr bgcolor='"+heading_color+"'><th colspan=7 align=center><font size=+2>" + tr("Machine Information") + "</font></th></tr>";
html += QString("<tr><td><b>%1</b></td><td><b>%2</b></td><td><b>%3</b></td><td><b>%4</b></td><td><b>%5</b></td><td><b>%6</b></td></tr>")
.arg(STR_TR_Brand)
.arg(STR_TR_Series)
.arg(STR_TR_Model)
.arg(STR_TR_Serial)
.arg(tr("First Use"))
.arg(tr("Last Use"));
html += "</thead>";
Machine *m;
for (int i = 0; i < mach.size(); i++) {
m = mach.at(i);
if (m->type() == MT_JOURNAL) { continue; }
QDate d1 = m->FirstDay();
QDate d2 = m->LastDay();
QString mn = m->modelnumber();
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td></tr>")
.arg(m->brand())
.arg(m->series())
.arg(m->model() +
(mn.isEmpty() ? "" : QString(" (") + mn + QString(")")))
.arg(m->serial())
.arg(d1.toString(Qt::SystemLocaleShortDate))
.arg(d2.toString(Qt::SystemLocaleShortDate));
}
html += "</table>";
html += "</div>";
}
return html;
}
QString Statistics::GenerateRXChanges()
{
updateRXChanges();
QString ahitxt;
bool rdi = p_profile->general->calculateRDI();
if (rdi) {
ahitxt = STR_TR_RDI;
} else {
ahitxt = STR_TR_AHI;
}
QString html = "<div align=center><br/>";
html += QString("<table class=curved style=\"page-break-before:always;\" "+table_width+">");
html += "<thead>";
html += "<tr bgcolor='"+heading_color+"'><th colspan=10 align=center><font size=+2>" + tr("Changes to Prescription Settings") + "</font></th></tr>";
QString extratxt;
QString tooltip;
QStringList hdrlist;
hdrlist.push_back(STR_TR_First);
hdrlist.push_back(STR_TR_Last);
hdrlist.push_back(tr("Days"));
hdrlist.push_back(ahitxt);
hdrlist.push_back(STR_TR_FL);
hdrlist.push_back(STR_TR_Machine);
hdrlist.push_back(tr("Pressure Relief"));
hdrlist.push_back(STR_TR_Mode);
hdrlist.push_back(tr("Pressure Settings"));
html+="<tr>\n";
for (int i=0; i < hdrlist.size(); ++i) {
html+=QString(" <th align=left><b>%1</b></th>\n").arg(hdrlist.at(i));
}
html+="</tr>\n";
html += "</thead>";
// html += "<tfoot>";
// html += "<tr><td colspan=10 align=center>";
// html += QString("<i>") +
// tr("Efficacy highlighting ignores prescription settings with less than %1 days of recorded data.").
// arg(rxthresh) + QString("</i><br/>");
// html += "</td></tr>";
// html += "</tfoot>";
QMapIterator<QDate, RXItem> it(rxitems);
it.toBack();
while (it.hasPrevious()) {
it.previous();
const RXItem & rx = it.value();
QString color;
if (rx.highlight == 1) {
color = "#c0ffc0";
} else if (rx.highlight == 2) {
color = "#e0ffe0";
} else if (rx.highlight == 3) {
color = "#ffe0e0";
} else if (rx.highlight == 4) {
color = "#ffc0c0";
} else { color = ""; }
QString datarowclass;
if (rx.highlight == 0) datarowclass="class=datarow";
html += QString("<tr %4 bgcolor='%1' onmouseover='ChangeColor(this, \"#dddddd\");' onmouseout='ChangeColor(this, \"%1\");' onclick='alert(\"overview=%2,%3\");'>")
.arg(color)
.arg(rx.start.toString(Qt::ISODate))
.arg(rx.end.toString(Qt::ISODate))
.arg(datarowclass);
double ahi = rdi ? (double(rx.rdi) / rx.hours) : (double(rx.ahi) /rx.hours);
double fli = double(rx.count(CPAP_FlowLimit)) / rx. hours;
html += QString("<td>%1</td>").arg(rx.start.toString())+
QString("<td>%1</td>").arg(rx.end.toString())+
QString("<td>%1</td>").arg(rx.days)+
QString("<td>%1</td>").arg(ahi, 0, 'f', 2)+
QString("<td>%1</td>").arg(fli, 0, 'f', 2)+
QString("<td>%1</td>").arg(rx.machine->loaderName())+
QString("<td>%1</td>").arg(rx.relief)+
QString("<td>%1</td>").arg(rx.mode)+
QString("<td>%1</td>").arg(rx.pressure)+
"</tr>";
}
html+="</table></div>";
return html;
}
QString Statistics::GenerateHTML() QString Statistics::GenerateHTML()
{ {
QList<Machine *> cpap_machines = p_profile->GetMachines(MT_CPAP);
QList<Machine *> oximeters = p_profile->GetMachines(MT_OXIMETER);
QList<Machine *> mach;
QString heading_color="#ffffff"; mach.append(cpap_machines);
QString subheading_color="#e0e0e0"; mach.append(oximeters);
bool havedata = false;
for (int i=0; i < mach.size(); ++i) {
int daysize = mach[i]->day.size();
if (daysize > 0) {
havedata = true;
break;
}
}
QString html = htmlHeader(havedata);
// return "";
// Find first and last days with valid CPAP data // Find first and last days with valid CPAP data
@ -525,23 +1051,6 @@ QString Statistics::GenerateHTML()
if (cpap6month < firstcpap) { cpap6month = firstcpap; } if (cpap6month < firstcpap) { cpap6month = firstcpap; }
if (cpapyear < firstcpap) { cpapyear = firstcpap; } if (cpapyear < firstcpap) { cpapyear = firstcpap; }
QList<Machine *> cpap_machines = p_profile->GetMachines(MT_CPAP);
QList<Machine *> oximeters = p_profile->GetMachines(MT_OXIMETER);
QList<Machine *> mach;
mach.append(cpap_machines);
mach.append(oximeters);
bool havedata = false;
for (int i=0; i < mach.size(); ++i) {
int daysize = mach[i]->day.size();
if (daysize > 0) {
havedata = true;
break;
}
}
QString html = htmlHeader(havedata);
if (!havedata) { if (!havedata) {
html += "<div align=center><table class=curved height=100% "+table_width+">"; html += "<div align=center><table class=curved height=100% "+table_width+">";
@ -555,6 +1064,7 @@ QString Statistics::GenerateHTML()
return html; return html;
} }
int cpapdays = p_profile->countDays(MT_CPAP, firstcpap, lastcpap); int cpapdays = p_profile->countDays(MT_CPAP, firstcpap, lastcpap);
// CPAPMode cpapmode = (CPAPMode)(int)p_profile->calcSettingsMax(CPAP_Mode, MT_CPAP, firstcpap, lastcpap); // CPAPMode cpapmode = (CPAPMode)(int)p_profile->calcSettingsMax(CPAP_Mode, MT_CPAP, firstcpap, lastcpap);
@ -592,7 +1102,6 @@ QString Statistics::GenerateHTML()
QList<Period> periods; QList<Period> periods;
bool skipsection = false;; bool skipsection = false;;
for (QList<StatisticsRow>::iterator i = rows.begin(); i != rows.end(); ++i) { for (QList<StatisticsRow>::iterator i = rows.begin(); i != rows.end(); ++i) {
StatisticsRow &row = (*i); StatisticsRow &row = (*i);
@ -702,7 +1211,7 @@ QString Statistics::GenerateHTML()
continue; continue;
} else { } else {
ChannelID id = schema::channel[row.src].id(); ChannelID id = schema::channel[row.src].id();
if ((id == NoChannel) || (!p_profile->hasChannel(id))) { if ((id == NoChannel) || (!p_profile->channelAvailable(id))) {
continue; continue;
} }
name = calcnames[row.calc].arg(schema::channel[id].fullname()); name = calcnames[row.calc].arg(schema::channel[id].fullname());
@ -733,6 +1242,7 @@ QString Statistics::GenerateHTML()
html += "</table>"; html += "</table>";
html += "</div>"; html += "</div>";
/*
QList<UsageData> AHI; QList<UsageData> AHI;
if (cpapdays > 0) { if (cpapdays > 0) {
@ -1083,13 +1593,13 @@ QString Statistics::GenerateHTML()
recbox += "</table>"; recbox += "</table>";
recbox += "</body></html>"; recbox += "</body></html>";
mainwin->setRecBoxHTML(recbox); mainwin->setRecBoxHTML(recbox); */
/*RXsort=RX_min; /*RXsort=RX_min;
RXorder=true; RXorder=true;
qSort(rxchange.begin(),rxchange.end());*/ qSort(rxchange.begin(),rxchange.end());*/
html += "<div align=center><br/>"; /* html += "<div align=center><br/>";
html += QString("<table class=curved style=\"page-break-before:always;\" "+table_width+">"); html += QString("<table class=curved style=\"page-break-before:always;\" "+table_width+">");
html += "<thead>"; html += "<thead>";
html += "<tr bgcolor='"+heading_color+"'><th colspan=10 align=center><font size=+2>" + tr("Changes to Prescription Settings") + "</font></th></tr>"; html += "<tr bgcolor='"+heading_color+"'><th colspan=10 align=center><font size=+2>" + tr("Changes to Prescription Settings") + "</font></th></tr>";
@ -1232,57 +1742,10 @@ QString Statistics::GenerateHTML()
html += "</table>"; html += "</table>";
html += "</div>"; html += "</div>";
} } */
html += GenerateRXChanges();
html += GenerateMachineList();
if (mach.size() > 0) {
html += "<div align=center><br/>";
html += QString("<table class=curved style=\"page-break-before:auto;\" "+table_width+">");
html += "<thead>";
html += "<tr bgcolor='"+heading_color+"'><th colspan=7 align=center><font size=+2>" + tr("Machine Information") + "</font></th></tr>";
html += QString("<tr><td><b>%1</b></td><td><b>%2</b></td><td><b>%3</b></td><td><b>%4</b></td><td><b>%5</b></td><td><b>%6</b></td><td><b>%7</b></td></tr>")
.arg(STR_TR_Brand)
.arg(STR_TR_Series)
.arg(STR_TR_Model)
.arg(STR_TR_Serial)
.arg(tr("First Use"))
.arg(tr("Last Use"))
.arg(STR_TR_AHI);
html += "</thead>";
Machine *m;
for (int i = 0; i < mach.size(); i++) {
m = mach.at(i);
if (m->type() == MT_JOURNAL) { continue; }
QDate d1 = m->FirstDay();
QDate d2 = m->LastDay();
QString ahi;
if (m->type() == MT_CPAP) {
float a = calcAHI(d1,d2);
ahi = QString::number(a,'f',2);
}
QString mn = m->modelnumber();
html += QString("<tr><td>%1</td><td>%2</td><td>%3</td><td>%4</td><td>%5</td><td>%6</td><td>%7</td></tr>")
.arg(m->brand())
.arg(m->series())
.arg(m->model() +
(mn.isEmpty() ? "" : QString(" (") + mn + QString(")")))
.arg(m->serial())
.arg(d1.toString(Qt::SystemLocaleShortDate))
.arg(d2.toString(Qt::SystemLocaleShortDate))
.arg(ahi);
}
html += "</table>";
html += "</div>";
}
html += "<script type='text/javascript' language='javascript' src='qrc:/docs/script.js'></script>"; html += "<script type='text/javascript' language='javascript' src='qrc:/docs/script.js'></script>";
//updateFavourites(); //updateFavourites();

View File

@ -13,6 +13,7 @@
#include <QHash> #include <QHash>
#include <QList> #include <QList>
#include "SleepLib/schema.h" #include "SleepLib/schema.h"
#include "SleepLib/machine.h"
enum StatCalcType { enum StatCalcType {
SC_UNDEFINED=0, SC_COLUMNHEADERS, SC_HEADING, SC_SUBHEADING, SC_MEDIAN, SC_AVG, SC_WAVG, SC_90P, SC_MIN, SC_MAX, SC_CPH, SC_SPH, SC_AHI, SC_HOURS, SC_COMPLIANCE, SC_DAYS, SC_ABOVE, SC_BELOW SC_UNDEFINED=0, SC_COLUMNHEADERS, SC_HEADING, SC_SUBHEADING, SC_MEDIAN, SC_AVG, SC_WAVG, SC_90P, SC_MIN, SC_MAX, SC_CPH, SC_SPH, SC_AHI, SC_HOURS, SC_COMPLIANCE, SC_DAYS, SC_ABOVE, SC_BELOW
@ -93,13 +94,70 @@ struct StatisticsRow {
QString value(QDate start, QDate end); QString value(QDate start, QDate end);
}; };
class RXItem {
public:
RXItem() {
machine = nullptr;
ahi = rdi = 0;
highlight = 0;
hours = 0;
}
RXItem(const RXItem & copy) {
start = copy.start;
end = copy.end;
days = copy.days;
s_count = copy.s_count;
s_sum = copy.s_sum;
ahi = copy.ahi;
rdi = copy.rdi;
hours = copy.hours;
machine = copy.machine;
relief = copy.relief;
mode = copy.mode;
pressure = copy.pressure;
dates = copy.dates;
highlight = copy.highlight;
}
inline quint64 count(ChannelID id) const {
QHash<ChannelID, quint64>::const_iterator it = s_count.find(id);
if (it == s_count.end()) return 0;
return it.value();
}
inline double sum(ChannelID id) const{
QHash<ChannelID, double>::const_iterator it = s_sum.find(id);
if (it == s_sum.end()) return 0;
return it.value();
}
QDate start;
QDate end;
int days;
QHash<ChannelID, quint64> s_count;
QHash<ChannelID, double> s_sum;
quint64 ahi;
quint64 rdi;
double hours;
Machine * machine;
QString relief;
QString mode;
QString pressure;
QMap<QDate, Day *> dates;
short highlight;
};
class Statistics : public QObject class Statistics : public QObject
{ {
Q_OBJECT Q_OBJECT
public: public:
explicit Statistics(QObject *parent = 0); explicit Statistics(QObject *parent = 0);
void loadRXChanges();
void saveRXChanges();
void updateRXChanges();
QString GenerateHTML(); QString GenerateHTML();
QString GenerateMachineList();
QString GenerateRXChanges();
protected: protected:
// Using a map to maintain order // Using a map to maintain order
@ -107,6 +165,8 @@ class Statistics : public QObject
QMap<StatCalcType, QString> calcnames; QMap<StatCalcType, QString> calcnames;
QMap<MachineType, QString> machinenames; QMap<MachineType, QString> machinenames;
QMap<QDate, RXItem> rxitems;
signals: signals:
public slots: public slots: