Refactor functions that act on all AHI-contributing channels.

Add new AllAhiChannels ChannelID to refer to all channels that contribute to AHI.
List of AHI-contributing channels is in "ahiChannels" QVector.
This commit is contained in:
Guy Scharf 2021-07-24 21:12:15 -07:00
parent 86d498f563
commit 9f5373ed91
21 changed files with 207 additions and 84 deletions

View File

@ -15,6 +15,8 @@
<br>Portions of OSCAR are © 2019-2021 by <br>Portions of OSCAR are © 2019-2021 by
<i>The OSCAR Team</i></p> <i>The OSCAR Team</i></p>
<ul> <ul>
<li>[new] Add support for ResMed AirSense 11 CPAP machines.</li>
<li>[new] Add support for Fisher & Paykel SleepStyle CPAP machines.</li>
<li>[new] Add support for DeVilbiss BLUE (DV6x) CPAP machines.</li> <li>[new] Add support for DeVilbiss BLUE (DV6x) CPAP machines.</li>
<li>[new] Additional Philips Respironics devices tested and fully supported: <li>[new] Additional Philips Respironics devices tested and fully supported:
<ul> <ul>
@ -51,6 +53,8 @@
<li>[fix] Correct Total Time and AHI in CSV Export when non-CPAP devices are used.</li> <li>[fix] Correct Total Time and AHI in CSV Export when non-CPAP devices are used.</li>
<li>[fix] Fix value display and bookmark behavior with clock drift.</li> <li>[fix] Fix value display and bookmark behavior with clock drift.</li>
<li>[fix] Ignore old sessions should not impact existing data.</li> <li>[fix] Ignore old sessions should not impact existing data.</li>
<li>[fix] Fix ocasional misordering of indexes on the Daily page.</li>
<li>[fix] Add Unclassified Apneas to the Statistics page.</li>
</ul> </ul>
<p> <p>
<b>Changes and fixes in OSCAR v1.2.0</b> <b>Changes and fixes in OSCAR v1.2.0</b>

View File

@ -1084,10 +1084,15 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion &region)
time /= 1000; time /= 1000;
QList<ChannelID> ahilist; QList<ChannelID> ahilist;
ahilist.push_back(CPAP_Hypopnea);
ahilist.push_back(CPAP_Obstructive); for (int i = 0; i < ahiChannels.size(); i++)
ahilist.push_back(CPAP_Apnea); ahilist.push_back(ahiChannels.at(i));
ahilist.push_back(CPAP_ClearAirway);
// ahilist.push_back(CPAP_Hypopnea);
// ahilist.push_back(CPAP_AllApnea);
// ahilist.push_back(CPAP_Obstructive);
// ahilist.push_back(CPAP_Apnea);
// ahilist.push_back(CPAP_ClearAirway);
QList<ChannelID> extras; QList<ChannelID> extras;
extras.push_back(CPAP_NRI); extras.push_back(CPAP_NRI);

View File

