mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
Removed machine section out of profile.xml to machine.xml, created profiles.xml list, set some default overview graphs
This commit is contained in:
parent
b3510d788b
commit
d1341787ba
@ -676,6 +676,15 @@ void gGraph::AddLayer(Layer *l, LayerPosition position, short width, short heigh
|
||||
l->addref();
|
||||
m_layers.push_back(l);
|
||||
}
|
||||
|
||||
void gGraph::dataChanged()
|
||||
{
|
||||
int size = m_layers.size();
|
||||
for (int i=0; i < size; i++) {
|
||||
m_layers[i]->dataChanged();
|
||||
}
|
||||
}
|
||||
|
||||
void gGraph::redraw()
|
||||
{
|
||||
m_graphview->redraw();
|
||||
|
@ -265,6 +265,8 @@ class gGraph : public QObject
|
||||
|
||||
double screenToTime(int xpos);
|
||||
|
||||
void dataChanged();
|
||||
|
||||
//! \brief Sets the margins for the four sides of this graph.
|
||||
void setMargins(short left, short right, short top, short bottom) {
|
||||
m_marginleft = left;
|
||||
|
@ -3365,6 +3365,14 @@ int gGraphView::visibleGraphs()
|
||||
return cnt;
|
||||
}
|
||||
|
||||
void gGraphView::dataChanged()
|
||||
{
|
||||
for (int i = 0; i < m_graphs.size(); i++) {
|
||||
m_graphs[i]->dataChanged();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void gGraphView::redraw()
|
||||
{
|
||||
#ifdef BROKEN_OPENGL_BUILD
|
||||
|
@ -676,6 +676,8 @@ class gGraphView
|
||||
ResetBounds(true);
|
||||
}
|
||||
|
||||
void dataChanged();
|
||||
|
||||
|
||||
bool hasSnapshots();
|
||||
|
||||
|
@ -154,10 +154,7 @@ void gSummaryChart::preCalc()
|
||||
{
|
||||
for (int i=0; i<calcitems.size(); ++i) {
|
||||
SummaryCalcItem & calc = calcitems[i];
|
||||
calc.min = 99999;
|
||||
calc.max = -99999;
|
||||
calc.sum = 0;
|
||||
calc.divisor = 0;
|
||||
calc.reset(idx_end - idx_start);
|
||||
}
|
||||
}
|
||||
|
||||
@ -171,21 +168,7 @@ void gSummaryChart::customCalc(Day *day, QList<SummaryChartSlice> & slices)
|
||||
const SummaryChartSlice & slice = slices.at(i);
|
||||
SummaryCalcItem & calc = calcitems[i];
|
||||
|
||||
calc.min = qMin(calc.min, slice.value);
|
||||
calc.max = qMax(calc.max, slice.value);
|
||||
|
||||
switch (calc.type) {
|
||||
case ST_CPH:
|
||||
case ST_SPH:
|
||||
calc.sum += slice.value * hour;
|
||||
calc.divisor += hour;
|
||||
break;
|
||||
default:
|
||||
calc.sum += slice.value;
|
||||
calc.divisor += 1;
|
||||
break;
|
||||
|
||||
}
|
||||
calc.update(slice.value, hour);
|
||||
}
|
||||
}
|
||||
|
||||
@ -198,37 +181,54 @@ void gSummaryChart::afterDraw(QPainter &painter, gGraph &graph, QRect rect)
|
||||
QStringList strlist;
|
||||
QString txt;
|
||||
|
||||
int mid = p_profile->general->prefCalcMiddle();
|
||||
int midcalc = p_profile->general->prefCalcMiddle();
|
||||
QString midstr;
|
||||
if (mid == 0) {
|
||||
if (midcalc == 0) {
|
||||
midstr = QObject::tr("Med.");
|
||||
} else if (mid == 1) {
|
||||
midstr = QObject::tr("Avg");
|
||||
} else {
|
||||
} else if (midcalc == 1) {
|
||||
midstr = QObject::tr("W-Avg");
|
||||
} else {
|
||||
midstr = QObject::tr("Avg");
|
||||
}
|
||||
|
||||
|
||||
float perc = p_profile->general->prefCalcPercentile();
|
||||
QString percstr = QObject::tr("%1%").arg(perc, 0, 'f',0);
|
||||
|
||||
schema::Channel & chan = schema::channel[calcitems.at(0).code];
|
||||
|
||||
for (int i=0; i<calcitems.size(); ++i) {
|
||||
const SummaryCalcItem & calc = calcitems.at(i);
|
||||
SummaryCalcItem & calc = calcitems[i];
|
||||
if (calcitems.size() == 1) {
|
||||
float val = calc.min;
|
||||
if (val < 99998)
|
||||
strlist.append(QObject::tr("Min: %1").arg(val,0,'f',2));
|
||||
}
|
||||
|
||||
float mid = 0;
|
||||
switch (midcalc) {
|
||||
case 0:
|
||||
if (calc.median_data.size() > 0) {
|
||||
mid = median(calc.median_data.begin(), calc.median_data.end());
|
||||
}
|
||||
break;
|
||||
case 1:
|
||||
mid = calc.wavg_sum / calc.divisor;
|
||||
break;
|
||||
case 2:
|
||||
mid = calc.avg_sum / calc.cnt;
|
||||
break;
|
||||
}
|
||||
|
||||
float val = 0;
|
||||
switch (calc.type) {
|
||||
case ST_CPH:
|
||||
val = calc.sum / calc.divisor;
|
||||
txt = QObject::tr("Avg: ");
|
||||
val = mid;
|
||||
txt = midstr+": ";
|
||||
break;
|
||||
case ST_SPH:
|
||||
val = calc.sum / calc.divisor;
|
||||
txt = QObject::tr("Avg: ");
|
||||
val = mid;
|
||||
txt = midstr+": ";
|
||||
break;
|
||||
case ST_MIN:
|
||||
val = calc.min;
|
||||
@ -251,15 +251,15 @@ void gSummaryChart::afterDraw(QPainter &painter, gGraph &graph, QRect rect)
|
||||
txt = QObject::tr("Max: ");
|
||||
break;
|
||||
case ST_MID:
|
||||
val = calc.sum / calc.divisor;
|
||||
val = mid;
|
||||
txt = QObject::tr("%1: ").arg(midstr);
|
||||
break;
|
||||
case ST_90P:
|
||||
val = calc.sum / calc.divisor;
|
||||
val = mid;
|
||||
txt = QObject::tr("%1: ").arg(percstr);
|
||||
break;
|
||||
default:
|
||||
val = calc.sum / calc.divisor;
|
||||
val = mid;
|
||||
txt = QObject::tr("???: ");
|
||||
break;
|
||||
}
|
||||
@ -421,7 +421,7 @@ void gSummaryChart::paint(QPainter &painter, gGraph &graph, const QRegion ®io
|
||||
nousedays = 0;
|
||||
totaldays = 0;
|
||||
|
||||
QRectF hl_rect, hl2_rect;
|
||||
QRectF hl_rect;
|
||||
QDate hl_date;
|
||||
Day * hl_day = nullptr;
|
||||
int hl_idx = -1;
|
||||
@ -555,7 +555,6 @@ void gSummaryChart::paint(QPainter &painter, gGraph &graph, const QRegion ®io
|
||||
|
||||
for (int i=0; i < listsize; ++i) {
|
||||
SummaryChartSlice & slice = list[i];
|
||||
SummaryCalcItem * calc = slice.calc;
|
||||
|
||||
val = slice.height;
|
||||
y1 = ((lastval-miny) * ymult);
|
||||
@ -654,20 +653,24 @@ void gUsageChart::preCalc()
|
||||
{
|
||||
compliance_threshold = p_profile->cpap->complianceHours();
|
||||
incompdays = 0;
|
||||
totalhours = 0;
|
||||
totaldays = 0;
|
||||
|
||||
hour_data.clear();
|
||||
hour_data.reserve(idx_end-idx_start);
|
||||
SummaryCalcItem & calc = calcitems[0];
|
||||
calc.reset(idx_end - idx_start);
|
||||
}
|
||||
|
||||
void gUsageChart::customCalc(Day *, QList<SummaryChartSlice> &list)
|
||||
{
|
||||
if (list.size() == 0) {
|
||||
incompdays++;
|
||||
return;
|
||||
}
|
||||
|
||||
SummaryChartSlice & slice = list[0];
|
||||
SummaryCalcItem & calc = calcitems[0];
|
||||
|
||||
if (slice.value < compliance_threshold) incompdays++;
|
||||
totalhours += slice.value;
|
||||
hour_data.append(slice.value);
|
||||
totaldays++;
|
||||
|
||||
calc.update(slice.value, 1);
|
||||
}
|
||||
|
||||
void gUsageChart::afterDraw(QPainter &, gGraph &graph, QRect rect)
|
||||
@ -676,30 +679,106 @@ void gUsageChart::afterDraw(QPainter &, gGraph &graph, QRect rect)
|
||||
|
||||
if (totaldays > 1) {
|
||||
float comp = 100.0 - ((float(incompdays + nousedays) / float(totaldays)) * 100.0);
|
||||
double avg = totalhours / double(totaldays);
|
||||
|
||||
float med = median(hour_data.begin(), hour_data.end());
|
||||
QString txt = QObject::tr("%1 low usage, %2 no usage, out of %3 days (%4% compliant.) Avg %5, Med %6 hours").
|
||||
arg(incompdays).arg(nousedays).arg(totaldays).arg(comp,0,'f',1).arg(avg, 0, 'f', 2).arg(med, 0, 'f', 2);
|
||||
int midcalc = p_profile->general->prefCalcMiddle();
|
||||
float mid = 0;
|
||||
SummaryCalcItem & calc = calcitems[0];
|
||||
switch (midcalc) {
|
||||
case 0: // median
|
||||
mid = median(calc.median_data.begin(), calc.median_data.end());
|
||||
break;
|
||||
case 1: // w-avg
|
||||
mid = calc.wavg_sum / calc.divisor;
|
||||
break;
|
||||
case 2:
|
||||
mid = calc.avg_sum / calc.cnt;
|
||||
break;
|
||||
}
|
||||
|
||||
QString txt = QObject::tr("%1 low usage, %2 no usage, out of %3 days (%4% compliant.) Length: %5 / %6 / %7").
|
||||
arg(incompdays).arg(nousedays).arg(totaldays).arg(comp,0,'f',1).arg(calc.min, 0, 'f', 2).arg(mid, 0, 'f', 2).arg(calc.max, 0, 'f', 2);;
|
||||
graph.renderText(txt, rect.left(), rect.top()-5*graph.printScaleY(), 0);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void gSessionTimesChart::preCalc() {
|
||||
num_slices = 0;
|
||||
num_days = 0;
|
||||
total_length = 0;
|
||||
SummaryCalcItem & calc = calcitems[0];
|
||||
|
||||
calc.reset((idx_end - idx_start) * 6);
|
||||
|
||||
SummaryCalcItem & calc1 = calcitems[1];
|
||||
|
||||
calc1.reset(idx_end - idx_start);
|
||||
|
||||
SummaryCalcItem & calc2 = calcitems[2];
|
||||
calc2.reset(idx_end - idx_start);
|
||||
}
|
||||
|
||||
void gSessionTimesChart::customCalc(Day *, QList<SummaryChartSlice> & slices) {
|
||||
int size = slices.size();
|
||||
num_slices += size;
|
||||
|
||||
SummaryCalcItem & calc1 = calcitems[1];
|
||||
calc1.update(slices.size(), 1);
|
||||
|
||||
EventDataType max = 0;
|
||||
for (int i=0; i<size; ++i) {
|
||||
SummaryChartSlice & slice = slices[i];
|
||||
SummaryCalcItem & calc = *slice.calc;
|
||||
|
||||
calc.update(slice.height, slice.height);
|
||||
|
||||
max = qMax(slice.height, max);
|
||||
}
|
||||
SummaryCalcItem & calc2 = calcitems[2];
|
||||
|
||||
calc2.update(max, max);
|
||||
|
||||
num_days++;
|
||||
}
|
||||
|
||||
void gSessionTimesChart::afterDraw(QPainter & /*painter */, gGraph &graph, QRect rect)
|
||||
{
|
||||
if (totaldays == nousedays) return;
|
||||
|
||||
float med = 0;
|
||||
if (session_data.size() > 0)
|
||||
med = median(session_data.begin(), session_data.end());
|
||||
SummaryCalcItem & calc = calcitems[0]; // session length
|
||||
SummaryCalcItem & calc1 = calcitems[1]; // number of sessions
|
||||
SummaryCalcItem & calc2 = calcitems[2]; // number of sessions
|
||||
|
||||
int midcalc = p_profile->general->prefCalcMiddle();
|
||||
|
||||
float mid = 0, mid1 = 0, midlongest = 0;
|
||||
switch (midcalc) {
|
||||
case 0:
|
||||
if (calc.median_data.size() > 0) {
|
||||
mid = median(calc.median_data.begin(), calc.median_data.end());
|
||||
mid1 = median(calc1.median_data.begin(), calc1.median_data.end());
|
||||
midlongest = median(calc2.median_data.begin(), calc2.median_data.end());
|
||||
}
|
||||
break;
|
||||
case 1:
|
||||
mid = calc.wavg_sum / calc.divisor;
|
||||
mid1 = calc1.wavg_sum / calc1.divisor;
|
||||
midlongest = calc2.wavg_sum / calc2.divisor;
|
||||
break;
|
||||
case 2:
|
||||
mid = calc.avg_sum / calc.cnt;
|
||||
mid1 = calc1.avg_sum / calc1.cnt;
|
||||
midlongest = calc2.avg_sum / calc2.cnt;
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
float avgsess = float(num_slices) / float(num_days);
|
||||
double avglength = total_length / double(num_slices);
|
||||
// float avgsess = float(num_slices) / float(num_days);
|
||||
|
||||
QString txt = QObject::tr("Avg Sessions: %1 Length Avg: %2 Med %3").arg(avgsess, 0, 'f', 1).arg(avglength, 0, 'f', 2).arg(med, 0, 'f', 2);
|
||||
QString txt = QObject::tr("Sessions: %1 / %2 / %3 Length: %4 / %5 / %6 Longest: %7 / %8 / %9")
|
||||
.arg(calc1.min, 0, 'f', 2).arg(mid1, 0, 'f', 2).arg(calc1.max, 0, 'f', 2)
|
||||
.arg(calc.min, 0, 'f', 2).arg(mid, 0, 'f', 2).arg(calc.max, 0, 'f', 2)
|
||||
.arg(calc2.min, 0, 'f', 2).arg(midlongest, 0, 'f', 2).arg(calc2.max, 0, 'f', 2);
|
||||
graph.renderText(txt, rect.left(), rect.top()-5*graph.printScaleY(), 0);
|
||||
}
|
||||
|
||||
@ -844,6 +923,8 @@ void gSessionTimesChart::paint(QPainter &painter, gGraph &graph, const QRegion &
|
||||
|
||||
|
||||
preCalc();
|
||||
totaldays = 0;
|
||||
nousedays = 0;
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
/// Main Loop scaling
|
||||
@ -933,7 +1014,8 @@ void gAHIChart::preCalc()
|
||||
{
|
||||
gSummaryChart::preCalc();
|
||||
|
||||
ahi_total = 0;
|
||||
ahi_wavg = 0;
|
||||
ahi_avg = 0;
|
||||
calc_cnt = 0;
|
||||
total_hours = 0;
|
||||
min_ahi = 99999;
|
||||
@ -954,11 +1036,18 @@ void gAHIChart::customCalc(Day *day, QList<SummaryChartSlice> &list)
|
||||
|
||||
EventDataType value = slice.value;
|
||||
|
||||
calc->sum += value;
|
||||
calc->wavg_sum += value;
|
||||
calc->divisor += hours;
|
||||
|
||||
calc->min = qMin(value / hours, calc->min);
|
||||
calc->max = qMax(value / hours, calc->max);
|
||||
calc->avg_sum += value;
|
||||
calc->cnt++;
|
||||
|
||||
float valh = value/ hours;
|
||||
|
||||
calc->median_data.append(valh);
|
||||
|
||||
calc->min = qMin(valh, calc->min);
|
||||
calc->max = qMax(valh, calc->max);
|
||||
|
||||
ahi_cnt += value;
|
||||
}
|
||||
@ -967,7 +1056,8 @@ void gAHIChart::customCalc(Day *day, QList<SummaryChartSlice> &list)
|
||||
|
||||
ahi_data.append(ahi_cnt / hours);
|
||||
|
||||
ahi_total += ahi_cnt;
|
||||
ahi_wavg += ahi_cnt;
|
||||
ahi_avg += ahi_cnt;
|
||||
total_hours += hours;
|
||||
calc_cnt++;
|
||||
}
|
||||
@ -977,6 +1067,8 @@ void gAHIChart::afterDraw(QPainter & /*painter */, gGraph &graph, QRect rect)
|
||||
|
||||
int size = idx_end - idx_start;
|
||||
|
||||
int midcalc = p_profile->general->prefCalcMiddle();
|
||||
|
||||
int mpos = size /2 ;
|
||||
|
||||
float med = 0;
|
||||
@ -996,8 +1088,22 @@ void gAHIChart::afterDraw(QPainter & /*painter */, gGraph &graph, QRect rect)
|
||||
if (calc.divisor > 0) {
|
||||
ChannelID code = calc.code;
|
||||
schema::Channel & chan = schema::channel[code];
|
||||
double indice = calc.sum / calc.divisor;
|
||||
txtlist.append(QString("%1 %2 / %3 / %4").arg(chan.label()).arg(calc.min, 0, 'f', 2).arg(indice, 0, 'f', 2).arg(calc.max, 0, 'f', 2));
|
||||
float mid = 0;
|
||||
switch (midcalc) {
|
||||
case 0:
|
||||
if (calc.median_data.size() > 0) {
|
||||
mid = median(calc.median_data.begin(), calc.median_data.end());
|
||||
}
|
||||
break;
|
||||
case 1:
|
||||
mid = calc.wavg_sum / calc.divisor;
|
||||
break;
|
||||
case 2:
|
||||
mid = calc.avg_sum / calc.divisor;
|
||||
break;
|
||||
}
|
||||
|
||||
txtlist.append(QString("%1 %2 / %3 / %4").arg(chan.label()).arg(calc.min, 0, 'f', 2).arg(mid, 0, 'f', 2).arg(calc.max, 0, 'f', 2));
|
||||
}
|
||||
}
|
||||
QString txt = txtlist.join(", ");
|
||||
|
@ -14,39 +14,6 @@
|
||||
#include "gGraphView.h"
|
||||
|
||||
|
||||
///Represents the exception for taking the median of an empty list
|
||||
class median_of_empty_list_exception:public std::exception{
|
||||
virtual const char* what() const throw() {
|
||||
return "Attempt to take the median of an empty list of numbers. "
|
||||
"The median of an empty list is undefined.";
|
||||
}
|
||||
};
|
||||
|
||||
///Return the median of a sequence of numbers defined by the random
|
||||
///access iterators begin and end. The sequence must not be empty
|
||||
///(median is undefined for an empty set).
|
||||
///
|
||||
///The numbers must be convertible to double.
|
||||
template<class RandAccessIter>
|
||||
double median(RandAccessIter begin, RandAccessIter end)
|
||||
throw(median_of_empty_list_exception){
|
||||
if(begin == end){ throw median_of_empty_list_exception(); }
|
||||
std::size_t size = end - begin;
|
||||
std::size_t middleIdx = size/2;
|
||||
RandAccessIter target = begin + middleIdx;
|
||||
std::nth_element(begin, target, end);
|
||||
|
||||
if(size % 2 != 0){ //Odd number of elements
|
||||
return *target;
|
||||
}else{ //Even number of elements
|
||||
double a = *target;
|
||||
RandAccessIter targetNeighbor= target-1;
|
||||
std::nth_element(begin, targetNeighbor, end);
|
||||
return (a+*targetNeighbor)/2.0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
struct TimeSpan
|
||||
{
|
||||
public:
|
||||
@ -66,27 +33,63 @@ struct SummaryCalcItem {
|
||||
code = 0;
|
||||
type = ST_CNT;
|
||||
color = Qt::black;
|
||||
wavg_sum = 0;
|
||||
avg_sum = 0;
|
||||
cnt = 0;
|
||||
divisor = 0;
|
||||
min = 0;
|
||||
max = 0;
|
||||
}
|
||||
SummaryCalcItem(const SummaryCalcItem & copy) {
|
||||
code = copy.code;
|
||||
type = copy.type;
|
||||
color = copy.color;
|
||||
|
||||
sum = copy.sum;
|
||||
divisor = copy.divisor;
|
||||
min = copy.min;
|
||||
max = copy.max;
|
||||
wavg_sum = 0;
|
||||
avg_sum = 0;
|
||||
cnt = 0;
|
||||
divisor = 0;
|
||||
min = 0;
|
||||
max = 0;
|
||||
}
|
||||
|
||||
SummaryCalcItem(ChannelID code, SummaryType type, QColor color)
|
||||
:code(code), type(type), color(color) {}
|
||||
:code(code), type(type), color(color) {
|
||||
}
|
||||
|
||||
inline void update(float value, float weight) {
|
||||
wavg_sum += value * weight;
|
||||
divisor += weight;
|
||||
avg_sum += value;
|
||||
cnt++;
|
||||
median_data.append(value);
|
||||
min = qMin(min, value);
|
||||
max = qMax(max, value);
|
||||
}
|
||||
|
||||
void reset(int reserve) {
|
||||
wavg_sum = 0;
|
||||
avg_sum = 0;
|
||||
divisor = 0;
|
||||
cnt = 0;
|
||||
min = 99999;
|
||||
max = -99999;
|
||||
median_data.clear();
|
||||
median_data.reserve(reserve);
|
||||
}
|
||||
ChannelID code;
|
||||
SummaryType type;
|
||||
QColor color;
|
||||
|
||||
double sum;
|
||||
double wavg_sum;
|
||||
double divisor;
|
||||
double avg_sum;
|
||||
int cnt;
|
||||
EventDataType min;
|
||||
EventDataType max;
|
||||
|
||||
QList<float> median_data;
|
||||
|
||||
};
|
||||
|
||||
struct SummaryChartSlice {
|
||||
@ -144,6 +147,11 @@ public:
|
||||
//! \brief Return any extra data to show beneath the date in the hover over tooltip
|
||||
virtual QString tooltipData(Day *, int);
|
||||
|
||||
virtual void dataChanged() {
|
||||
cache.clear();
|
||||
}
|
||||
|
||||
|
||||
void addCalc(ChannelID code, SummaryType type, QColor color) {
|
||||
calcitems.append(SummaryCalcItem(code, type, color));
|
||||
}
|
||||
@ -162,12 +170,14 @@ public:
|
||||
layer->m_empty = m_empty;
|
||||
layer->firstday = firstday;
|
||||
layer->lastday = lastday;
|
||||
layer->cache = cache;
|
||||
layer->calcitems = calcitems;
|
||||
// layer->calcitems = calcitems;
|
||||
layer->expected_slices = expected_slices;
|
||||
layer->nousedays = nousedays;
|
||||
layer->totaldays = totaldays;
|
||||
layer->peak_value = peak_value;
|
||||
layer->idx_start = idx_start;
|
||||
layer->idx_end = idx_end;
|
||||
layer->cache.clear();
|
||||
}
|
||||
|
||||
protected:
|
||||
@ -220,6 +230,8 @@ public:
|
||||
gSessionTimesChart()
|
||||
:gSummaryChart("SessionTimes", MT_CPAP) {
|
||||
addCalc(NoChannel, ST_SESSIONS, QColor(64,128,255));
|
||||
addCalc(NoChannel, ST_SESSIONS, QColor(64,128,255));
|
||||
addCalc(NoChannel, ST_SESSIONS, QColor(64,128,255));
|
||||
}
|
||||
virtual ~gSessionTimesChart() {}
|
||||
|
||||
@ -231,29 +243,10 @@ public:
|
||||
m_maxy = 28;
|
||||
}
|
||||
|
||||
virtual void preCalc() {
|
||||
num_slices = 0;
|
||||
num_days = 0;
|
||||
total_length = 0;
|
||||
session_data.clear();
|
||||
session_data.reserve(idx_end-idx_start);
|
||||
|
||||
}
|
||||
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;
|
||||
session_data.append(slice.height);
|
||||
}
|
||||
|
||||
num_days++;
|
||||
}
|
||||
virtual void preCalc();
|
||||
virtual void customCalc(Day *, QList<SummaryChartSlice> & slices);
|
||||
virtual void afterDraw(QPainter &, gGraph &, QRect);
|
||||
|
||||
|
||||
//! \brief Renders the graph to the QPainter object
|
||||
virtual void paint(QPainter &painter, gGraph &graph, const QRegion ®ion);
|
||||
virtual Layer * Clone() {
|
||||
@ -308,11 +301,6 @@ public:
|
||||
private:
|
||||
int incompdays;
|
||||
EventDataType compliance_threshold;
|
||||
double totalhours;
|
||||
int totaldays;
|
||||
QList<float> hour_data;
|
||||
|
||||
|
||||
};
|
||||
|
||||
class gAHIChart : public gSummaryChart
|
||||
@ -345,11 +333,20 @@ public:
|
||||
}
|
||||
|
||||
void CloneInto(gAHIChart * layer) {
|
||||
layer->ahi_total = ahi_total;
|
||||
layer->calc_cnt = calc_cnt;
|
||||
// layer->ahicalc = ahicalc;
|
||||
// layer->ahi_wavg = ahi_wavg;
|
||||
// layer->ahi_avg = ahi_avg;
|
||||
// layer->total_hours = total_hours;
|
||||
// layer->max_ahi = max_ahi;
|
||||
// layer->min_ahi = min_ahi;
|
||||
// layer->calc_cnt = calc_cnt;
|
||||
// layer->ahi_data = ahi_data;
|
||||
}
|
||||
|
||||
double ahi_total;
|
||||
// SummaryCalcItem ahicalc;
|
||||
double ahi_wavg;
|
||||
double ahi_avg;
|
||||
|
||||
double total_hours;
|
||||
float max_ahi;
|
||||
float min_ahi;
|
||||
|
@ -129,6 +129,8 @@ class Layer
|
||||
|
||||
inline bool recalculating() const { return m_recalculating; }
|
||||
|
||||
virtual void dataChanged() {}
|
||||
|
||||
/*! \brief Override this for the drawing code, using GLBuffer components for drawing
|
||||
\param gGraph & gv Graph Object that holds this layer
|
||||
\param int left
|
||||
|
@ -56,6 +56,39 @@ QString weightString(float kg, UnitSystem us = US_Undefined);
|
||||
//! \brief Mercilessly trash a directory
|
||||
bool removeDir(const QString &path);
|
||||
|
||||
///Represents the exception for taking the median of an empty list
|
||||
class median_of_empty_list_exception:public std::exception{
|
||||
virtual const char* what() const throw() {
|
||||
return "Attempt to take the median of an empty list of numbers. "
|
||||
"The median of an empty list is undefined.";
|
||||
}
|
||||
};
|
||||
|
||||
///Return the median of a sequence of numbers defined by the random
|
||||
///access iterators begin and end. The sequence must not be empty
|
||||
///(median is undefined for an empty set).
|
||||
///
|
||||
///The numbers must be convertible to double.
|
||||
template<class RandAccessIter>
|
||||
double median(RandAccessIter begin, RandAccessIter end)
|
||||
throw (median_of_empty_list_exception)
|
||||
{
|
||||
if (begin == end) { throw median_of_empty_list_exception(); }
|
||||
std::size_t size = end - begin;
|
||||
std::size_t middleIdx = size/2;
|
||||
RandAccessIter target = begin + middleIdx;
|
||||
std::nth_element(begin, target, end);
|
||||
|
||||
if (size % 2 != 0) { //Odd number of elements
|
||||
return *target;
|
||||
} else { //Even number of elements
|
||||
double a = *target;
|
||||
RandAccessIter targetNeighbor= target-1;
|
||||
std::nth_element(begin, targetNeighbor, end);
|
||||
return (a+*targetNeighbor)/2.0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
#ifndef nullptr
|
||||
#define nullptr NULL
|
||||
|
@ -34,7 +34,8 @@ Profile *p_profile;
|
||||
|
||||
Profile::Profile(QString path)
|
||||
: is_first_day(true),
|
||||
m_opened(false)
|
||||
m_opened(false),
|
||||
m_machopened(false)
|
||||
{
|
||||
p_name = STR_GEN_Profile;
|
||||
|
||||
@ -94,7 +95,7 @@ Profile::~Profile()
|
||||
bool Profile::Save(QString filename)
|
||||
{
|
||||
if (m_opened) {
|
||||
return Preferences::Save(filename);
|
||||
return Preferences::Save(filename) && p_profile->StoreMachines();
|
||||
} else return false;
|
||||
}
|
||||
|
||||
@ -132,14 +133,6 @@ bool Profile::Open(QString filename)
|
||||
}
|
||||
bool b = Preferences::Open(filename);
|
||||
|
||||
QString lockfile=p_path+"/lockfile";
|
||||
QFile file(lockfile);
|
||||
file.open(QFile::WriteOnly);
|
||||
QByteArray ba;
|
||||
ba.append(QHostInfo::localHostName());
|
||||
file.write(ba);
|
||||
file.close();
|
||||
|
||||
m_opened=true;
|
||||
doctor = new DoctorInfo(this);
|
||||
user = new UserInfo(this);
|
||||
@ -151,6 +144,192 @@ bool Profile::Open(QString filename)
|
||||
return b;
|
||||
}
|
||||
|
||||
const QString STR_PROP_Brand = "brand";
|
||||
const QString STR_PROP_Model = "model";
|
||||
const QString STR_PROP_Series = "series";
|
||||
const QString STR_PROP_ModelNumber = "modelnumber";
|
||||
const QString STR_PROP_SubModel = "submodel";
|
||||
const QString STR_PROP_Serial = "serial";
|
||||
const QString STR_PROP_DataVersion = "dataversion";
|
||||
const QString STR_PROP_LastImported = "lastimported";
|
||||
|
||||
bool Profile::OpenMachines()
|
||||
{
|
||||
if (m_machopened)
|
||||
return true;
|
||||
|
||||
if (!m_opened) {
|
||||
Open();
|
||||
}
|
||||
QFile lockfile(p_path+"lockfile");
|
||||
lockfile.open(QFile::WriteOnly);
|
||||
QByteArray ba;
|
||||
ba.append(QHostInfo::localHostName());
|
||||
lockfile.write(ba);
|
||||
lockfile.close();
|
||||
|
||||
QString filename = p_path+"machines.xml";
|
||||
QFile file(filename);
|
||||
if (!file.open(QFile::ReadOnly)) {
|
||||
qWarning() << "Could not open" << QDir::toNativeSeparators(filename);
|
||||
return false;
|
||||
}
|
||||
QDomDocument doc("machines.xml");
|
||||
|
||||
if (!doc.setContent(&file)) {
|
||||
qWarning() << "Invalid XML Content in" << QDir::toNativeSeparators(filename);
|
||||
return false;
|
||||
}
|
||||
file.close();
|
||||
QDomElement root = doc.firstChild().toElement();
|
||||
|
||||
if (root.tagName().toLower() != "machines") {
|
||||
//qDebug() << "No Machines Tag in Profiles.xml";
|
||||
return false;
|
||||
}
|
||||
|
||||
QDomElement elem = root.firstChildElement();
|
||||
|
||||
while (!elem.isNull()) {
|
||||
QString pKey = elem.tagName();
|
||||
|
||||
if (pKey.toLower() != "machine") {
|
||||
qWarning() << "Profile::ExtraLoad() pKey!=\"machine\"";
|
||||
elem = elem.nextSiblingElement();
|
||||
continue;
|
||||
}
|
||||
|
||||
int m_id;
|
||||
bool ok;
|
||||
m_id = elem.attribute("id", "").toInt(&ok);
|
||||
int mt;
|
||||
mt = elem.attribute("type", "").toInt(&ok);
|
||||
MachineType m_type = (MachineType)mt;
|
||||
|
||||
QString m_class = elem.attribute("class", "");
|
||||
|
||||
MachineInfo info;
|
||||
|
||||
info.type = m_type;
|
||||
info.loadername = m_class;
|
||||
|
||||
QHash<QString, QString> prop;
|
||||
|
||||
QDomElement e = elem.firstChildElement();
|
||||
|
||||
for (; !e.isNull(); e = e.nextSiblingElement()) {
|
||||
QString pKey = e.tagName();
|
||||
QString key = pKey.toLower();
|
||||
if (key == STR_PROP_Brand) {
|
||||
info.brand = e.text();
|
||||
} else if (key == STR_PROP_Model) {
|
||||
info.model = e.text();
|
||||
} else if (key == STR_PROP_ModelNumber) {
|
||||
info.modelnumber = e.text();
|
||||
} else if (key == STR_PROP_Serial) {
|
||||
info.serial = e.text();
|
||||
} else if (key == STR_PROP_Series) {
|
||||
info.series = e.text();
|
||||
} else if (key == STR_PROP_DataVersion) {
|
||||
info.version = e.text().toInt();
|
||||
} else if (key == STR_PROP_LastImported) {
|
||||
info.lastimported = QDateTime::fromString(e.text(), Qt::ISODate);
|
||||
} else if (key == "properties") {
|
||||
QDomElement pe = e.firstChildElement();
|
||||
for (; !pe.isNull(); pe = pe.nextSiblingElement()) {
|
||||
prop[pe.tagName()] = pe.text();
|
||||
}
|
||||
} else {
|
||||
// skip any old rubbish
|
||||
if ((key == "backuppath") || (key == "path") || (key == "submodel")) continue;
|
||||
|
||||
prop[pKey] = e.text();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Machine *m = nullptr;
|
||||
|
||||
m = MachineLoader::CreateMachine(info, m_id);
|
||||
//m->setId(m_id);
|
||||
if (m) m->properties = prop;
|
||||
|
||||
elem = elem.nextSiblingElement();
|
||||
}
|
||||
|
||||
m_machopened = true;
|
||||
|
||||
return true;
|
||||
|
||||
}
|
||||
|
||||
bool Profile::StoreMachines()
|
||||
{
|
||||
QDomDocument doc("Machines");
|
||||
QDomElement elem = ExtraSave(doc);
|
||||
doc.appendChild(elem);
|
||||
|
||||
QDomElement mach = doc.createElement("machines");
|
||||
|
||||
for (QHash<MachineID, Machine *>::iterator i = machlist.begin(); i != machlist.end(); i++) {
|
||||
QDomElement me = doc.createElement("machine");
|
||||
Machine *m = i.value();
|
||||
me.setAttribute("id", (int)m->id());
|
||||
me.setAttribute("type", (int)m->type());
|
||||
me.setAttribute("class", m->loaderName());
|
||||
|
||||
QDomElement pe = doc.createElement("properties");
|
||||
me.appendChild(pe);
|
||||
|
||||
for (QHash<QString, QString>::iterator j = i.value()->properties.begin(); j != i.value()->properties.end(); j++) {
|
||||
QDomElement pp = doc.createElement(j.key());
|
||||
pp.appendChild(doc.createTextNode(j.value()));
|
||||
pe.appendChild(pp);
|
||||
}
|
||||
|
||||
QDomElement mp = doc.createElement(STR_PROP_Brand);
|
||||
mp.appendChild(doc.createTextNode(m->brand()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_Model);
|
||||
mp.appendChild(doc.createTextNode(m->model()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_ModelNumber);
|
||||
mp.appendChild(doc.createTextNode(m->modelnumber()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_Serial);
|
||||
mp.appendChild(doc.createTextNode(m->serial()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_Series);
|
||||
mp.appendChild(doc.createTextNode(m->series()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_DataVersion);
|
||||
mp.appendChild(doc.createTextNode(QString::number(m->version())));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_LastImported);
|
||||
mp.appendChild(doc.createTextNode(m->lastImported().toString(Qt::ISODate)));
|
||||
me.appendChild(mp);
|
||||
|
||||
mach.appendChild(me);
|
||||
}
|
||||
|
||||
doc.appendChild(mach);
|
||||
|
||||
QString filename = p_path+"machines.xml";
|
||||
QFile file(filename);
|
||||
if (!file.open(QFile::WriteOnly)) {
|
||||
return false;
|
||||
}
|
||||
file.write(doc.toByteArray());
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
#if defined(Q_OS_WIN)
|
||||
class Environment
|
||||
{
|
||||
@ -387,6 +566,7 @@ void Profile::DataFormatError(Machine *m)
|
||||
}
|
||||
void Profile::LoadMachineData()
|
||||
{
|
||||
if (!m_machopened) OpenMachines();
|
||||
QHash<MachineID, QMap<QDate, QHash<ChannelID, EventDataType> > > cache;
|
||||
|
||||
for (QHash<MachineID, Machine *>::iterator i = machlist.begin(); i != machlist.end(); i++) {
|
||||
@ -411,96 +591,30 @@ void Profile::LoadMachineData()
|
||||
}
|
||||
}
|
||||
|
||||
const QString STR_PROP_Brand = "brand";
|
||||
const QString STR_PROP_Model = "model";
|
||||
const QString STR_PROP_Series = "series";
|
||||
const QString STR_PROP_ModelNumber = "modelnumber";
|
||||
const QString STR_PROP_SubModel = "submodel";
|
||||
const QString STR_PROP_Serial = "serial";
|
||||
const QString STR_PROP_DataVersion = "dataversion";
|
||||
const QString STR_PROP_LastImported = "lastimported";
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @brief Machine XML section in profile.
|
||||
* @brief Upgrade Machine XML section from old "profile.xml"
|
||||
* @param root
|
||||
*/
|
||||
void Profile::ExtraLoad(QDomElement &root)
|
||||
{
|
||||
if (root.tagName().toLower() != "machines") {
|
||||
qDebug() << "No Machines Tag in Profiles.xml";
|
||||
// Good!
|
||||
return;
|
||||
}
|
||||
|
||||
QDomElement elem = root.firstChildElement();
|
||||
// Save this sucker
|
||||
QDomDocument doc("Machines");
|
||||
|
||||
while (!elem.isNull()) {
|
||||
QString pKey = elem.tagName();
|
||||
doc.appendChild(root);
|
||||
|
||||
if (pKey.toLower() != "machine") {
|
||||
qWarning() << "Profile::ExtraLoad() pKey!=\"machine\"";
|
||||
elem = elem.nextSiblingElement();
|
||||
continue;
|
||||
}
|
||||
QFile file(p_path+"/machines.xml");
|
||||
|
||||
int m_id;
|
||||
bool ok;
|
||||
m_id = elem.attribute("id", "").toInt(&ok);
|
||||
int mt;
|
||||
mt = elem.attribute("type", "").toInt(&ok);
|
||||
MachineType m_type = (MachineType)mt;
|
||||
file.open(QFile::WriteOnly);
|
||||
|
||||
QString m_class = elem.attribute("class", "");
|
||||
file.write(doc.toByteArray());
|
||||
|
||||
MachineInfo info;
|
||||
|
||||
info.type = m_type;
|
||||
info.loadername = m_class;
|
||||
|
||||
QHash<QString, QString> prop;
|
||||
|
||||
QDomElement e = elem.firstChildElement();
|
||||
|
||||
for (; !e.isNull(); e = e.nextSiblingElement()) {
|
||||
QString pKey = e.tagName();
|
||||
QString key = pKey.toLower();
|
||||
if (key == STR_PROP_Brand) {
|
||||
info.brand = e.text();
|
||||
} else if (key == STR_PROP_Model) {
|
||||
info.model = e.text();
|
||||
} else if (key == STR_PROP_ModelNumber) {
|
||||
info.modelnumber = e.text();
|
||||
} else if (key == STR_PROP_Serial) {
|
||||
info.serial = e.text();
|
||||
} else if (key == STR_PROP_Series) {
|
||||
info.series = e.text();
|
||||
} else if (key == STR_PROP_DataVersion) {
|
||||
info.version = e.text().toInt();
|
||||
} else if (key == STR_PROP_LastImported) {
|
||||
info.lastimported = QDateTime::fromString(e.text(), Qt::ISODate);
|
||||
} else if (key == "properties") {
|
||||
QDomElement pe = e.firstChildElement();
|
||||
for (; !pe.isNull(); pe = pe.nextSiblingElement()) {
|
||||
prop[pe.tagName()] = pe.text();
|
||||
}
|
||||
} else {
|
||||
// skip any old rubbish
|
||||
if ((key == "backuppath") || (key == "path") || (key == "submodel")) continue;
|
||||
|
||||
prop[pKey] = e.text();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Machine *m = nullptr;
|
||||
|
||||
m = MachineLoader::CreateMachine(info, m_id);
|
||||
//m->setId(m_id);
|
||||
if (m) m->properties = prop;
|
||||
|
||||
elem = elem.nextSiblingElement();
|
||||
}
|
||||
file.close();
|
||||
}
|
||||
void Profile::AddMachine(Machine *m)
|
||||
{
|
||||
@ -523,63 +637,6 @@ void Profile::DelMachine(Machine *m)
|
||||
machlist.erase(machlist.find(m->id()));
|
||||
}
|
||||
|
||||
|
||||
// Potential Memory Leak Here..
|
||||
QDomElement Profile::ExtraSave(QDomDocument &doc)
|
||||
{
|
||||
QDomElement mach = doc.createElement("machines");
|
||||
|
||||
for (QHash<MachineID, Machine *>::iterator i = machlist.begin(); i != machlist.end(); i++) {
|
||||
QDomElement me = doc.createElement("machine");
|
||||
Machine *m = i.value();
|
||||
me.setAttribute("id", (int)m->id());
|
||||
me.setAttribute("type", (int)m->type());
|
||||
me.setAttribute("class", m->loaderName());
|
||||
|
||||
QDomElement pe = doc.createElement("properties");
|
||||
me.appendChild(pe);
|
||||
|
||||
for (QHash<QString, QString>::iterator j = i.value()->properties.begin(); j != i.value()->properties.end(); j++) {
|
||||
QDomElement pp = doc.createElement(j.key());
|
||||
pp.appendChild(doc.createTextNode(j.value()));
|
||||
pe.appendChild(pp);
|
||||
}
|
||||
|
||||
QDomElement mp = doc.createElement(STR_PROP_Brand);
|
||||
mp.appendChild(doc.createTextNode(m->brand()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_Model);
|
||||
mp.appendChild(doc.createTextNode(m->model()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_ModelNumber);
|
||||
mp.appendChild(doc.createTextNode(m->modelnumber()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_Serial);
|
||||
mp.appendChild(doc.createTextNode(m->serial()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_Series);
|
||||
mp.appendChild(doc.createTextNode(m->series()));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_DataVersion);
|
||||
mp.appendChild(doc.createTextNode(QString::number(m->version())));
|
||||
me.appendChild(mp);
|
||||
|
||||
mp = doc.createElement(STR_PROP_LastImported);
|
||||
mp.appendChild(doc.createTextNode(m->lastImported().toString(Qt::ISODate)));
|
||||
me.appendChild(mp);
|
||||
|
||||
mach.appendChild(me);
|
||||
}
|
||||
|
||||
return mach;
|
||||
}
|
||||
|
||||
|
||||
Day *Profile::addDay(QDate date)
|
||||
{
|
||||
QMap<QDate, Day *>::iterator dit = daylist.find(date);
|
||||
@ -870,6 +927,32 @@ Profile *Get()
|
||||
return profiles[getUserName()];;
|
||||
}
|
||||
|
||||
void saveProfileList()
|
||||
{
|
||||
QString filename = PREF.Get("{home}/profiles.xml");
|
||||
|
||||
QDomDocument doc("profiles");
|
||||
|
||||
QDomElement root = doc.createElement("profiles");
|
||||
doc.appendChild(root);
|
||||
|
||||
QMap<QString, Profile *>::iterator it;
|
||||
|
||||
for (it = profiles.begin(); it != profiles.end(); ++it) {
|
||||
QDomElement elem = doc.createElement("profile");
|
||||
elem.setAttribute("name", it.key());
|
||||
// Not technically nessesary..
|
||||
elem.setAttribute("path", QString("{home}/Profiles/%1/Profile.xml").arg(it.key()));
|
||||
root.appendChild(elem);
|
||||
}
|
||||
|
||||
QFile file(filename);
|
||||
file.open(QFile::WriteOnly);
|
||||
|
||||
file.write(doc.toByteArray());
|
||||
|
||||
file.close();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
@ -899,10 +982,13 @@ void Scan()
|
||||
QFileInfo fi = list.at(i);
|
||||
QString npath = fi.canonicalFilePath();
|
||||
Profile *prof = new Profile(npath);
|
||||
//prof->Open();
|
||||
|
||||
profiles[fi.fileName()] = prof;
|
||||
}
|
||||
|
||||
// Update profiles.xml for mobile version
|
||||
saveProfileList();
|
||||
}
|
||||
|
||||
|
||||
|
@ -50,6 +50,10 @@ class Profile : public Preferences
|
||||
//! \brief Open profile, parse profile.xml file, and initialize helper classes
|
||||
virtual bool Open(QString filename = "");
|
||||
|
||||
//! \brief Parse machines.xml
|
||||
bool OpenMachines();
|
||||
bool StoreMachines();
|
||||
|
||||
//! \brief Returns hostname that locked profile, or empty string if unlocked
|
||||
QString checkLock();
|
||||
|
||||
@ -71,6 +75,8 @@ class Profile : public Preferences
|
||||
//! \brief Barf because data format has changed. This does a purge of CPAP data for machine *m
|
||||
void DataFormatError(Machine *m);
|
||||
|
||||
QString path() { return p_path; }
|
||||
|
||||
/*! \brief Import Machine Data
|
||||
\param path containing import location
|
||||
*/
|
||||
@ -177,7 +183,6 @@ class Profile : public Preferences
|
||||
|
||||
// XML load components
|
||||
virtual void ExtraLoad(QDomElement &root);
|
||||
virtual QDomElement ExtraSave(QDomDocument &doc);
|
||||
|
||||
//! \brief Looks for the first date containing a day record matching machinetype
|
||||
QDate FirstDay(MachineType mt = MT_UNKNOWN);
|
||||
@ -218,6 +223,7 @@ class Profile : public Preferences
|
||||
QDate m_last;
|
||||
|
||||
bool m_opened;
|
||||
bool m_machopened;
|
||||
};
|
||||
|
||||
class MachineLoader;
|
||||
|
@ -542,6 +542,13 @@ void init()
|
||||
ZEO_TimeInLight = schema::channel["TimeInLight"].id();
|
||||
ZEO_TimeInDeep = schema::channel["TimeInDeep"].id();
|
||||
ZEO_TimeToZ = schema::channel["TimeToZ"].id();
|
||||
|
||||
schema::channel[CPAP_Leak].setShowInOverview(true);
|
||||
schema::channel[CPAP_RespRate].setShowInOverview(true);
|
||||
schema::channel[CPAP_MinuteVent].setShowInOverview(true);
|
||||
schema::channel[CPAP_TidalVolume].setShowInOverview(true);
|
||||
schema::channel[CPAP_CSR].setShowInOverview(true);
|
||||
schema::channel[CPAP_LargeLeak].setShowInOverview(true);
|
||||
}
|
||||
|
||||
|
||||
|
@ -453,6 +453,7 @@ void Daily::doToggleSession(Session * sess)
|
||||
sess->setEnabled(!sess->enabled());
|
||||
|
||||
LoadDate(previous_date);
|
||||
mainwin->getOverview()->graphView()->dataChanged();
|
||||
}
|
||||
|
||||
void Daily::Link_clicked(const QUrl &url)
|
||||
@ -903,10 +904,7 @@ QString Daily::getSessionInformation(Day * day)
|
||||
int s1=len % 60;
|
||||
|
||||
Session *sess=*s;
|
||||
if (!sess->settings.contains(SESSION_ENABLED)) {
|
||||
sess->settings[SESSION_ENABLED]=true;
|
||||
}
|
||||
bool b=sess->settings[SESSION_ENABLED].toBool();
|
||||
bool b=sess->enabled();
|
||||
|
||||
html+=QString("<tr class='datarow2'><td colspan=5 align=center>%2</td></tr>"
|
||||
"<tr class='datarow2'>"
|
||||
@ -969,8 +967,8 @@ QString Daily::getMachineSettings(Day * day) {
|
||||
ChannelID cpapmode = loader->CPAPModeChannel();
|
||||
schema::Channel & chan = schema::channel[cpapmode];
|
||||
first[cpapmode] = QString("<tr class='datarow'><td><a class='info' href='#'>%1<span>%2</span></a></td><td colspan=4>%3</td></tr>")
|
||||
.arg(schema::channel[cpapmode].label())
|
||||
.arg(schema::channel[cpapmode].description())
|
||||
.arg(chan.label())
|
||||
.arg(chan.description())
|
||||
.arg(day->getCPAPMode());
|
||||
|
||||
|
||||
@ -1248,15 +1246,15 @@ QString Daily::getStatisticsInfo(Day * day)
|
||||
html+="<tr><td colspan=3 align='left' bgcolor='white'>"+tr("Time outside of ramp")+
|
||||
QString("</td><td colspan=2 bgcolor='white'>%1:%2:%3</td></tr>").arg(q / 3600, 2, 10, QChar('0')).arg((q / 60) % 60, 2, 10, QChar('0')).arg(q % 60, 2, 10, QChar('0'));
|
||||
|
||||
EventDataType hc = day->count(CPAP_Hypopnea) - day->countInsideSpan(CPAP_Ramp, CPAP_Hypopnea);
|
||||
EventDataType oc = day->count(CPAP_Obstructive) - day->countInsideSpan(CPAP_Ramp, CPAP_Obstructive);
|
||||
// EventDataType hc = day->count(CPAP_Hypopnea) - day->countInsideSpan(CPAP_Ramp, CPAP_Hypopnea);
|
||||
// EventDataType oc = day->count(CPAP_Obstructive) - day->countInsideSpan(CPAP_Ramp, CPAP_Obstructive);
|
||||
|
||||
EventDataType tc = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive);
|
||||
EventDataType ahi = (hc+oc) / v;
|
||||
//EventDataType tc = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive);
|
||||
// EventDataType ahi = (hc+oc) / v;
|
||||
// Not sure if i was trying to be funny, and left on my replication of Devilbiss's bug here... :P
|
||||
|
||||
html+="<tr><td colspan=3 align='left' bgcolor='white'>"+tr("AHI excluding ramp")+
|
||||
QString("</td><td colspan=2 bgcolor='white'>%1</td></tr>").arg(ahi, 0, 'f', 2);
|
||||
// html+="<tr><td colspan=3 align='left' bgcolor='white'>"+tr("AHI excluding ramp")+
|
||||
// QString("</td><td colspan=2 bgcolor='white'>%1</td></tr>").arg(ahi, 0, 'f', 2);
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -1,12 +1,14 @@
|
||||
<html>
|
||||
<head><meta charset="UTF-8"></head>
|
||||
<body>
|
||||
<h1><image src="qrc:/docs/sheep.png" width=64 height=64>SleepyHead v0.9.7-11 <b>Testing</b></h1>
|
||||
<h1><image src="qrc:/docs/sheep.png" width=64 height=64>SleepyHead v0.9.8 <b>Testing</b></h1>
|
||||
|
||||
<p><h2><b>Release Notes</b></h2></p>
|
||||
<p>Greetings!</p>
|
||||
<p>Here is a definitely new and improved SleepyHead build :)</p>
|
||||
|
||||
<p>The New Features and Bugfixes list is getting a little scary, and I completely rewrote some important parts.. so let's have a Version Bump!</p>
|
||||
|
||||
<p>Right clicky menu has tons of new stuff for you to play with... some of this you're hopefully going to love! :)</p>
|
||||
<p>New Graph Clone ability allows you to make a temporary copy of a graph, and operate it completely independently...
|
||||
You can even take these graph clones with you to another day! They aren't saved though. they are gone when you close SleepyHead.</li>
|
||||
@ -18,6 +20,14 @@ I still need to to see more AirSense data to get everything right though!</p>
|
||||
<p><b>Sleep Well, and good luck!</b></p>
|
||||
<p><b><i>JediMark</i></b></p>
|
||||
|
||||
<br/>
|
||||
<b>New features & bug fixes in v0.9.8</b><br/>
|
||||
<list>
|
||||
<li>Removed old unused Mask Preferences and other junk</li>
|
||||
<li>Complete Overview SummaryChart overhaul</li>
|
||||
<li>Prescription changes now caches to disk to save having to reload all data every time</li>
|
||||
<li>Implemented Demand loading for SleepyHead Summary data files</li>
|
||||
</list>
|
||||
<br/>
|
||||
<b>New features & bug fixes in v0.9.7</b><br/>
|
||||
<list>
|
||||
|
@ -68,7 +68,7 @@ PreferencesDialog::PreferencesDialog(QWidget *parent, Profile *_profile) :
|
||||
shortformat.replace("yy", "yyyy");
|
||||
}
|
||||
|
||||
Qt::DayOfWeek dow = firstDayOfWeekFromLocale();
|
||||
// Qt::DayOfWeek dow = firstDayOfWeekFromLocale();
|
||||
|
||||
// QTextCharFormat format = ui->startedUsingMask->calendarWidget()->weekdayTextFormat(Qt::Saturday);
|
||||
// format.setForeground(QBrush(Qt::black, Qt::SolidPattern));
|
||||
|
@ -905,6 +905,20 @@ Defaults to 60 minutes.. Highly recommend it's left at this value.</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="2" column="1">
|
||||
<widget class="QSpinBox" name="prefCalcPercentile">
|
||||
<property name="toolTip">
|
||||
<string>For consistancy, ResMed users should use 95% here,
|
||||
as this is the only value available on summary-only days.</string>
|
||||
</property>
|
||||
<property name="suffix">
|
||||
<string notr="true">%</string>
|
||||
</property>
|
||||
<property name="minimum">
|
||||
<number>1</number>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="0">
|
||||
<widget class="QLabel" name="label_37">
|
||||
<property name="sizePolicy">
|
||||
@ -940,20 +954,6 @@ Defaults to 60 minutes.. Highly recommend it's left at this value.</string>
|
||||
</item>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="2" column="1">
|
||||
<widget class="QSpinBox" name="prefCalcPercentile">
|
||||
<property name="toolTip">
|
||||
<string>For consistancy, ResMed users should use 95% here,
|
||||
as this is the only value available on summary-only days.</string>
|
||||
</property>
|
||||
<property name="suffix">
|
||||
<string notr="true">%</string>
|
||||
</property>
|
||||
<property name="minimum">
|
||||
<number>1</number>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="3" column="1">
|
||||
<widget class="QComboBox" name="prefCalcMax">
|
||||
<property name="sizePolicy">
|
||||
@ -969,7 +969,7 @@ as this is the only value available on summary-only days.</string>
|
||||
</size>
|
||||
</property>
|
||||
<property name="toolTip">
|
||||
<string>ResMed users probably should use 99th Percentile for visual consistency.</string>
|
||||
<string><html><head/><body><p>True maximum is the maximum of the data set.</p><p>99th percentile filters out the rarest outliers.</p></body></html></string>
|
||||
</property>
|
||||
<item>
|
||||
<property name="text">
|
||||
@ -983,6 +983,37 @@ as this is the only value available on summary-only days.</string>
|
||||
</item>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="4" column="1">
|
||||
<widget class="QComboBox" name="culminativeIndices">
|
||||
<item>
|
||||
<property name="text">
|
||||
<string>Combined Count divided by Total Hours</string>
|
||||
</property>
|
||||
</item>
|
||||
<item>
|
||||
<property name="text">
|
||||
<string>Time Weighted average of Indice</string>
|
||||
</property>
|
||||
</item>
|
||||
<item>
|
||||
<property name="text">
|
||||
<string>Standard average of indice</string>
|
||||
</property>
|
||||
</item>
|
||||
<item>
|
||||
<property name="text">
|
||||
<string>Median</string>
|
||||
</property>
|
||||
</item>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="4" column="0">
|
||||
<widget class="QLabel" name="label_29">
|
||||
<property name="text">
|
||||
<string>Culminative Indices</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
|
@ -1 +1 @@
|
||||
11
|
||||
0
|
||||
|
@ -17,7 +17,7 @@
|
||||
|
||||
const int major_version = 0; // incompatible API changes
|
||||
const int minor_version = 9; // new features that don't break things
|
||||
const int revision_number = 7; // bugfixes, revisions
|
||||
const int revision_number = 8; // bugfixes, revisions
|
||||
|
||||
#ifdef TEST_BUILD
|
||||
const QString ReleaseStatus = "testing";
|
||||
|
Loading…
Reference in New Issue
Block a user