@ -13,6 +13,7 @@
#include "mainwindow.h" #include "mainwindow.h"
#include "SleepLib/profiles.h" #include "SleepLib/profiles.h"
#include "SleepLib/machine_common.h"
#include "gSessionTimesChart.h" #include "gSessionTimesChart.h"
#include "gYAxis.h" #include "gYAxis.h"
@ -1134,7 +1135,9 @@ void gTTIAChart::afterDraw(QPainter &, gGraph &graph, QRectF rect)
void gTTIAChart::populate(Day *day, int idx) void gTTIAChart::populate(Day *day, int idx)
{ {
QVector<SummaryChartSlice> & slices = cache[idx]; QVector<SummaryChartSlice> & slices = cache[idx];
float ttia = day->sum(CPAP_Obstructive) + day->sum(CPAP_ClearAirway) + day->sum(CPAP_Apnea) + day->sum(CPAP_Hypopnea); // float ttia = day->sum(CPAP_AllApnea) + day->sum(CPAP_Obstructive) + day->sum(CPAP_ClearAirway) + day->sum(CPAP_Apnea) + day->sum(CPAP_Hypopnea);
float ttia = day->sum(AllAhiChannels);
int h = ttia / 3600; int h = ttia / 3600;
int m = int(ttia) / 60 % 60; int m = int(ttia) / 60 % 60;
int s = int(ttia) % 60; int s = int(ttia) % 60;

View File

@ -377,10 +377,14 @@ class gAHIChart : public gSummaryChart
public: public:
gAHIChart() gAHIChart()
:gSummaryChart("AHIChart", MT_CPAP) { :gSummaryChart("AHIChart", MT_CPAP) {
addCalc(CPAP_ClearAirway, ST_CPH); for (int i = 0; i < ahiChannels.size(); i++)
addCalc(CPAP_Obstructive, ST_CPH); addCalc(ahiChannels.at(i), ST_CPH);
addCalc(CPAP_Apnea, ST_CPH);
addCalc(CPAP_Hypopnea, ST_CPH); // addCalc(CPAP_ClearAirway, ST_CPH);
// addCalc(CPAP_AllApnea, ST_CPH);
// addCalc(CPAP_Obstructive, ST_CPH);
// addCalc(CPAP_Apnea, ST_CPH);
// addCalc(CPAP_Hypopnea, ST_CPH);
if (p_profile->general->calculateRDI()) if (p_profile->general->calculateRDI())
addCalc(CPAP_RERA, ST_CPH); addCalc(CPAP_RERA, ST_CPH);
} }

View File

@ -22,10 +22,14 @@ void gDailySummary::SetDay(Day *day)
{ {
QList<ChannelID> piechans; QList<ChannelID> piechans;
piechans.append(CPAP_ClearAirway); for (int i = 0; i < ahiChannels.size(); i++)
piechans.append(CPAP_Obstructive); piechans.append(ahiChannels.at(i));
piechans.append(CPAP_Apnea);
piechans.append(CPAP_Hypopnea); // piechans.append(CPAP_ClearAirway);
// piechans.append(CPAP_AllApnea);
// piechans.append(CPAP_Obstructive);
// piechans.append(CPAP_Apnea);
// piechans.append(CPAP_Hypopnea);
piechans.append(CPAP_RERA); piechans.append(CPAP_RERA);
piechans.append(CPAP_FlowLimit); piechans.append(CPAP_FlowLimit);

View File

@ -82,17 +82,25 @@ bool SearchEvent(Session * session, ChannelID code, qint64 time, int dur, bool u
bool SearchApnea(Session *session, qint64 time, double dur) bool SearchApnea(Session *session, qint64 time, double dur)
{ {
if (SearchEvent(session, CPAP_Obstructive, time, dur))
return true;
if (SearchEvent(session, CPAP_Apnea, time, dur)) for (int i = 0; i < ahiChannels.size(); i++)
return true; if (SearchEvent(session, ahiChannels.at(i), time, dur))
return true;
if (SearchEvent(session, CPAP_ClearAirway, time, dur)) // if (SearchEvent(session, CPAP_AllApnea, time, dur))
return true; // return true;
//
if (SearchEvent(session, CPAP_Hypopnea, time, dur)) // if (SearchEvent(session, CPAP_Obstructive, time, dur))
return true; // return true;
//
// if (SearchEvent(session, CPAP_Apnea, time, dur))
// return true;
//
// if (SearchEvent(session, CPAP_ClearAirway, time, dur))
// return true;
//
// if (SearchEvent(session, CPAP_Hypopnea, time, dur))
// return true;
if (SearchEvent(session, CPAP_UserFlag1, time, dur, false)) if (SearchEvent(session, CPAP_UserFlag1, time, dur, false))
return true; return true;
@ -965,10 +973,11 @@ EventDataType calcAHI(Session *session, qint64 start, qint64 end)
if (start < 0) { if (start < 0) {
// much faster.. // much faster..
hours = session->hours(); hours = session->hours();
cnt = session->count(CPAP_Obstructive) cnt = session->count(AllAhiChannels);
+ session->count(CPAP_Hypopnea) // + session->count(CPAP_AllApnea)
+ session->count(CPAP_ClearAirway) // + session->count(CPAP_Hypopnea)
+ session->count(CPAP_Apnea); // + session->count(CPAP_ClearAirway)
// + session->count(CPAP_Apnea);
if (rdi) { if (rdi) {
cnt += session->count(CPAP_RERA); cnt += session->count(CPAP_RERA);
@ -980,10 +989,12 @@ EventDataType calcAHI(Session *session, qint64 start, qint64 end)
if (hours == 0) { return 0; } if (hours == 0) { return 0; }
cnt = session->rangeCount(CPAP_Obstructive, start, end) cnt = session->rangeCount(AllAhiChannels, start, end);
+ session->rangeCount(CPAP_Hypopnea, start, end) // cnt = session->rangeCount(CPAP_Obstructive, start, end)
+ session->rangeCount(CPAP_ClearAirway, start, end) // + session->rangeCount(CPAP_AllApnea, start, end)
+ session->rangeCount(CPAP_Apnea, start, end); // + session->rangeCount(CPAP_Hypopnea, start, end)
// + session->rangeCount(CPAP_ClearAirway, start, end)
// + session->rangeCount(CPAP_Apnea, start, end);
if (rdi) { if (rdi) {
cnt += session->rangeCount(CPAP_RERA, start, end); cnt += session->rangeCount(CPAP_RERA, start, end);
@ -1019,12 +1030,18 @@ int calcAHIGraph(Session *session)
session->destroyEvent(CPAP_RDI); session->destroyEvent(CPAP_RDI);
} }
if (!session->channelExists(CPAP_Obstructive) && bool gotsome = false;
!session->channelExists(CPAP_Hypopnea) && for (int i = 0; i < ahiChannels.size(); i++)
!session->channelExists(CPAP_Apnea) && gotsome = gotsome || session->channelExists(ahiChannels.at(i));
!session->channelExists(CPAP_ClearAirway) &&
!session->channelExists(CPAP_RERA) // if (!session->channelExists(CPAP_Obstructive) &&
) { return 0; } // !session->channelExists(CPAP_AllApnea) &&
// !session->channelExists(CPAP_Hypopnea) &&
// !session->channelExists(CPAP_Apnea) &&
// !session->channelExists(CPAP_ClearAirway) &&
// !session->channelExists(CPAP_RERA)
if (!gotsome)
return 0;
qint64 first = session->first(), qint64 first = session->first(),
last = session->last(), last = session->last(),
@ -1062,10 +1079,12 @@ int calcAHIGraph(Session *session)
break; break;
} }
events = session->rangeCount(CPAP_Obstructive, ti, t) events = session->rangeCount(AllAhiChannels, ti, t);
+ session->rangeCount(CPAP_Hypopnea, ti, t) // events = session->rangeCount(CPAP_Obstructive, ti, t)
+ session->rangeCount(CPAP_ClearAirway, ti, t) // + session->rangeCount(CPAP_Hypopnea, ti, t)
+ session->rangeCount(CPAP_Apnea, ti, t); // + session->rangeCount(CPAP_Hypopnea, ti, t)
// + session->rangeCount(CPAP_ClearAirway, ti, t)
// + session->rangeCount(CPAP_Apnea, ti, t);
ahi = events / hours; ahi = events / hours;
@ -1091,10 +1110,13 @@ int calcAHIGraph(Session *session)
f = ti - window_size_ms; f = ti - window_size_ms;
//hours=window_size; //double(ti-f)/3600000L; //hours=window_size; //double(ti-f)/3600000L;
events = session->rangeCount(CPAP_Obstructive, f, ti) // events = session->rangeCount(CPAP_Obstructive, f, ti)
+ session->rangeCount(CPAP_Hypopnea, f, ti) // + session->rangeCount(CPAP_AllApnea, f, ti)
+ session->rangeCount(CPAP_ClearAirway, f, ti) // + session->rangeCount(CPAP_Hypopnea, f, ti)
+ session->rangeCount(CPAP_Apnea, f, ti); // + session->rangeCount(CPAP_ClearAirway, f, ti)
// + session->rangeCount(CPAP_Apnea, f, ti);
events = session->rangeCount(AllAhiChannels, f, ti);
ahi = events / hours; ahi = events / hours;
avgahi += ahi; avgahi += ahi;

View File

@ -539,6 +539,7 @@ QString STR_TR_Humidifier;
QString STR_TR_H; // Short form of Hypopnea QString STR_TR_H; // Short form of Hypopnea
QString STR_TR_OA; // Short form of Obstructive Apnea QString STR_TR_OA; // Short form of Obstructive Apnea
QString STR_TR_A; // Short form of Apnea
QString STR_TR_UA; // Short form of Unspecified Apnea QString STR_TR_UA; // Short form of Unspecified Apnea
QString STR_TR_CA; // Short form of Clear Airway Apnea QString STR_TR_CA; // Short form of Clear Airway Apnea
QString STR_TR_FL; // Short form of Flow Limitation QString STR_TR_FL; // Short form of Flow Limitation
@ -746,7 +747,8 @@ void initializeStrings()
STR_TR_H = QObject::tr("H"); // Short form of Hypopnea STR_TR_H = QObject::tr("H"); // Short form of Hypopnea
STR_TR_OA = QObject::tr("OA"); // Short form of Obstructive Apnea STR_TR_OA = QObject::tr("OA"); // Short form of Obstructive Apnea
STR_TR_UA = QObject::tr("A"); // Short form of Unspecified Apnea STR_TR_A = QObject::tr("A"); // Short form of All Apnea
STR_TR_UA = QObject::tr("UA"); // Short form of Unspecified Apnea
STR_TR_CA = QObject::tr("CA"); // Short form of Clear Airway Apnea STR_TR_CA = QObject::tr("CA"); // Short form of Clear Airway Apnea
STR_TR_FL = QObject::tr("FL"); // Short form of Flow Limitation STR_TR_FL = QObject::tr("FL"); // Short form of Flow Limitation
STR_TR_SA = QObject::tr("SA"); // Short form of Flow Limitation STR_TR_SA = QObject::tr("SA"); // Short form of Flow Limitation

View File

@ -262,6 +262,7 @@ extern QString STR_TR_Humidifier;
extern QString STR_TR_H; // Short form of Hypopnea extern QString STR_TR_H; // Short form of Hypopnea
extern QString STR_TR_OA; // Short form of Obstructive Apnea extern QString STR_TR_OA; // Short form of Obstructive Apnea
extern QString STR_TR_A; // Short form of Apnea
extern QString STR_TR_UA; // Short form of Unspecified Apnea extern QString STR_TR_UA; // Short form of Unspecified Apnea
extern QString STR_TR_CA; // Short form of Clear Airway Apnea extern QString STR_TR_CA; // Short form of Clear Airway Apnea
extern QString STR_TR_FL; // Short form of Flow Limitation extern QString STR_TR_FL; // Short form of Flow Limitation

View File

@ -638,6 +638,12 @@ EventDataType Day::sum(ChannelID code)
// Cache this? // Cache this?
EventDataType val = 0; EventDataType val = 0;
if (code == AllAhiChannels) {
for (int i = 0; i < ahiChannels.size(); i++)
val += sum(ahiChannels.at(i));
return val;
}
for (auto & sess : sessions) { for (auto & sess : sessions) {
if (sess->enabled() && sess->m_sum.contains(code)) { if (sess->enabled() && sess->m_sum.contains(code)) {
val += sess->sum(code); val += sess->sum(code);
@ -1118,6 +1124,12 @@ EventDataType Day::count(ChannelID code)
{ {
EventDataType total = 0; EventDataType total = 0;
if (code == AllAhiChannels) {
for (int i = 0; i < ahiChannels.size(); i++)
total += count(ahiChannels.at(i));
return total;
}
for (auto & sess : sessions) { for (auto & sess : sessions) {
if (sess->enabled() && sess->m_cnt.contains(code)) { if (sess->enabled() && sess->m_cnt.contains(code)) {
total += sess->count(code); total += sess->count(code);

View File

@ -250,14 +250,14 @@ 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(AllAhiChannels);
EventDataType minutes = hours(MT_CPAP) * 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(AllAhiChannels) + count(CPAP_RERA);
EventDataType minutes = hours(MT_CPAP) * 60.0; EventDataType minutes = hours(MT_CPAP) * 60.0;
return (c * 60.0) / minutes; return (c * 60.0) / minutes;
} }
@ -280,13 +280,13 @@ class Day
//! \brief SleepyyHead Events Index, AHI combined with OSCAR detected events.. :) //! \brief SleepyyHead Events Index, AHI combined with OSCAR 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(AllAhiChannels) + count(CPAP_UserFlag1) + count(CPAP_UserFlag2);
EventDataType minutes = hours(MT_CPAP) * 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,
EventDataType calcTTIA() { EventDataType calcTTIA() {
EventDataType c = sum(CPAP_Hypopnea) + sum(CPAP_Obstructive) + sum(CPAP_Apnea) + sum(CPAP_ClearAirway); EventDataType c = sum(AllAhiChannels);
return c; return c;
} }
bool hasEvents(); bool hasEvents();

View File

@ -9,12 +9,14 @@
#include "machine_common.h" #include "machine_common.h"
ChannelID AllAhiChannels = 0xffff;
QVector<ChannelID> ahiChannels;
ChannelID NoChannel, SESSION_ENABLED, CPAP_SummaryOnly; ChannelID NoChannel, SESSION_ENABLED, CPAP_SummaryOnly;
ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, CPAP_Pressure, ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, CPAP_Pressure,
CPAP_PS, CPAP_Mode, CPAP_AHI, CPAP_PS, CPAP_Mode, CPAP_AHI,
CPAP_PressureMin, CPAP_PressureMax, CPAP_Ramp, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive, CPAP_PressureMin, CPAP_PressureMax, CPAP_Ramp, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive,
CPAP_Hypopnea, CPAP_Hypopnea, CPAP_AllApnea,
CPAP_ClearAirway, CPAP_Apnea, CPAP_PB, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore, CPAP_ClearAirway, CPAP_Apnea, CPAP_PB, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore,
CPAP_VSnore2, CPAP_VSnore2,
CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_SensAwake, CPAP_FlowRate, CPAP_MaskPressure, CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_SensAwake, CPAP_FlowRate, CPAP_MaskPressure,

View File

@ -141,13 +141,15 @@ struct MachineInfo {
enum MCDataType enum MCDataType
{ MC_bool = 0, MC_int, MC_long, MC_float, MC_double, MC_string, MC_datetime }; { MC_bool = 0, MC_int, MC_long, MC_float, MC_double, MC_string, MC_datetime };
extern ChannelID AllAhiChannels;
extern QVector<ChannelID> ahiChannels;
extern ChannelID NoChannel, SESSION_ENABLED, CPAP_SummaryOnly; extern ChannelID NoChannel, SESSION_ENABLED, CPAP_SummaryOnly;
extern ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, extern ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi,
CPAP_Pressure, CPAP_PS, CPAP_PSMin, CPAP_PSMax, CPAP_Pressure, CPAP_PS, CPAP_PSMin, CPAP_PSMax,
CPAP_Mode, CPAP_AHI, CPAP_Mode, CPAP_AHI,
CPAP_PressureMin, CPAP_PressureMax, CPAP_Ramp, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive, CPAP_PressureMin, CPAP_PressureMax, CPAP_Ramp, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive,
CPAP_Hypopnea, CPAP_Hypopnea, CPAP_AllApnea,
CPAP_ClearAirway, CPAP_Apnea, CPAP_PB, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore, CPAP_ClearAirway, CPAP_Apnea, CPAP_PB, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore,
CPAP_VSnore2, CPAP_VSnore2,
CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_SensAwake, CPAP_FlowRate, CPAP_MaskPressure, CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_SensAwake, CPAP_FlowRate, CPAP_MaskPressure,

View File

@ -211,10 +211,13 @@ QList<ChannelID> CPAPLoader::eventFlags(Day * day)
return list; return list;
} }
list.push_back(CPAP_ClearAirway); for (int i = 0; i < ahiChannels.size(); i++)
list.push_back(CPAP_Obstructive); list.push_back(ahiChannels.at(i));
list.push_back(CPAP_Hypopnea); // list.push_back(CPAP_ClearAirway);
list.push_back(CPAP_Apnea); // list.push_back(CPAP_AllApnea);
// list.push_back(CPAP_Obstructive);
// list.push_back(CPAP_Hypopnea);
// list.push_back(CPAP_Apnea);
return list; return list;
} }

View File

@ -65,6 +65,7 @@ void setOrders() {
schema::channel[CPAP_ClearAirway].setOrder(order++); schema::channel[CPAP_ClearAirway].setOrder(order++);
schema::channel[CPAP_NRI].setOrder(order++); schema::channel[CPAP_NRI].setOrder(order++);
schema::channel[CPAP_AllApnea].setOrder(order++);
schema::channel[CPAP_Obstructive].setOrder(order++); schema::channel[CPAP_Obstructive].setOrder(order++);
schema::channel[CPAP_Apnea].setOrder(order++); schema::channel[CPAP_Apnea].setOrder(order++);
schema::channel[CPAP_Hypopnea].setOrder(order++); schema::channel[CPAP_Hypopnea].setOrder(order++);
@ -160,6 +161,8 @@ void init()
QObject::tr("Hypopnea"), QObject::tr("A partially obstructed airway"), QObject::tr("H"), STR_UNIT_EventsPerHour, DEFAULT, QColor("blue"))); QObject::tr("Hypopnea"), QObject::tr("A partially obstructed airway"), QObject::tr("H"), STR_UNIT_EventsPerHour, DEFAULT, QColor("blue")));
schema::channel.add(GRP_CPAP, new Channel(CPAP_Apnea = 0x1004, FLAG, MT_CPAP, SESSION, "Apnea", schema::channel.add(GRP_CPAP, new Channel(CPAP_Apnea = 0x1004, FLAG, MT_CPAP, SESSION, "Apnea",
QObject::tr("Unclassified Apnea"), QObject::tr("An apnea that couldn't be determined as Central or Obstructive."),QObject::tr("UA"), STR_UNIT_EventsPerHour, DEFAULT, QColor("dark green"))); QObject::tr("Unclassified Apnea"), QObject::tr("An apnea that couldn't be determined as Central or Obstructive."),QObject::tr("UA"), STR_UNIT_EventsPerHour, DEFAULT, QColor("dark green")));
schema::channel.add(GRP_CPAP, new Channel(CPAP_AllApnea = 0x1010, FLAG, MT_CPAP, SESSION, "AllApnea",
QObject::tr("Apnea"), QObject::tr("An apnea reportred by your CPAP machine."),QObject::tr("A"), STR_UNIT_EventsPerHour, DEFAULT, QColor("#40c0ff")));
schema::channel.add(GRP_CPAP, new Channel(CPAP_FlowLimit = 0x1005, FLAG, MT_CPAP, SESSION, "FlowLimit", schema::channel.add(GRP_CPAP, new Channel(CPAP_FlowLimit = 0x1005, FLAG, MT_CPAP, SESSION, "FlowLimit",
QObject::tr("Flow Limitation"), QObject::tr("A restriction in breathing from normal, causing a flattening of the flow waveform."), QObject::tr("FL"), STR_UNIT_EventsPerHour, DEFAULT, QColor("#404040"))); QObject::tr("Flow Limitation"), QObject::tr("A restriction in breathing from normal, causing a flattening of the flow waveform."), 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",
@ -367,6 +370,18 @@ void init()
schema::channel[CPAP_PB].setShowInOverview(true); schema::channel[CPAP_PB].setShowInOverview(true);
schema::channel[CPAP_LargeLeak].setShowInOverview(true); schema::channel[CPAP_LargeLeak].setShowInOverview(true);
schema::channel[CPAP_FLG].setShowInOverview(true); schema::channel[CPAP_FLG].setShowInOverview(true);
// Identify the channels that contribute to AHI calculation
// When adding more AHI-contributing channels,
// 1) update this list
// 2) Update setOrders() above
// 3) Search source for CPAP_Obstructive to look for possible other places to add new channel
// 4) Search for AllAhiChannels to find all uses of the AHI-contributing channel list
ahiChannels.append(CPAP_ClearAirway);
ahiChannels.append(CPAP_AllApnea);
ahiChannels.append(CPAP_Obstructive);
ahiChannels.append(CPAP_Hypopnea);
ahiChannels.append(CPAP_Apnea);
} }

View File

@ -1623,6 +1623,13 @@ EventDataType Session::countInsideSpan(ChannelID span, ChannelID code)
EventDataType Session::rangeCount(ChannelID id, qint64 first, qint64 last) EventDataType Session::rangeCount(ChannelID id, qint64 first, qint64 last)
{ {
int total = 0, cnt;
if (id == AllAhiChannels) {
for (int i = 0; i < ahiChannels.size(); i++)
total += rangeCount(ahiChannels.at(i), first, last);
}
QHash<ChannelID, QVector<EventList *> >::iterator j = eventlist.find(id); QHash<ChannelID, QVector<EventList *> >::iterator j = eventlist.find(id);
if (j == eventlist.end()) { if (j == eventlist.end()) {
@ -1630,7 +1637,6 @@ EventDataType Session::rangeCount(ChannelID id, qint64 first, qint64 last)
} }
QVector<EventList *> &evec = j.value(); QVector<EventList *> &evec = j.value();
int total = 0, cnt;
qint64 t, start; qint64 t, start;
@ -1897,6 +1903,14 @@ EventDataType Session::rangeMax(ChannelID id, qint64 first, qint64 last)
EventDataType Session::count(ChannelID id) EventDataType Session::count(ChannelID id)
{ {
int sum = 0;
if (id == AllAhiChannels) {
for (int i = 0; i < ahiChannels.size(); i++)
sum += count(ahiChannels.at(i));
return sum;
}
QHash<ChannelID, EventDataType>::iterator i = m_cnt.find(id); QHash<ChannelID, EventDataType>::iterator i = m_cnt.find(id);
if (i != m_cnt.end()) { if (i != m_cnt.end()) {
@ -1912,7 +1926,6 @@ EventDataType Session::count(ChannelID id)
QVector<EventList *> &evec = j.value(); QVector<EventList *> &evec = j.value();
int sum = 0;
int evec_size=evec.size(); int evec_size=evec.size();
if (evec_size == 0) if (evec_size == 0)
return 0; return 0;

View File

@ -291,6 +291,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared)
evseg->AddSlice(CPAP_Apnea,QColor(0x20,0x80,0x20,0xff),STR_TR_UA); evseg->AddSlice(CPAP_Apnea,QColor(0x20,0x80,0x20,0xff),STR_TR_UA);
evseg->AddSlice(CPAP_Obstructive,QColor(0x40,0xaf,0xbf,0xff),STR_TR_OA); evseg->AddSlice(CPAP_Obstructive,QColor(0x40,0xaf,0xbf,0xff),STR_TR_OA);
evseg->AddSlice(CPAP_ClearAirway,QColor(0xb2,0x54,0xcd,0xff),STR_TR_CA); evseg->AddSlice(CPAP_ClearAirway,QColor(0xb2,0x54,0xcd,0xff),STR_TR_CA);
evseg->AddSlice(CPAP_AllApnea,QColor(0x40,0xaf,0xbf,0xff),STR_TR_A);
evseg->AddSlice(CPAP_RERA,QColor(0xff,0xff,0x80,0xff),STR_TR_RE); evseg->AddSlice(CPAP_RERA,QColor(0xff,0xff,0x80,0xff),STR_TR_RE);
evseg->AddSlice(CPAP_NRI,QColor(0x00,0x80,0x40,0xff),STR_TR_NR); evseg->AddSlice(CPAP_NRI,QColor(0x00,0x80,0x40,0xff),STR_TR_NR);
evseg->AddSlice(CPAP_FlowLimit,QColor(0x40,0x40,0x40,0xff),STR_TR_FL); evseg->AddSlice(CPAP_FlowLimit,QColor(0x40,0x40,0x40,0xff),STR_TR_FL);
@ -1395,7 +1396,7 @@ QString Daily::getStatisticsInfo(Day * day)
html+="<tr><td colspan=5>&nbsp;</td></tr>"; html+="<tr><td colspan=5>&nbsp;</td></tr>";
if ((cpap->loaderName() == STR_MACH_ResMed) || ((cpap->loaderName() == STR_MACH_PRS1) && (p_profile->cpap->resyncFromUserFlagging()))) { if ((cpap->loaderName() == STR_MACH_ResMed) || ((cpap->loaderName() == STR_MACH_PRS1) && (p_profile->cpap->resyncFromUserFlagging()))) {
int ttia = day->sum(CPAP_Obstructive) + day->sum(CPAP_ClearAirway) + day->sum(CPAP_Apnea) + day->sum(CPAP_Hypopnea); int ttia = day->sum(AllAhiChannels);
int h = ttia / 3600; int h = ttia / 3600;
int m = ttia / 60 % 60; int m = ttia / 60 % 60;
int s = ttia % 60; int s = ttia % 60;
@ -1500,15 +1501,24 @@ QString Daily::getPieChart (float values, Day * day) {
} else { } else {
html += "<tr><td align=center>"+tr("Unable to display Pie Chart on this system")+"</td></tr>\n"; html += "<tr><td align=center>"+tr("Unable to display Pie Chart on this system")+"</td></tr>\n";
} }
} else if ( day->channelHasData(CPAP_Obstructive) } else {
|| day->channelHasData(CPAP_Hypopnea) bool gotsome = false;
|| day->channelHasData(CPAP_ClearAirway) for (int i = 0; i < ahiChannels.size(); i++)
gotsome = gotsome || day->channelHasData(ahiChannels.at(i));
// if ( day->channelHasData(CPAP_Obstructive)
// || day->channelHasData(CPAP_AllApnea)
// || day->channelHasData(CPAP_Hypopnea)
// || day->channelHasData(CPAP_ClearAirway)
// || day->channelHasData(CPAP_Apnea)
if ( gotsome
|| day->channelHasData(CPAP_RERA) || day->channelHasData(CPAP_RERA)
|| day->channelHasData(CPAP_Apnea)
|| day->channelHasData(CPAP_FlowLimit) || day->channelHasData(CPAP_FlowLimit)
|| day->channelHasData(CPAP_SensAwake) || day->channelHasData(CPAP_SensAwake)
) { ) {
html += "<tr><td align=center><img src=\"qrc:/docs/0.0.gif\"></td></tr>\n"; html += "<tr><td align=center><img src=\"qrc:/docs/0.0.gif\"></td></tr>\n";
}
} }
html+="</table>\n"; html+="</table>\n";
html+="<hr/>\n"; html+="<hr/>\n";
@ -1650,7 +1660,12 @@ void Daily::Load(QDate date)
if (GraphView->isEmpty() && (hours>0)) { if (GraphView->isEmpty() && (hours>0)) {
// TODO: Eventually we should get isBrick from the loader, since some summary days // TODO: Eventually we should get isBrick from the loader, since some summary days
// on a non-brick might legitimately have no graph data. // on a non-brick might legitimately have no graph data.
if (!p_profile->hasChannel(CPAP_Obstructive) && !p_profile->hasChannel(CPAP_Hypopnea)) { bool gotsome = false;
for (int i = 0; i < ahiChannels.size(); i++)
gotsome = gotsome || p_profile->hasChannel(ahiChannels.at(i));
// if (!p_profile->hasChannel(CPAP_Obstructive) && !p_profile->hasChannel(CPAP_Hypopnea) && !p_profile->hasChannel(CPAP_AllApnea) && !p_profile->hasChannel(CPAP_ClearAirway)) {
if (!gotsome) {
GraphView->setEmptyText(STR_Empty_Brick); GraphView->setEmptyText(STR_Empty_Brick);
GraphView->setEmptyImage(QPixmap(":/icons/sadface.png")); GraphView->setEmptyImage(QPixmap(":/icons/sadface.png"));
@ -1664,7 +1679,7 @@ void Daily::Load(QDate date)
modestr=schema::channel[CPAP_Mode].m_options[mode]; modestr=schema::channel[CPAP_Mode].m_options[mode];
EventDataType ahi=(day->count(CPAP_Obstructive)+day->count(CPAP_Hypopnea)+day->count(CPAP_ClearAirway)+day->count(CPAP_Apnea)); EventDataType ahi=day->count(AllAhiChannels);
if (p_profile->general->calculateRDI()) ahi+=day->count(CPAP_RERA); if (p_profile->general->calculateRDI()) ahi+=day->count(CPAP_RERA);
ahi/=hours; ahi/=hours;
@ -1742,7 +1757,7 @@ void Daily::Load(QDate date)
htmlLeftIndices+="</table><hr/>"; htmlLeftIndices+="</table><hr/>";
htmlLeftPieChart = getPieChart((values[CPAP_Obstructive] + values[CPAP_Hypopnea] + htmlLeftPieChart = getPieChart((values[CPAP_Obstructive] + values[CPAP_Hypopnea] + values[CPAP_AllApnea] +
values[CPAP_ClearAirway] + values[CPAP_Apnea] + values[CPAP_RERA] + values[CPAP_ClearAirway] + values[CPAP_Apnea] + values[CPAP_RERA] +
values[CPAP_FlowLimit] + values[CPAP_SensAwake]), day); values[CPAP_FlowLimit] + values[CPAP_SensAwake]), day);

View File

@ -168,10 +168,14 @@ void ExportCSV::on_exportButton_clicked()
QList<ChannelID> countlist, avglist, p90list, maxlist; QList<ChannelID> countlist, avglist, p90list, maxlist;
countlist.append(CPAP_Hypopnea); for (int i = 0; i < ahiChannels.size(); i++)
countlist.append(CPAP_Obstructive); countlist.append(ahiChannels.at(i));
countlist.append(CPAP_Apnea);
countlist.append(CPAP_ClearAirway); // countlist.append(CPAP_Hypopnea);
// countlist.append(CPAP_Obstructive);
// countlist.append(CPAP_Apnea);
// countlist.append(CPAP_ClearAirway);
// countlist.append(CPAP_AllApnea);
countlist.append(CPAP_VSnore); countlist.append(CPAP_VSnore);
countlist.append(CPAP_VSnore2); countlist.append(CPAP_VSnore2);
countlist.append(CPAP_RERA); countlist.append(CPAP_RERA);
@ -298,8 +302,9 @@ void ExportCSV::on_exportButton_clicked()
int s = int(time) % 60; int s = int(time) % 60;
data += sep + QString().sprintf("%02i:%02i:%02i", h, m, s); data += sep + QString().sprintf("%02i:%02i:%02i", h, m, s);
float ahi = sess->count(CPAP_Obstructive) + sess->count(CPAP_Hypopnea) + sess->count( float ahi = sess->count(AllAhiChannels);
CPAP_Apnea) + sess->count(CPAP_ClearAirway); //sess->count(CPAP_AllApnea) + sess->count(CPAP_Obstructive) + sess->count(CPAP_Hypopnea)
// + sess->count(CPAP_Apnea) + sess->count(CPAP_ClearAirway);
ahi /= sess->hours(); ahi /= sess->hours();
data += sep + QString::number(ahi, 'f', 3); data += sep + QString::number(ahi, 'f', 3);

View File

@ -208,8 +208,7 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
cpapinfo += /*QObject::tr("Pressure Relief")+": "+ */day->getPressureRelief() + "\n"; cpapinfo += /*QObject::tr("Pressure Relief")+": "+ */day->getPressureRelief() + "\n";
} }
float ahi = (day->count(CPAP_Obstructive) + day->count(CPAP_Hypopnea) + float ahi = day->count(AllAhiChannels);
day->count(CPAP_ClearAirway) + day->count(CPAP_Apnea));
if (p_profile->general->calculateRDI()) { ahi += day->count(CPAP_RERA); } if (p_profile->general->calculateRDI()) { ahi += day->count(CPAP_RERA); }
@ -219,6 +218,7 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
//float pb = (100.0 / hours) * (day->sum(CPAP_PB) / 3600.0); //float pb = (100.0 / hours) * (day->sum(CPAP_PB) / 3600.0);
float uai = day->count(CPAP_Apnea) / hours; float uai = day->count(CPAP_Apnea) / hours;
float oai = day->count(CPAP_Obstructive) / hours; float oai = day->count(CPAP_Obstructive) / hours;
float ai = day->count(CPAP_AllApnea) / hours;
float hi = (day->count(CPAP_ExP) + day->count(CPAP_Hypopnea)) / hours; float hi = (day->count(CPAP_ExP) + day->count(CPAP_Hypopnea)) / hours;
float cai = day->count(CPAP_ClearAirway) / hours; float cai = day->count(CPAP_ClearAirway) / hours;
float rei = day->count(CPAP_RERA) / hours; float rei = day->count(CPAP_RERA) / hours;
@ -290,6 +290,8 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
} else if (cpap->loaderName() == STR_MACH_Intellipap) { } else if (cpap->loaderName() == STR_MACH_Intellipap) {
stats += QObject::tr("NRI=%1 LKI=%2 EPI=%3") stats += QObject::tr("NRI=%1 LKI=%2 EPI=%3")
.arg(nri, 0, 'f', 2).arg(lki, 0, 'f', 2).arg(exp, 0,'f', 2); .arg(nri, 0, 'f', 2).arg(lki, 0, 'f', 2).arg(exp, 0,'f', 2);
} else if (cpap->loaderName() == STR_MACH_SleepStyle) {
stats += QObject::tr("AI=%1 ").arg(ai, 0, 'f', 2);
} }
bounds = painter.boundingRect(QRectF(0, top + ttop, virt_width, 0), stats, bounds = painter.boundingRect(QRectF(0, top + ttop, virt_width, 0), stats,

View File

@ -240,7 +240,7 @@ void Statistics::updateRXChanges()
} }
// Update AHI/RDI/Time counts // Update AHI/RDI/Time counts
tmp = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Apnea) + day->count(CPAP_ClearAirway); tmp = day->count(AllAhiChannels);
rx.ahi += tmp; rx.ahi += tmp;
rx.rdi += tmp + day->count(CPAP_RERA); rx.rdi += tmp + day->count(CPAP_RERA);
rx.hours += day->hours(MT_CPAP); rx.hours += day->hours(MT_CPAP);
@ -264,7 +264,7 @@ void Statistics::updateRXChanges()
rx1.days = 1; rx1.days = 1;
// Only this days AHI/RDI counts // Only this days AHI/RDI counts
tmp = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Apnea) + day->count(CPAP_ClearAirway); tmp = day->count(AllAhiChannels);
rx1.ahi = tmp; rx1.ahi = tmp;
rx1.rdi = tmp + day->count(CPAP_RERA); rx1.rdi = tmp + day->count(CPAP_RERA);
@ -322,7 +322,7 @@ void Statistics::updateRXChanges()
Day * dy = rx.dates[di.key()] = p_profile->GetDay(di.key(), MT_CPAP); Day * dy = rx.dates[di.key()] = p_profile->GetDay(di.key(), MT_CPAP);
// Update AHI/RDI counts // Update AHI/RDI counts
tmp = dy->count(CPAP_Hypopnea) + dy->count(CPAP_Obstructive) + dy->count(CPAP_Apnea) + dy->count(CPAP_ClearAirway);; tmp = dy->count(AllAhiChannels);
rx.ahi += tmp; rx.ahi += tmp;
rx.rdi += tmp + dy->count(CPAP_RERA); rx.rdi += tmp + dy->count(CPAP_RERA);
@ -350,7 +350,7 @@ void Statistics::updateRXChanges()
Day * dy = rx2.dates[di.key()] = p_profile->GetDay(di.key(), MT_CPAP); Day * dy = rx2.dates[di.key()] = p_profile->GetDay(di.key(), MT_CPAP);
// Update AHI/RDI counts // Update AHI/RDI counts
tmp = dy->count(CPAP_Hypopnea) + dy->count(CPAP_Obstructive) + dy->count(CPAP_Apnea) + dy->count(CPAP_ClearAirway);; tmp = dy->count(AllAhiChannels);
rx2.ahi += tmp; rx2.ahi += tmp;
rx2.rdi += tmp + dy->count(CPAP_RERA); rx2.rdi += tmp + dy->count(CPAP_RERA);
@ -434,7 +434,7 @@ void Statistics::updateRXChanges()
if ((rx.relief == relief) && (rx.mode == mode) && (rx.pressure == pressure) && (rx.machine == mach) ) { if ((rx.relief == relief) && (rx.mode == mode) && (rx.pressure == pressure) && (rx.machine == mach) ) {
// Update AHI/RDI // Update AHI/RDI
tmp = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Apnea) + day->count(CPAP_ClearAirway); tmp = day->count(AllAhiChannels);
rx.ahi += tmp; rx.ahi += tmp;
rx.rdi += tmp + day->count(CPAP_RERA); rx.rdi += tmp + day->count(CPAP_RERA);
@ -466,7 +466,7 @@ void Statistics::updateRXChanges()
rx.days = 1; rx.days = 1;
// Set AHI/RDI for just this day // Set AHI/RDI for just this day
tmp = day->count(CPAP_Hypopnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Apnea) + day->count(CPAP_ClearAirway); tmp = day->count(AllAhiChannels);
rx.ahi = tmp; rx.ahi = tmp;
rx.rdi = tmp + day->count(CPAP_RERA); rx.rdi = tmp + day->count(CPAP_RERA);
@ -542,6 +542,7 @@ Statistics::Statistics(QObject *parent) :
rows.push_back(StatisticsRow(tr("Therapy Efficacy"), SC_SUBHEADING, MT_CPAP)); rows.push_back(StatisticsRow(tr("Therapy Efficacy"), SC_SUBHEADING, MT_CPAP));
rows.push_back(StatisticsRow("AHI", SC_AHI, MT_CPAP)); rows.push_back(StatisticsRow("AHI", SC_AHI, MT_CPAP));
rows.push_back(StatisticsRow("AllApnea", SC_CPH, MT_CPAP));
rows.push_back(StatisticsRow("Obstructive", SC_CPH, MT_CPAP)); rows.push_back(StatisticsRow("Obstructive", SC_CPH, MT_CPAP));
rows.push_back(StatisticsRow("Hypopnea", SC_CPH, MT_CPAP)); rows.push_back(StatisticsRow("Hypopnea", SC_CPH, MT_CPAP));
rows.push_back(StatisticsRow("Apnea", SC_CPH, MT_CPAP)); rows.push_back(StatisticsRow("Apnea", SC_CPH, MT_CPAP));
@ -745,10 +746,16 @@ QString Statistics::generateFooter(bool showinfo)
// Add RERA if calculating RDI instead of just AHI // Add RERA if calculating RDI instead of just AHI
EventDataType calcAHI(QDate start, QDate end) EventDataType calcAHI(QDate start, QDate end)
{ {
EventDataType val = (p_profile->calcCount(CPAP_Obstructive, MT_CPAP, start, end) EventDataType val = 0;
+ p_profile->calcCount(CPAP_Hypopnea, MT_CPAP, start, end)
+ p_profile->calcCount(CPAP_ClearAirway, MT_CPAP, start, end) for (int i = 0; i < ahiChannels.size(); i++)
+ p_profile->calcCount(CPAP_Apnea, MT_CPAP, start, end)); val += p_profile->calcCount(ahiChannels.at(i), MT_CPAP, start, end);
// (p_profile->calcCount(CPAP_Obstructive, MT_CPAP, start, end)
// + p_profile->calcCount(CPAP_AllApnea, MT_CPAP, start, end)
// + p_profile->calcCount(CPAP_Hypopnea, MT_CPAP, start, end)
// + p_profile->calcCount(CPAP_ClearAirway, MT_CPAP, start, end)
// + p_profile->calcCount(CPAP_Apnea, MT_CPAP, start, end));
if (p_profile->general->calculateRDI()) { if (p_profile->general->calculateRDI()) {
val += p_profile->calcCount(CPAP_RERA, MT_CPAP, start, end); val += p_profile->calcCount(CPAP_RERA, MT_CPAP, start, end);

View File

@ -148,6 +148,7 @@ static QString eventChannel(ChannelID i)
QString s; QString s;
do { do {
CHANNELNAME(CPAP_Obstructive); CHANNELNAME(CPAP_Obstructive);
CHANNELNAME(CPAP_AllApnea);
CHANNELNAME(CPAP_Hypopnea); CHANNELNAME(CPAP_Hypopnea);
CHANNELNAME(CPAP_PB); CHANNELNAME(CPAP_PB);
CHANNELNAME(CPAP_LeakTotal); CHANNELNAME(CPAP_LeakTotal);

View File

@ -202,7 +202,8 @@ QString Welcome::GenerateCPAPHTML()
QDate endtime = date.addDays(-1); QDate endtime = date.addDays(-1);
EventDataType ahi = (day->count(CPAP_Obstructive) + day->count(CPAP_Hypopnea) + day->count(CPAP_ClearAirway) + day->count(CPAP_Apnea)) / hours; // EventDataType ahi = (day->count(CPAP_AllApnea) + day->count(CPAP_Obstructive) + day->count(CPAP_Hypopnea) + day->count(CPAP_ClearAirway) + day->count(CPAP_Apnea)) / hours;
EventDataType ahi = day->count(AllAhiChannels) / hours;
EventDataType ahidays = calcAHI(starttime, endtime); EventDataType ahidays = calcAHI(starttime, endtime);
const QString under = tr("under"); const QString under = tr("under");