mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
Channel System Rework
This commit is contained in:
parent
9f459c97f0
commit
af617a15ca
@ -7,7 +7,7 @@
|
||||
#include "gFooBar.h"
|
||||
|
||||
gShadowArea::gShadowArea(QColor shadow_color)
|
||||
:Layer(EmptyChannel),m_shadow_color(shadow_color)
|
||||
:Layer(""),m_shadow_color(shadow_color)
|
||||
{
|
||||
QColor col=Qt::blue;
|
||||
addGLBuf(quads=new GLBuffer(shadow_color,20,GL_QUADS));
|
||||
@ -46,7 +46,7 @@ void gShadowArea::paint(gGraph & w,int left, int top, int width, int height)
|
||||
}
|
||||
|
||||
gFooBar::gFooBar(int offset,QColor handle_color,QColor line_color)
|
||||
:Layer(EmptyChannel),m_offset(offset),m_handle_color(handle_color),m_line_color(line_color)
|
||||
:Layer(""),m_offset(offset),m_handle_color(handle_color),m_line_color(line_color)
|
||||
{
|
||||
}
|
||||
gFooBar::~gFooBar()
|
||||
@ -66,7 +66,7 @@ void gFooBar::paint(gGraph & w,int left, int top, int width, int height)
|
||||
int start_px=left;
|
||||
//int end_px=left+width;
|
||||
|
||||
float h=top;
|
||||
//float h=top;
|
||||
|
||||
/* glLineWidth(1);
|
||||
glBegin(GL_LINES);
|
||||
|
@ -357,7 +357,7 @@ void Layer::drawGLBuf()
|
||||
|
||||
void Layer::SetDay(Day * d)
|
||||
{
|
||||
if (d && m_code!=EmptyChannel) {
|
||||
if (d && !m_code.isEmpty()) {
|
||||
m_day=d;
|
||||
m_minx=d->first(m_code);
|
||||
m_maxx=d->last(m_code);
|
||||
@ -382,7 +382,7 @@ void Layer::setLayout(LayerPosition position, short width, short height, short o
|
||||
}
|
||||
|
||||
LayerGroup::LayerGroup() :
|
||||
Layer(EmptyChannel)
|
||||
Layer("")
|
||||
{
|
||||
}
|
||||
LayerGroup::~LayerGroup()
|
||||
|
@ -36,6 +36,8 @@ void gLineChart::paint(gGraph & w,int left, int top, int width, int height)
|
||||
if (!m_day)
|
||||
return;
|
||||
|
||||
if (!m_day->channelExists(m_code)) return;
|
||||
|
||||
if (width<0)
|
||||
return;
|
||||
|
||||
|
@ -9,7 +9,7 @@
|
||||
|
||||
|
||||
gSegmentChart::gSegmentChart(GraphSegmentType type,QColor gradient_color,QColor outline_color)
|
||||
:Layer(EmptyChannel),m_graph_type(type),m_gradient_color(gradient_color),m_outline_color(outline_color)
|
||||
:Layer(""),m_graph_type(type),m_gradient_color(gradient_color),m_outline_color(outline_color)
|
||||
{
|
||||
// m_gradient_color=QColor(200,200,200);
|
||||
m_empty=true;
|
||||
|
@ -9,7 +9,7 @@
|
||||
#include "gSessionTime.h"
|
||||
|
||||
gTimeYAxis::gTimeYAxis(QColor col)
|
||||
:gYAxis(EmptyChannel,col)
|
||||
:gYAxis("",col)
|
||||
{
|
||||
}
|
||||
gTimeYAxis::~gTimeYAxis()
|
||||
|
@ -23,7 +23,7 @@ public:
|
||||
class gSessionTime:public Layer
|
||||
{
|
||||
public:
|
||||
gSessionTime(ChannelID=EmptyChannel,QColor col=QColor("blue"),Qt::Orientation o=Qt::Horizontal);
|
||||
gSessionTime(ChannelID="",QColor col=QColor("blue"),Qt::Orientation o=Qt::Horizontal);
|
||||
virtual ~gSessionTime();
|
||||
|
||||
virtual void paint(gGraph & w,int left,int top, int width, int height);
|
||||
|
@ -8,11 +8,11 @@
|
||||
#include <QLabel>
|
||||
#include <QDateTime>
|
||||
#include "gYAxis.h"
|
||||
#include "gBarChart.h"
|
||||
#include "gSummaryChart.h"
|
||||
|
||||
extern QLabel * qstatus2;
|
||||
SummaryChart::SummaryChart(Profile *p,QString label,GraphType type)
|
||||
:Layer(EmptyChannel),m_profile(p),m_label(label),m_graphtype(type)
|
||||
:Layer(""),m_profile(p),m_label(label),m_graphtype(type)
|
||||
{
|
||||
QColor color=Qt::black;
|
||||
addGLBuf(quads=new GLBuffer(color,20000,GL_QUADS));
|
||||
@ -311,7 +311,7 @@ void SummaryChart::paint(gGraph & w,int left, int top, int width, int height)
|
||||
int x,y;
|
||||
for (int j=0;j<m_codes.size();j++) {
|
||||
if (totalvalues[j]==0) continue;
|
||||
a=channel[m_codes[j]].label();
|
||||
a=schema::channel[m_codes[j]].label();
|
||||
a+=" ";
|
||||
switch(m_type[j]) {
|
||||
case ST_WAVG: a+="Avg"; break;
|
@ -17,7 +17,7 @@ const quint64 divisors[]={
|
||||
const int divcnt=sizeof(divisors)/sizeof(quint64);
|
||||
|
||||
gXAxis::gXAxis(QColor col,bool fadeout)
|
||||
:Layer(EmptyChannel)
|
||||
:Layer("")
|
||||
{
|
||||
m_line_color=col;
|
||||
m_text_color=col;
|
||||
|
@ -8,12 +8,13 @@
|
||||
#include <QDebug>
|
||||
#include "gYAxis.h"
|
||||
|
||||
gYSpacer::gYSpacer(int spacer) :Layer(EmptyChannel)
|
||||
gYSpacer::gYSpacer(int spacer)
|
||||
:Layer("")
|
||||
{
|
||||
};
|
||||
}
|
||||
|
||||
gXGrid::gXGrid(QColor col)
|
||||
:Layer(EmptyChannel)
|
||||
:Layer("")
|
||||
{
|
||||
m_major_color=QColor(180,180,180,128);
|
||||
m_minor_color=QColor(220,220,220,128);
|
||||
|
@ -39,7 +39,7 @@ protected:
|
||||
class gYAxis:public Layer
|
||||
{
|
||||
public:
|
||||
gYAxis(ChannelID code=EmptyChannel,QColor col=QColor("black"));
|
||||
gYAxis(ChannelID code="",QColor col=QColor("black"));
|
||||
virtual ~gYAxis();
|
||||
virtual void paint(gGraph & w,int left,int top, int width, int height);
|
||||
void SetShowMinorLines(bool b) { m_show_minor_lines=b; }
|
||||
|
@ -16,5 +16,7 @@
|
||||
<file>icons/oximeter.png</file>
|
||||
<file>docs/0.0.gif</file>
|
||||
<file>docs/template_overview.sht</file>
|
||||
<file>docs/schema.xml</file>
|
||||
<file>docs/channels.xml</file>
|
||||
</qresource>
|
||||
</RCC>
|
||||
|
@ -300,7 +300,7 @@ EventDataType Day::max(ChannelID code)
|
||||
EventDataType Day::cph(ChannelID code)
|
||||
{
|
||||
EventDataType sum=0;
|
||||
EventDataType h=0;
|
||||
//EventDataType h=0;
|
||||
for (int i=0;i<sessions.size();i++) {
|
||||
if (!sessions[i]->m_cph.contains(code)) continue;
|
||||
sum+=sessions[i]->cph(code)*sessions[i]->hours();
|
||||
@ -334,12 +334,18 @@ int Day::count(ChannelID code)
|
||||
}
|
||||
bool Day::channelExists(ChannelID id)
|
||||
{
|
||||
if (machine->hasChannel(id)) return true;
|
||||
/*for (int i=0;i<sessions.size();i++) {
|
||||
if (sessions[i]->channelExists(id))
|
||||
return true;
|
||||
} */
|
||||
return false;
|
||||
return machine->hasChannel(id);
|
||||
}
|
||||
bool Day::channelHasData(ChannelID id)
|
||||
{
|
||||
bool r=false;
|
||||
for (int i=0;i<sessions.size();i++) {
|
||||
if (sessions[i]->channelExists(id)) {
|
||||
r=true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return r;
|
||||
}
|
||||
|
||||
void Day::OpenEvents()
|
||||
|
@ -68,6 +68,7 @@ public:
|
||||
void OpenEvents();
|
||||
QVector<Session *> & getSessions() { return sessions; }
|
||||
bool channelExists(ChannelID id);
|
||||
bool channelHasData(ChannelID id);
|
||||
protected:
|
||||
QVector<Session *> sessions;
|
||||
qint64 d_first,d_last;
|
||||
|
@ -7,8 +7,8 @@
|
||||
#include <QDebug>
|
||||
#include "event.h"
|
||||
|
||||
EventList::EventList(ChannelID code,EventListType et,EventDataType gain, EventDataType offset, EventDataType min, EventDataType max,double rate)
|
||||
:m_code(code),m_type(et),m_gain(gain),m_offset(offset),m_min(min),m_max(max),m_rate(rate)
|
||||
EventList::EventList(EventListType et,EventDataType gain, EventDataType offset, EventDataType min, EventDataType max,double rate)
|
||||
:m_type(et),m_gain(gain),m_offset(offset),m_min(min),m_max(max),m_rate(rate)
|
||||
{
|
||||
m_first=m_last=0;
|
||||
m_count=0;
|
||||
|
@ -1,24 +1,23 @@
|
||||
/********************************************************************
|
||||
/*
|
||||
SleepLib Event Class Header
|
||||
Copyright (c)2011 Mark Watkins <jedimark@users.sourceforge.net>
|
||||
License: GPL
|
||||
*********************************************************************/
|
||||
*/
|
||||
|
||||
#ifndef EVENT_H
|
||||
#define EVENT_H
|
||||
|
||||
#include <QDateTime>
|
||||
#include "SleepLib/session.h"
|
||||
//#include "SleepLib/session.h"
|
||||
#include "machine_common.h"
|
||||
|
||||
|
||||
enum EventListType { EVL_Waveform, EVL_Event };
|
||||
|
||||
class EventList
|
||||
{
|
||||
friend class Session;
|
||||
public:
|
||||
EventList(ChannelID code,EventListType et,EventDataType gain=1.0, EventDataType offset=0.0, EventDataType min=0.0, EventDataType max=0.0, double rate=0.0);
|
||||
EventList(EventListType et,EventDataType gain=1.0, EventDataType offset=0.0, EventDataType min=0.0, EventDataType max=0.0, double rate=0.0);
|
||||
~EventList();
|
||||
|
||||
void AddEvent(qint64 time, EventStoreType data);
|
||||
@ -44,7 +43,7 @@ public:
|
||||
void setMin(EventDataType v) { m_min=v; }
|
||||
void setMax(EventDataType v) { m_max=v; }
|
||||
void setRate(EventDataType v) { m_rate=v; }
|
||||
void setCode(ChannelID id) { m_code=id; }
|
||||
//void setCode(ChannelID id) { m_code=id; }
|
||||
|
||||
inline const EventDataType & min() { return m_min; }
|
||||
inline const EventDataType & max() { return m_max; }
|
||||
@ -52,7 +51,7 @@ public:
|
||||
inline const EventDataType & offset() { return m_offset; }
|
||||
inline const EventDataType & rate() { return m_rate; }
|
||||
inline const EventListType & type() { return m_type; }
|
||||
inline const ChannelID & code() { return m_code; }
|
||||
//inline const ChannelID & code() { return m_code; }
|
||||
inline const bool & update_minmax() { return m_update_minmax; }
|
||||
|
||||
QString dimension() { return m_dimension; }
|
||||
@ -63,7 +62,7 @@ public:
|
||||
protected:
|
||||
QVector<quint32> m_time; // 32bitalize this.. add offsets to m_first
|
||||
QVector<EventStoreType> m_data;
|
||||
ChannelID m_code;
|
||||
//ChannelID m_code;
|
||||
EventListType m_type;
|
||||
int m_count;
|
||||
|
||||
|
@ -224,10 +224,9 @@ bool CMS50Loader::OpenSPORFile(QString path,Machine *mach,Profile *profile)
|
||||
|
||||
Session *sess=new Session(mach,sessid);
|
||||
sess->updateFirst(starttime);
|
||||
EventList *oxip=new EventList(OXI_Pulse,EVL_Event);
|
||||
EventList *oxis=new EventList(OXI_SPO2,EVL_Event);
|
||||
sess->eventlist[OXI_Pulse].push_back(oxip);
|
||||
sess->eventlist[OXI_SPO2].push_back(oxis);
|
||||
EventList *oxip=sess->AddEventList(OXI_Pulse,EVL_Event);
|
||||
EventList *oxis=sess->AddEventList(OXI_SPO2,EVL_Event);
|
||||
|
||||
oxip->AddEvent(starttime,last_pulse);
|
||||
oxis->AddEvent(starttime,last_spo2);
|
||||
|
||||
|
@ -15,6 +15,7 @@ License: GPL
|
||||
#include <QProgressBar>
|
||||
#include <QDebug>
|
||||
#include <cmath>
|
||||
#include "SleepLib/schema.h"
|
||||
#include "prs1_loader.h"
|
||||
#include "SleepLib/session.h"
|
||||
|
||||
@ -43,7 +44,6 @@ PRS1::PRS1(Profile *p,MachineID id):CPAP(p,id)
|
||||
properties["Brand"]="Philips Respironics";
|
||||
properties["Model"]="System One";
|
||||
|
||||
//SleepFlags= { CPAP_RERA, PRS1_VSnore2, CPAP_FlowLimit, CPAP_Hypopnea, CPAP_Obstructive, CPAP_ClearAirway, CPAP_CSR };
|
||||
}
|
||||
PRS1::~PRS1()
|
||||
{
|
||||
@ -313,20 +313,22 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile)
|
||||
continue;
|
||||
}
|
||||
|
||||
|
||||
if (sess->count(CPAP_IPAP)>0) {
|
||||
//sess->summaryCPAP_Mode]!=MODE_ASV)
|
||||
sess->settings[CPAP_Mode]=MODE_BIPAP;
|
||||
|
||||
if (sess->settings[PRS1_PressureReliefType].toInt()!=PR_NONE) {
|
||||
sess->settings[PRS1_PressureReliefType]=PR_BIFLEX;
|
||||
if (sess->settings[PRS1_FlexMode].toInt()!=PR_NONE) {
|
||||
sess->settings[PRS1_FlexMode]=PR_BIFLEX;
|
||||
}
|
||||
|
||||
|
||||
sess->setAvg(CPAP_Pressure,(sess->avg(CPAP_EPAP)+sess->avg(CPAP_IPAP))/2.0);
|
||||
sess->setWavg(CPAP_Pressure,(sess->wavg(CPAP_EPAP)+sess->wavg(CPAP_IPAP))/2.0);
|
||||
sess->setMin(CPAP_Pressure,sess->min(CPAP_EPAP));
|
||||
sess->setMax(CPAP_Pressure,sess->max(CPAP_IPAP));
|
||||
sess->set90p(CPAP_Pressure,sess->p90(CPAP_IPAP));
|
||||
sess->p90(CPAP_EPAP);
|
||||
sess->set90p(CPAP_Pressure,(sess->p90(CPAP_IPAP)+sess->p90(CPAP_EPAP))/2.0);
|
||||
//sess->p90(CPAP_EPAP);
|
||||
//sess->p90(CPAP_IPAP);
|
||||
} else {
|
||||
sess->avg(CPAP_Pressure);
|
||||
@ -336,20 +338,20 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile)
|
||||
sess->max(CPAP_Pressure);
|
||||
sess->cph(CPAP_Pressure);
|
||||
|
||||
if (!sess->settings.contains(PRS1_PressureMin)) {
|
||||
if (!sess->settings.contains(CPAP_PressureMin)) {
|
||||
sess->settings[CPAP_BrokenSummary]=true;
|
||||
//sess->set_last(sess->first());
|
||||
if (sess->min(CPAP_Pressure)==sess->max(CPAP_Pressure)) {
|
||||
sess->settings[CPAP_Mode]=MODE_CPAP; // no ramp
|
||||
sess->settings["PRS1Mode"]=MODE_CPAP; // no ramp
|
||||
} else {
|
||||
sess->settings[CPAP_Mode]=MODE_UNKNOWN;
|
||||
sess->settings["PRS1Mode"]=MODE_UNKNOWN;
|
||||
}
|
||||
sess->settings[PRS1_PressureReliefType]=PR_UNKNOWN;
|
||||
//sess->Set("FlexMode",PR_UNKNOWN);
|
||||
}
|
||||
|
||||
}
|
||||
if (sess->settings[CPAP_Mode]==MODE_CPAP) {
|
||||
sess->settings[PRS1_PressureMax]=sess->settings[PRS1_PressureMin];
|
||||
sess->settings[CPAP_PressureMax]=sess->settings[CPAP_PressureMin];
|
||||
}
|
||||
|
||||
//Printf(sess->start().Format()+wxT(" avgsummary=%.3f avgmine=%.3f\n"),sess->summary[CPAP_PressureAverage].GetDouble(),sess->weighted_avg_event_field(CPAP_Pressure,0));
|
||||
@ -433,16 +435,16 @@ bool PRS1Loader::OpenSummary(Session *session,QString filename)
|
||||
session->set_first(date);
|
||||
|
||||
double max;
|
||||
session->settings[PRS1_PressureMin]=(EventDataType)buffer[0x03]/10.0;
|
||||
session->settings[PRS1_PressureMax]=max=(EventDataType)buffer[0x04]/10.0;
|
||||
session->settings[CPAP_PressureMin]=(EventDataType)buffer[0x03]/10.0;
|
||||
session->settings[CPAP_PressureMax]=max=(EventDataType)buffer[0x04]/10.0;
|
||||
int offset=0;
|
||||
if (buffer[0x05]!=0) { // This is a time value for ASV stuff
|
||||
// non zero adds extra fields..
|
||||
offset=4;
|
||||
}
|
||||
|
||||
session->settings[PRS1_RampTime]=(int)buffer[offset+0x06]; // Minutes. Convert to seconds/hours here?
|
||||
session->settings[PRS1_RampPressure]=(EventDataType)buffer[offset+0x07]/10.0;
|
||||
session->settings[CPAP_RampTime]=(int)buffer[offset+0x06]; // Minutes. Convert to seconds/hours here?
|
||||
session->settings[CPAP_RampPressure]=(EventDataType)buffer[offset+0x07]/10.0;
|
||||
|
||||
if (max>0) { // Ignoring bipap until I see some more data.
|
||||
session->settings[CPAP_Mode]=(int)MODE_APAP;
|
||||
@ -451,21 +453,21 @@ bool PRS1Loader::OpenSummary(Session *session,QString filename)
|
||||
// This is incorrect..
|
||||
if (buffer[offset+0x08] & 0x80) { // Flex Setting
|
||||
if (buffer[offset+0x08] & 0x08) {
|
||||
if (max>0) session->settings[PRS1_PressureReliefType]=(int)PR_AFLEX;
|
||||
else session->settings[PRS1_PressureReliefType]=(int)PR_CFLEXPLUS;
|
||||
} else session->settings[PRS1_PressureReliefType]=(int)PR_CFLEX;
|
||||
} else session->settings[PRS1_PressureReliefType]=(int)PR_NONE;
|
||||
if (max>0) session->settings[PRS1_FlexMode]=(int)PR_AFLEX;
|
||||
else session->settings[PRS1_FlexMode]=(int)PR_CFLEXPLUS;
|
||||
} else session->settings[PRS1_FlexMode]=(int)PR_CFLEX;
|
||||
} else session->settings[PRS1_FlexMode]=(int)PR_NONE;
|
||||
|
||||
session->settings[PRS1_PressureReliefSetting]=(int)buffer[offset+0x08] & 3;
|
||||
session->settings[PRS1_HumidifierSetting]=(int)buffer[offset+0x09]&0x0f;
|
||||
session->settings[PRS1_HumidifierStatus]=(buffer[offset+0x09]&0x80)==0x80;
|
||||
session->settings[PRS1_SystemLockStatus]=(buffer[offset+0x0a]&0x80)==0x80;
|
||||
session->settings[PRS1_SystemOneResistanceStatus]=(buffer[offset+0x0a]&0x40)==0x40;
|
||||
session->settings[PRS1_SystemOneResistanceSetting]=(int)buffer[offset+0x0a]&7;
|
||||
session->settings[PRS1_HoseDiameter]=(int)((buffer[offset+0x0a]&0x08)?15:22);
|
||||
session->settings[PRS1_AutoOff]=(buffer[offset+0x0c]&0x10)==0x10;
|
||||
session->settings[PRS1_MaskAlert]=(buffer[offset+0x0c]&0x08)==0x08;
|
||||
session->settings[PRS1_ShowAHI]=(buffer[offset+0x0c]&0x04)==0x04;
|
||||
session->settings["FlexSet"]=(int)buffer[offset+0x08] & 3;
|
||||
session->settings["HumidSet"]=(int)buffer[offset+0x09]&0x0f;
|
||||
session->settings["HumidStat"]=(buffer[offset+0x09]&0x80)==0x80;
|
||||
session->settings["SysLock"]=(buffer[offset+0x0a]&0x80)==0x80;
|
||||
session->settings["SysOneResistStat"]=(buffer[offset+0x0a]&0x40)==0x40;
|
||||
session->settings["SysOneResistSet"]=(int)buffer[offset+0x0a]&7;
|
||||
session->settings["HoseDiam"]=((buffer[offset+0x0a]&0x08)?"15mm":"22mm");
|
||||
session->settings["AutoOff"]=(buffer[offset+0x0c]&0x10)==0x10;
|
||||
session->settings["MaskAlert"]=(buffer[offset+0x0c]&0x08)==0x08;
|
||||
session->settings["ShowAHI"]=(buffer[offset+0x0c]&0x04)==0x04;
|
||||
|
||||
unsigned duration=buffer[offset+0x14] | (buffer[0x15] << 8);
|
||||
//session->settings[CPAP_Duration]=(int)duration;
|
||||
@ -476,13 +478,13 @@ bool PRS1Loader::OpenSummary(Session *session,QString filename)
|
||||
return false;
|
||||
|
||||
session->set_last(date+qint64(duration)*1000L);
|
||||
session->settings[PRS1_PressureMinAchieved]=buffer[offset+0x16]/10.0;
|
||||
session->settings[PRS1_PressureMaxAchieved]=buffer[offset+0x17]/10.0;
|
||||
session->settings[PRS1_PressureAvg]=buffer[offset+0x18]/10.0;
|
||||
session->settings[PRS1_Pressure90]=buffer[offset+0x19]/10.0;
|
||||
//session->settings[PRS1_PressureMinAchieved]=buffer[offset+0x16]/10.0;
|
||||
//session->settings[PRS1_PressureMaxAchieved]=buffer[offset+0x17]/10.0;
|
||||
//session->settings[PRS1_PressureAvg]=buffer[offset+0x18]/10.0;
|
||||
//session->settings[PRS1_Pressure90]=buffer[offset+0x19]/10.0;
|
||||
|
||||
if (max==0) {
|
||||
session->settings[PRS1_PressureAvg]=session->settings[PRS1_PressureMin];
|
||||
// session->settings[PRS1_PressureAvg]=session->settings[PRS1_PressureMin];
|
||||
}
|
||||
|
||||
// Not using these because sometimes this summary is broken.
|
||||
@ -500,13 +502,14 @@ bool PRS1Loader::OpenSummary(Session *session,QString filename)
|
||||
// v2 event parser.
|
||||
bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64 timestamp)
|
||||
{
|
||||
ChannelID Codes[]={
|
||||
/*ChannelID Codes[]={
|
||||
PRS1_Unknown00, PRS1_Unknown01, CPAP_Pressure, CPAP_EPAP, CPAP_PressurePulse, CPAP_RERA, CPAP_Obstructive, CPAP_ClearAirway,
|
||||
PRS1_Unknown08, PRS1_Unknown09, CPAP_Hypopnea, PRS1_Unknown0B, CPAP_FlowLimit, CPAP_VSnore, PRS1_Unknown0E, CPAP_CSR, PRS1_Unknown10,
|
||||
CPAP_Leak, PRS1_Unknown12
|
||||
};
|
||||
int ncodes=sizeof(Codes)/sizeof(ChannelID);
|
||||
int ncodes=sizeof(Codes)/sizeof(ChannelID); */
|
||||
|
||||
//QHash<ChannelID,EventList *> Code;
|
||||
EventList * Code[0x20]={NULL};
|
||||
|
||||
EventDataType data[10];
|
||||
@ -520,7 +523,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
short delta;
|
||||
while (pos<size) {
|
||||
unsigned char code=buffer[pos++];
|
||||
if (code>=ncodes) {
|
||||
if (code>0x12) {
|
||||
qDebug() << "Illegal PRS1 code " << hex << int(code) << " appeared at " << hex << pos+16;
|
||||
return false;
|
||||
}
|
||||
@ -534,42 +537,31 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
|
||||
cnt++;
|
||||
switch (code) {
|
||||
|
||||
case 0x00: // Unknown 00
|
||||
if (!Code[0]) {
|
||||
Code[0]=new EventList(PRS1_Unknown00,EVL_Event);
|
||||
session->eventlist[PRS1_Unknown00].push_back(Code[0]);
|
||||
session->machine()->registerChannel(PRS1_Unknown00);
|
||||
if (!(Code[0]=session->AddEventList(PRS1_00,EVL_Event))) return false;
|
||||
}
|
||||
Code[0]->AddEvent(t,buffer[pos++]);
|
||||
break;
|
||||
case 0x01: // Unknown
|
||||
if (!Code[1]) {
|
||||
Code[1]=new EventList(PRS1_Unknown01,EVL_Event);
|
||||
session->eventlist[PRS1_Unknown01].push_back(Code[1]);
|
||||
session->machine()->registerChannel(PRS1_Unknown01);
|
||||
if (!(Code[1]=session->AddEventList(PRS1_01,EVL_Event))) return false;
|
||||
}
|
||||
Code[1]->AddEvent(t,0);
|
||||
break;
|
||||
case 0x02: // Pressure
|
||||
if (!Code[2]) {
|
||||
Code[2]=new EventList(CPAP_Pressure,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_Pressure].push_back(Code[2]);
|
||||
session->machine()->registerChannel(CPAP_Pressure);
|
||||
Code[2]=session->AddEventList(CPAP_Pressure,EVL_Event,0.1);
|
||||
if (!Code[2]) return false;
|
||||
}
|
||||
Code[2]->AddEvent(t,buffer[pos++]);
|
||||
break;
|
||||
case 0x03: // BIPAP Pressure
|
||||
if (!Code[3]) {
|
||||
Code[3]=new EventList(CPAP_EPAP,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_EPAP].push_back(Code[3]);
|
||||
Code[4]=new EventList(CPAP_IPAP,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_IPAP].push_back(Code[4]);
|
||||
Code[5]=new EventList(CPAP_PressureSupport,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_PressureSupport].push_back(Code[5]);
|
||||
session->machine()->registerChannel(CPAP_EPAP);
|
||||
session->machine()->registerChannel(CPAP_IPAP);
|
||||
session->machine()->registerChannel(CPAP_PressureSupport);
|
||||
|
||||
if (!(Code[3]=session->AddEventList(CPAP_EPAP,EVL_Event,0.1))) return false;
|
||||
if (!(Code[4]=session->AddEventList(CPAP_IPAP,EVL_Event,0.1))) return false;
|
||||
if (!(Code[5]=session->AddEventList(CPAP_PS,EVL_Event,0.1))) return false;
|
||||
}
|
||||
Code[3]->AddEvent(t,data[0]=buffer[pos++]);
|
||||
Code[4]->AddEvent(t,data[1]=buffer[pos++]);
|
||||
@ -577,9 +569,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
break;
|
||||
case 0x04: // Pressure Pulse
|
||||
if (!Code[6]) {
|
||||
Code[6]=new EventList(CPAP_PressurePulse,EVL_Event);
|
||||
session->eventlist[CPAP_PressurePulse].push_back(Code[6]);
|
||||
session->machine()->registerChannel(CPAP_PressurePulse);
|
||||
if (!(Code[6]=session->AddEventList(CPAP_PressurePulse,EVL_Event))) return false;
|
||||
}
|
||||
Code[6]->AddEvent(t,buffer[pos++]);
|
||||
//qDebug() << hex << data[0];
|
||||
@ -588,9 +578,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[0]=buffer[pos++];
|
||||
tt=t-(qint64(data[0])*1000L);
|
||||
if (!Code[7]) {
|
||||
Code[7]=new EventList(CPAP_RERA,EVL_Event);
|
||||
session->eventlist[CPAP_RERA].push_back(Code[7]);
|
||||
session->machine()->registerChannel(CPAP_RERA);
|
||||
if (!(Code[7]=session->AddEventList(CPAP_RERA,EVL_Event))) return false;
|
||||
}
|
||||
Code[7]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -599,9 +587,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[0]=buffer[pos++];
|
||||
tt=t-(qint64(data[0])*1000L);
|
||||
if (!Code[8]) {
|
||||
Code[8]=new EventList(CPAP_Obstructive,EVL_Event);
|
||||
session->eventlist[CPAP_Obstructive].push_back(Code[8]);
|
||||
session->machine()->registerChannel(CPAP_Obstructive);
|
||||
if (!(Code[8]=session->AddEventList(CPAP_Obstructive,EVL_Event))) return false;
|
||||
}
|
||||
Code[8]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -609,9 +595,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[0]=buffer[pos++];
|
||||
tt=t-(qint64(data[0])*1000L);
|
||||
if (!Code[9]) {
|
||||
Code[9]=new EventList(CPAP_ClearAirway,EVL_Event);
|
||||
session->eventlist[CPAP_ClearAirway].push_back(Code[9]);
|
||||
session->machine()->registerChannel(CPAP_ClearAirway);
|
||||
if (!(Code[9]=session->AddEventList(CPAP_ClearAirway,EVL_Event))) return false;
|
||||
}
|
||||
Code[9]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -619,9 +603,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[0]=buffer[pos++];
|
||||
tt=t-(qint64(data[0])*1000L);
|
||||
if (!Code[10]) {
|
||||
Code[10]=new EventList(CPAP_Hypopnea,EVL_Event);
|
||||
session->eventlist[CPAP_Hypopnea].push_back(Code[10]);
|
||||
session->machine()->registerChannel(CPAP_Hypopnea);
|
||||
if (!(Code[10]=session->AddEventList(CPAP_Hypopnea,EVL_Event))) return false;
|
||||
}
|
||||
Code[10]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -629,9 +611,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[0]=buffer[pos++];
|
||||
tt=t-(qint64(data[0])*1000L);
|
||||
if (!Code[11]) {
|
||||
Code[11]=new EventList(CPAP_FlowLimit,EVL_Event);
|
||||
session->eventlist[CPAP_FlowLimit].push_back(Code[11]);
|
||||
session->machine()->registerChannel(CPAP_FlowLimit);
|
||||
if (!(Code[11]=session->AddEventList(CPAP_FlowLimit,EVL_Event))) return false;
|
||||
}
|
||||
Code[11]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -640,18 +620,14 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[0]=buffer[pos++];
|
||||
data[1]=buffer[pos++];
|
||||
if (!Code[12]) {
|
||||
Code[12]=new EventList(PRS1_Unknown0B,EVL_Event);
|
||||
session->eventlist[PRS1_Unknown0B].push_back(Code[12]);
|
||||
session->machine()->registerChannel(PRS1_Unknown0B);
|
||||
if (!(Code[12]=session->AddEventList(PRS1_0B,EVL_Event))) return false;
|
||||
}
|
||||
// FIXME
|
||||
Code[12]->AddEvent(t,data[0]);
|
||||
break;
|
||||
case 0x0d: // Vibratory Snore
|
||||
if (!Code[13]) {
|
||||
Code[13]=new EventList(CPAP_VSnore,EVL_Event);
|
||||
session->eventlist[CPAP_VSnore].push_back(Code[13]);
|
||||
session->machine()->registerChannel(CPAP_VSnore);
|
||||
if (!(Code[13]=session->AddEventList(CPAP_VSnore,EVL_Event))) return false;
|
||||
}
|
||||
Code[13]->AddEvent(t,0);
|
||||
break;
|
||||
@ -659,20 +635,14 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[0]=buffer[pos++];
|
||||
data[1]=buffer[pos++];
|
||||
if (!Code[14]) {
|
||||
Code[14]=new EventList(CPAP_Leak,EVL_Event);
|
||||
session->eventlist[CPAP_Leak].push_back(Code[14]);
|
||||
Code[15]=new EventList(CPAP_Snore,EVL_Event);
|
||||
session->eventlist[CPAP_Snore].push_back(Code[15]);
|
||||
session->machine()->registerChannel(CPAP_Leak);
|
||||
session->machine()->registerChannel(CPAP_Snore);
|
||||
if (!(Code[14]=session->AddEventList(CPAP_Leak,EVL_Event))) return false;
|
||||
if (!(Code[15]=session->AddEventList(CPAP_Snore,EVL_Event))) return false;
|
||||
}
|
||||
Code[14]->AddEvent(t,data[0]);
|
||||
Code[15]->AddEvent(t,data[1]);
|
||||
if (data[1]>0) {
|
||||
if (!Code[16]) {
|
||||
Code[16]=new EventList(PRS1_VSnore2,EVL_Event);
|
||||
session->eventlist[PRS1_VSnore2].push_back(Code[16]);
|
||||
session->machine()->registerChannel(PRS1_VSnore2);
|
||||
if (!(Code[16]=session->AddEventList(CPAP_VSnore2,EVL_Event))) return false;
|
||||
}
|
||||
Code[16]->AddEvent(t,data[1]);
|
||||
}
|
||||
@ -685,9 +655,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[2]=buffer[pos++];
|
||||
|
||||
if (!Code[17]) {
|
||||
Code[17]=new EventList(PRS1_Unknown0E,EVL_Event);
|
||||
session->eventlist[PRS1_Unknown0E].push_back(Code[17]);
|
||||
session->machine()->registerChannel(PRS1_Unknown0E);
|
||||
if (!(Code[17]=session->AddEventList(PRS1_0E,EVL_Event))) return false;
|
||||
}
|
||||
Code[17]->AddEvent(t,data[0]);
|
||||
//qDebug() << hex << data[0] << data[1] << data[2];
|
||||
@ -700,9 +668,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[1]=buffer[pos++];
|
||||
data[2]=buffer[pos++];
|
||||
if (!Code[20]) {
|
||||
Code[20]=new EventList(PRS1_Unknown10,EVL_Event);
|
||||
session->eventlist[PRS1_Unknown10].push_back(Code[20]);
|
||||
session->machine()->registerChannel(PRS1_Unknown10);
|
||||
if (!(Code[20]=session->AddEventList(PRS1_10,EVL_Event))) return false;
|
||||
}
|
||||
Code[20]->AddEvent(t,data[0]);
|
||||
break;
|
||||
@ -712,9 +678,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[1]=buffer[pos++];
|
||||
tt=t-qint64(data[1])*1000L;
|
||||
if (!Code[23]) {
|
||||
Code[23]=new EventList(CPAP_CSR,EVL_Event);
|
||||
session->eventlist[CPAP_CSR].push_back(Code[23]);
|
||||
session->machine()->registerChannel(CPAP_CSR);
|
||||
if (!(Code[23]=session->AddEventList(CPAP_CSR,EVL_Event))) return false;
|
||||
}
|
||||
Code[23]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -724,9 +688,7 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
data[2]=buffer[pos+1]<<8 | buffer[pos];
|
||||
pos+=2;
|
||||
if (!Code[24]) {
|
||||
Code[24]=new EventList(PRS1_Unknown12,EVL_Event);
|
||||
session->eventlist[PRS1_Unknown12].push_back(Code[24]);
|
||||
session->machine()->registerChannel(PRS1_Unknown12);
|
||||
if (!(Code[24]=session->AddEventList(PRS1_12,EVL_Event))) return false;
|
||||
}
|
||||
Code[24]->AddEvent(t,data[0]);
|
||||
break;
|
||||
@ -743,14 +705,14 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64
|
||||
|
||||
bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qint64 timestamp)
|
||||
{
|
||||
ChannelID Codes[]={
|
||||
PRS1_Unknown00, PRS1_Unknown01, CPAP_Pressure, CPAP_EPAP, CPAP_PressurePulse, CPAP_Obstructive,
|
||||
CPAP_ClearAirway, CPAP_Hypopnea, PRS1_Unknown08, CPAP_FlowLimit, PRS1_Unknown0A, CPAP_CSR,
|
||||
PRS1_Unknown0C, CPAP_VSnore, PRS1_Unknown0E, PRS1_Unknown0F, PRS1_Unknown10,
|
||||
CPAP_Leak, PRS1_Unknown12
|
||||
QString Codes[]={
|
||||
PRS1_00, PRS1_01, CPAP_Pressure, CPAP_EPAP, CPAP_PressurePulse, CPAP_Obstructive,
|
||||
CPAP_ClearAirway, CPAP_Hypopnea, PRS1_08, CPAP_FlowLimit, PRS1_0A, CPAP_CSR,
|
||||
PRS1_0C, CPAP_VSnore, PRS1_0E, PRS1_0F, PRS1_10,
|
||||
CPAP_Leak, PRS1_12
|
||||
};
|
||||
|
||||
int ncodes=sizeof(Codes)/sizeof(ChannelID);
|
||||
int ncodes=sizeof(Codes)/sizeof(QString);
|
||||
|
||||
EventList * Code[0x20]={NULL};
|
||||
//for (int i=0;i<0x20;i++) Code[i]=NULL;
|
||||
@ -780,7 +742,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
pos+=2;
|
||||
t+=qint64(delta)*1000L;
|
||||
}
|
||||
ChannelID cpapcode=Codes[(int)code];
|
||||
QString cpapcode=Codes[(int)code];
|
||||
//EventDataType PS;
|
||||
tt=t;
|
||||
cnt++;
|
||||
@ -802,9 +764,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
break;
|
||||
case 0x01: // Unknown
|
||||
if (!Code[1]) {
|
||||
Code[1]=new EventList(cpapcode,EVL_Event,0.1);
|
||||
session->eventlist[cpapcode].push_back(Code[1]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[1]=session->AddEventList(cpapcode,EVL_Event,0.1))) return false;
|
||||
}
|
||||
Code[1]->AddEvent(t,0);
|
||||
break;
|
||||
@ -812,18 +772,14 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
case 0x02: // Pressure
|
||||
data[0]=buffer[pos++];
|
||||
if (!Code[2]) {
|
||||
Code[2]=new EventList(cpapcode,EVL_Event,0.1);
|
||||
session->eventlist[cpapcode].push_back(Code[2]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[2]=session->AddEventList(cpapcode,EVL_Event,0.1))) return false;
|
||||
}
|
||||
Code[2]->AddEvent(t,data[0]);
|
||||
break;
|
||||
case 0x04: // Pressure Pulse
|
||||
data[0]=buffer[pos++];
|
||||
if (!Code[3]) {
|
||||
Code[3]=new EventList(cpapcode,EVL_Event);
|
||||
session->eventlist[cpapcode].push_back(Code[3]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[3]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[3]->AddEvent(t,data[0]);
|
||||
break;
|
||||
@ -832,9 +788,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
data[0]=buffer[pos++];
|
||||
tt-=qint64(data[0])*1000L; // Subtract Time Offset
|
||||
if (!Code[4]) {
|
||||
Code[4]=new EventList(cpapcode,EVL_Event);
|
||||
session->eventlist[cpapcode].push_back(Code[4]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[4]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[4]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -843,9 +797,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
data[0]=buffer[pos++];
|
||||
tt-=qint64(data[0])*1000L; // Subtract Time Offset
|
||||
if (!Code[5]) {
|
||||
Code[5]=new EventList(cpapcode,EVL_Event);
|
||||
session->eventlist[cpapcode].push_back(Code[5]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[5]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[5]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -853,9 +805,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
data[0]=buffer[pos++];
|
||||
tt-=qint64(data[0])*1000L; // Subtract Time Offset
|
||||
if (!Code[6]) {
|
||||
Code[6]=new EventList(cpapcode,EVL_Event);
|
||||
session->eventlist[cpapcode].push_back(Code[6]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[6]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[6]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -863,9 +813,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
data[0]=buffer[pos++];
|
||||
tt-=qint64(data[0])*1000L; // Subtract Time Offset
|
||||
if (!Code[10]) {
|
||||
Code[10]=new EventList(cpapcode,EVL_Event);
|
||||
session->eventlist[cpapcode].push_back(Code[10]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[10]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[10]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -873,9 +821,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
data[0]=buffer[pos++];
|
||||
tt-=qint64(data[0])*1000L; // Subtract Time Offset
|
||||
if (!Code[11]) {
|
||||
Code[11]=new EventList(cpapcode,EVL_Event);
|
||||
session->eventlist[cpapcode].push_back(Code[11]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[11]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[11]->AddEvent(tt,data[0]);
|
||||
|
||||
@ -885,9 +831,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
data[0]=buffer[pos++];
|
||||
tt-=qint64(data[0])*1000L; // Subtract Time Offset
|
||||
if (!Code[7]) {
|
||||
Code[7]=new EventList(cpapcode,EVL_Event);
|
||||
session->eventlist[cpapcode].push_back(Code[7]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[7]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[7]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -896,9 +840,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
data[0]=buffer[pos++];
|
||||
tt-=qint64(data[0])*1000L; // Subtract Time Offset
|
||||
if (!Code[8]) {
|
||||
Code[8]=new EventList(cpapcode,EVL_Event);
|
||||
session->eventlist[cpapcode].push_back(Code[8]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[8]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[8]->AddEvent(tt,data[0]);
|
||||
break;
|
||||
@ -913,59 +855,25 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
//tt-=delta;
|
||||
tt-=qint64(data[1])*1000L;
|
||||
if (!Code[9]) {
|
||||
Code[9]=new EventList(cpapcode,EVL_Event,2.0);
|
||||
session->eventlist[cpapcode].push_back(Code[9]);
|
||||
session->machine()->registerChannel(cpapcode);
|
||||
if (!(Code[9]=session->AddEventList(cpapcode,EVL_Event))) return false;
|
||||
}
|
||||
Code[9]->AddEvent(tt,data[0]);
|
||||
//session->AddEvent(new Event(tt,cpapcode, data[0], data, 2));
|
||||
break;
|
||||
case 0x0d: // All the other ASV graph stuff.
|
||||
if (!Code[12]) {
|
||||
Code[12]=new EventList(CPAP_IPAP,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_IPAP].push_back(Code[12]);
|
||||
session->machine()->registerChannel(CPAP_IPAP);
|
||||
|
||||
Code[13]=new EventList(CPAP_IPAP_Low,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_IPAP_Low].push_back(Code[13]);
|
||||
session->machine()->registerChannel(CPAP_IPAP_Low);
|
||||
|
||||
Code[14]=new EventList(CPAP_IPAP_High,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_IPAP_High].push_back(Code[14]);
|
||||
session->machine()->registerChannel(CPAP_IPAP_High);
|
||||
|
||||
Code[15]=new EventList(CPAP_Leak,EVL_Event);
|
||||
session->eventlist[CPAP_Leak].push_back(Code[15]);
|
||||
session->machine()->registerChannel(CPAP_Leak);
|
||||
|
||||
Code[16]=new EventList(CPAP_RespiratoryRate,EVL_Event);
|
||||
session->eventlist[CPAP_RespiratoryRate].push_back(Code[16]);
|
||||
session->machine()->registerChannel(CPAP_RespiratoryRate);
|
||||
|
||||
Code[17]=new EventList(CPAP_PatientTriggeredBreaths,EVL_Event);
|
||||
session->eventlist[CPAP_PatientTriggeredBreaths].push_back(Code[17]);
|
||||
session->machine()->registerChannel(CPAP_PatientTriggeredBreaths);
|
||||
|
||||
Code[18]=new EventList(CPAP_MinuteVentilation,EVL_Event);
|
||||
session->eventlist[CPAP_MinuteVentilation].push_back(Code[18]);
|
||||
session->machine()->registerChannel(CPAP_MinuteVentilation);
|
||||
|
||||
Code[19]=new EventList(CPAP_TidalVolume,EVL_Event,10.0);
|
||||
session->eventlist[CPAP_TidalVolume].push_back(Code[19]);
|
||||
session->machine()->registerChannel(CPAP_TidalVolume);
|
||||
|
||||
Code[20]=new EventList(CPAP_Snore,EVL_Event);
|
||||
session->eventlist[CPAP_Snore].push_back(Code[20]);
|
||||
session->machine()->registerChannel(CPAP_Snore);
|
||||
|
||||
Code[22]=new EventList(CPAP_EPAP,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_EPAP].push_back(Code[22]);
|
||||
session->machine()->registerChannel(CPAP_EPAP);
|
||||
|
||||
Code[23]=new EventList(CPAP_PressureSupport,EVL_Event,0.1);
|
||||
session->eventlist[CPAP_PressureSupport].push_back(Code[23]);
|
||||
session->machine()->registerChannel(CPAP_PressureSupport);
|
||||
if (!(Code[12]=session->AddEventList("IPAP",EVL_Event))) return false;
|
||||
if (!(Code[13]=session->AddEventList("IPAPLo",EVL_Event))) return false;
|
||||
if (!(Code[14]=session->AddEventList("IPAPHi",EVL_Event))) return false;
|
||||
if (!(Code[15]=session->AddEventList("Leak",EVL_Event))) return false;
|
||||
if (!(Code[16]=session->AddEventList("RespRate",EVL_Event))) return false;
|
||||
if (!(Code[17]=session->AddEventList("PTB",EVL_Event))) return false;
|
||||
|
||||
if (!(Code[18]=session->AddEventList("MinuteVent",EVL_Event))) return false;
|
||||
if (!(Code[19]=session->AddEventList("TidalVolume",EVL_Event))) return false;
|
||||
if (!(Code[20]=session->AddEventList("Snore",EVL_Event))) return false;
|
||||
if (!(Code[22]=session->AddEventList("EPAP",EVL_Event))) return false;
|
||||
if (!(Code[23]=session->AddEventList("PS",EVL_Event))) return false;
|
||||
}
|
||||
Code[12]->AddEvent(t,data[0]=buffer[pos++]); // IAP
|
||||
Code[13]->AddEvent(t,buffer[pos++]); // IAP Low
|
||||
@ -978,9 +886,7 @@ bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qin
|
||||
Code[20]->AddEvent(t,data[2]=buffer[pos++]); // Snore
|
||||
if (data[2]>0) {
|
||||
if (!Code[21]) {
|
||||
Code[21]=new EventList(CPAP_VSnore,EVL_Event);
|
||||
session->eventlist[CPAP_VSnore].push_back(Code[21]);
|
||||
session->machine()->registerChannel(CPAP_VSnore);
|
||||
if (!(Code[21]=session->AddEventList("VSnore",EVL_Event))) return false;
|
||||
}
|
||||
Code[21]->AddEvent(t,0); //data[2]); // VSnore
|
||||
}
|
||||
@ -1181,7 +1087,9 @@ bool PRS1Loader::OpenWaveforms(Session *session,QString filename)
|
||||
//EventList *FlowData=EventList(CPAP_FlowRate, EVL_Waveform,1,0,-128,128);
|
||||
//EventList *MaskData=EventList(CPAP_MaskPressure, EVL_Waveform,1,0,-128,128);
|
||||
|
||||
ChannelID wc[2]={CPAP_FlowRate,CPAP_MaskPressure};
|
||||
QString FlowRate="FlowRate";
|
||||
QString MaskPressure="MaskPressure";
|
||||
QString wc[2]={FlowRate,MaskPressure};
|
||||
do {
|
||||
timestamp=m_buffer[pos+0xb] | m_buffer[pos+0xc] << 8 | m_buffer[pos+0xd] << 16 | m_buffer[pos+0x0e] << 24;
|
||||
register unsigned char sum8=0;
|
||||
@ -1226,24 +1134,24 @@ bool PRS1Loader::OpenWaveforms(Session *session,QString filename)
|
||||
double rate=(double(wdur[i])*1000.0)/double(wlength[i]);
|
||||
double gain;
|
||||
if (i==1) gain=0.1; else gain=1;
|
||||
EventList *a=new EventList(wc[i],EVL_Waveform,gain,0,0,0,rate);
|
||||
session->machine()->registerChannel(wc[i]);
|
||||
EventList *a=session->AddEventList(wc[i],EVL_Waveform,gain,0,0,0,rate);
|
||||
//EventList *a=new EventList(wc[i],EVL_Waveform,gain,0,0,0,rate);
|
||||
//session->machine()->registerChannel(wc[i]);
|
||||
if (whl[i].sample_format)
|
||||
a->AddWaveform(qint64(start)*1000L,(unsigned char *)waveform[i],wlength[i],qint64(wdur[i])*1000L);
|
||||
else {
|
||||
a->AddWaveform(qint64(start)*1000L,(char *)waveform[i],wlength[i],qint64(wdur[i])*1000L);
|
||||
}
|
||||
if (wc[i]==CPAP_FlowRate) {
|
||||
if (wc[i]==FlowRate) {
|
||||
a->setMax(120);
|
||||
a->setMin(-120);
|
||||
} else if (wc[i]==CPAP_MaskPressure) {
|
||||
} else if (wc[i]==MaskPressure) {
|
||||
/* int v=ceil(a->max()/5);
|
||||
a->setMax(v*5);
|
||||
v=floor(a->min()/5);
|
||||
a->setMin(v*5); */
|
||||
}
|
||||
|
||||
session->eventlist[wc[i]].push_back(a);
|
||||
session->updateLast(start+(qint64(wdur[i])*1000L));
|
||||
wlength[i]=0;
|
||||
wdur[i]=0;
|
||||
@ -1281,20 +1189,18 @@ bool PRS1Loader::OpenWaveforms(Session *session,QString filename)
|
||||
double rate=(double(wdur[i])*1000.0)/double(wlength[i]);
|
||||
double gain;
|
||||
if (i==1) gain=0.1; else gain=1;
|
||||
EventList *a=new EventList(wc[i],EVL_Waveform,gain,0,0,0,rate);
|
||||
session->machine()->registerChannel(wc[i]);
|
||||
EventList *a=session->AddEventList(wc[i],EVL_Waveform,gain,0,0,0,rate);
|
||||
|
||||
if (whl[i].sample_format)
|
||||
a->AddWaveform(qint64(start)*1000L,(unsigned char *)waveform[i],wlength[i],qint64(wdur[i])*1000L);
|
||||
else {
|
||||
a->AddWaveform(qint64(start)*1000L,(char *)waveform[i],wlength[i],qint64(wdur[i])*1000L);
|
||||
}
|
||||
if (wc[i]==CPAP_FlowRate) {
|
||||
if (wc[i]==FlowRate) {
|
||||
a->setMax(120);
|
||||
a->setMin(-120);
|
||||
} else if (wc[i]==CPAP_MaskPressure) {
|
||||
} else if (wc[i]==MaskPressure) {
|
||||
}
|
||||
session->eventlist[wc[i]].push_back(a);
|
||||
session->updateLast(start+qint64(wdur[i])*1000L);
|
||||
}
|
||||
return true;
|
||||
|
@ -121,6 +121,7 @@ bool EDFParser::Parse()
|
||||
edfsignals.push_back(signal);
|
||||
signal->data=NULL;
|
||||
edfsignals[i]->label=Read(16);
|
||||
lookup[edfsignals[i]->label]=signal;
|
||||
}
|
||||
|
||||
for (int i=0;i<num_signals;i++) edfsignals[i]->transducer_type=Read(80);
|
||||
@ -257,6 +258,29 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
}
|
||||
}
|
||||
}
|
||||
QString strfile=path+"/STR.edf";
|
||||
EDFParser stredf(strfile);
|
||||
|
||||
if (!stredf.Parse()) {
|
||||
qDebug() << "Faulty file" << strfile;
|
||||
return 0;
|
||||
}
|
||||
|
||||
qint64 duration=stredf.GetNumDataRecords()*stredf.GetDuration();
|
||||
int days=duration/86400000L;
|
||||
|
||||
QDateTime dt1=QDateTime::fromTime_t(stredf.startdate/1000L);
|
||||
QDateTime dt2=QDateTime::fromTime_t(stredf.enddate/1000L);
|
||||
QDate dd1=dt1.date();
|
||||
QDate dd2=dt2.date();
|
||||
for (int s=0;s<stredf.GetNumSignals();s++) {
|
||||
EDFSignal & es=*stredf.edfsignals[s];
|
||||
long recs=stredf.edfsignals[s]->nr*stredf.GetNumDataRecords();
|
||||
|
||||
qDebug() << "STREDF:" << stredf.edfsignals[s]->label << recs;
|
||||
}
|
||||
|
||||
|
||||
QDir dir(newpath);
|
||||
|
||||
if ((!dir.exists() || !dir.isReadable()))
|
||||
@ -324,6 +348,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
if (edf.serialnumber!=i.value()) {
|
||||
qDebug() << "edf Serial number doesn't match Identification.tgt";
|
||||
}
|
||||
if (edf.serialnumber!=stredf.serialnumber) {
|
||||
qDebug() << "edf Serial number doesn't match STR.edf!";
|
||||
}
|
||||
} else if (i.key()=="PNA") {
|
||||
m->properties["Model"]=i.value();
|
||||
} else if (i.key()=="PCD") {
|
||||
@ -363,14 +390,49 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
continue;
|
||||
} else {
|
||||
sess->SetChanged(true);
|
||||
m->AddSession(sess,profile); // Adding earlier than I really like here..
|
||||
}
|
||||
if (!done && sess) {
|
||||
qint64 dif=sess->first()-stredf.startdate;
|
||||
int dn=dif/86400000L;
|
||||
if (dn<days) {
|
||||
int mode;
|
||||
mode=(*stredf.lookup["Mode"]).data[dn];
|
||||
sess->settings["EPR"]=(*stredf.lookup["EPR"]).data[dn];
|
||||
sess->settings["EPRSet"]=(*stredf.lookup["EPR Level"]).data[dn];
|
||||
|
||||
EDFSignal *sig;
|
||||
if (mode==0) {
|
||||
sess->settings["PAPMode"]=MODE_CPAP;
|
||||
//m->registerChannel(CPAP_EPAP,false);
|
||||
//m->registerChannel("IPAP",false);
|
||||
sig=stredf.lookup["Set Pressure"];
|
||||
EventDataType pressure=sig->data[dn]*sig->gain;
|
||||
sess->settings[CPAP_Pressure]=pressure;
|
||||
sess->setWavg(CPAP_Pressure,pressure);
|
||||
sess->setAvg(CPAP_Pressure,pressure);
|
||||
sess->set90p(CPAP_Pressure,pressure);
|
||||
sess->setMax(CPAP_Pressure,pressure);
|
||||
sess->setMin(CPAP_Pressure,pressure);
|
||||
} else {
|
||||
if (mode>5) {
|
||||
sess->settings[CPAP_Mode]=MODE_BIPAP;
|
||||
} else {
|
||||
sess->settings[CPAP_Mode]=MODE_APAP;
|
||||
}
|
||||
sig=stredf.lookup["Min Pressure"];
|
||||
if (sig)
|
||||
sess->setMin(CPAP_Pressure,sig->data[dn]*sig->gain);
|
||||
sig=stredf.lookup["Max Pressure"];
|
||||
if (sig)
|
||||
sess->setMax(CPAP_Pressure,sig->data[dn]*sig->gain);
|
||||
}
|
||||
|
||||
}
|
||||
m->AddSession(sess,profile); // Adding earlier than I really like here..
|
||||
}
|
||||
}
|
||||
|
||||
if (m) {
|
||||
m->Save();
|
||||
}
|
||||
if (qprogress) qprogress->setValue(100);
|
||||
qDebug() << "Total Events " << event_cnt;
|
||||
return 1;
|
||||
@ -457,36 +519,24 @@ bool ResmedLoader::LoadEVE(Session *sess,EDFParser &edf)
|
||||
if (!t.isEmpty()) {
|
||||
//code=MC_UNKNOWN;
|
||||
if (t=="obstructive apnea") {
|
||||
code=CPAP_Obstructive;
|
||||
if (!EL[0]) {
|
||||
EL[0]=new EventList(code,EVL_Event);
|
||||
sess->eventlist[code].push_back(EL[0]);
|
||||
sess->machine()->registerChannel(code);
|
||||
|
||||
if (!(EL[0]=sess->AddEventList(CPAP_Obstructive,EVL_Event))) return false;
|
||||
}
|
||||
EL[0]->AddEvent(tt,duration);
|
||||
} else if (t=="hypopnea") {
|
||||
code=CPAP_Hypopnea;
|
||||
if (!EL[1]) {
|
||||
EL[1]=new EventList(code,EVL_Event);
|
||||
sess->eventlist[code].push_back(EL[1]);
|
||||
sess->machine()->registerChannel(code);
|
||||
if (!(EL[1]=sess->AddEventList(CPAP_Hypopnea,EVL_Event))) return false;
|
||||
}
|
||||
EL[1]->AddEvent(tt,duration+10); // Only Hyponea's Need the extra duration???
|
||||
} else if (t=="apnea") {
|
||||
code=CPAP_Apnea;
|
||||
if (!EL[2]) {
|
||||
EL[2]=new EventList(code,EVL_Event);
|
||||
sess->eventlist[code].push_back(EL[2]);
|
||||
sess->machine()->registerChannel(code);
|
||||
if (!(EL[2]=sess->AddEventList(CPAP_Apnea,EVL_Event))) return false;
|
||||
}
|
||||
EL[2]->AddEvent(tt,duration);
|
||||
} else if (t=="central apnea") {
|
||||
code=CPAP_ClearAirway;
|
||||
if (!EL[3]) {
|
||||
EL[3]=new EventList(code,EVL_Event);
|
||||
sess->eventlist[code].push_back(EL[3]);
|
||||
sess->machine()->registerChannel(code);
|
||||
if (!(EL[3]=sess->AddEventList(CPAP_ClearAirway,EVL_Event))) return false;
|
||||
}
|
||||
EL[3]->AddEvent(tt,duration);
|
||||
} else {
|
||||
@ -529,9 +579,8 @@ bool ResmedLoader::LoadBRP(Session *sess,EDFParser &edf)
|
||||
} else if (edf.edfsignals[s]->label.startsWith("Mask Pres")) {
|
||||
code=CPAP_MaskPressure;
|
||||
sess->machine()->registerChannel(code);
|
||||
|
||||
} else if (es.label.startsWith("Resp Event")) {
|
||||
code=CPAP_RespiratoryEvent;
|
||||
code=CPAP_RespEvent;
|
||||
sess->machine()->registerChannel(code);
|
||||
} else {
|
||||
qDebug() << "Unobserved ResMed BRP Signal " << edf.edfsignals[s]->label;
|
||||
@ -539,16 +588,12 @@ bool ResmedLoader::LoadBRP(Session *sess,EDFParser &edf)
|
||||
}
|
||||
double rate=double(duration)/double(recs);
|
||||
//es.gain=1;
|
||||
EventList *a=new EventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
EventList *a=sess->AddEventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
a->setDimension(es.physical_dimension);
|
||||
a->AddWaveform(edf.startdate,es.data,recs,duration);
|
||||
|
||||
if (code==CPAP_MaskPressure) {
|
||||
} else if (code==CPAP_FlowRate) {
|
||||
}
|
||||
sess->setMin(code,a->min());
|
||||
sess->setMax(code,a->max());
|
||||
sess->eventlist[code].push_back(a);
|
||||
//sess->eventlist[code].push_back(a);
|
||||
//delete edf.edfsignals[s]->data;
|
||||
//edf.edfsignals[s]->data=NULL; // so it doesn't get deleted when edf gets trashed.
|
||||
}
|
||||
@ -562,15 +607,14 @@ EventList * ResmedLoader::ToTimeDelta(Session *sess,EDFParser &edf, EDFSignal &
|
||||
//sess->UpdateFirst(tt);
|
||||
EventDataType c,last;
|
||||
//if (gain==0) gain=1;
|
||||
EventList *el=new EventList(code,EVL_Event,es.gain,es.offset,min,max);
|
||||
sess->eventlist[code].push_back(el);
|
||||
|
||||
EventList *el=sess->AddEventList(code,EVL_Event,es.gain,es.offset,min,max);
|
||||
int startpos=0;
|
||||
|
||||
/*if ((code==CPAP_Pressure) || (code==CPAP_IPAP) || (code==CPAP_EPAP)
|
||||
|| (code==CPAP_TherapyPressure)) {
|
||||
startpos=0;
|
||||
if ((code==CPAP_Pressure) || (code==CPAP_IPAP) || (code==CPAP_EPAP)) {
|
||||
startpos=20; // Shave the first 20 seconds of pressure data
|
||||
tt+=rate*startpos;
|
||||
}*/
|
||||
}
|
||||
for (int i=startpos;i<recs;i++) {
|
||||
c=es.data[i];
|
||||
|
||||
@ -603,11 +647,11 @@ bool ResmedLoader::LoadSAD(Session *sess,EDFParser &edf)
|
||||
long recs=edf.edfsignals[s]->nr*edf.GetNumDataRecords();
|
||||
ChannelID code;
|
||||
if (edf.edfsignals[s]->label=="Pulse") {
|
||||
code=CPAP_Pulse;
|
||||
sess->machine()->registerChannel(code);
|
||||
code=OXI_Pulse;
|
||||
//sess->machine()->registerChannel(code);
|
||||
} else if (edf.edfsignals[s]->label=="SpO2") {
|
||||
code=CPAP_SPO2;
|
||||
sess->machine()->registerChannel(code);
|
||||
code=OXI_SPO2;
|
||||
//sess->machine()->registerChannel(code);
|
||||
} else {
|
||||
qDebug() << "Unobserved ResMed SAD Signal " << edf.edfsignals[s]->label;
|
||||
continue;
|
||||
@ -637,8 +681,8 @@ bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf)
|
||||
// Is it save to assume the order does not change here?
|
||||
enum PLDType { MaskPres=0, TherapyPres, ExpPress, Leak, RR, Vt, Mv, SnoreIndex, FFLIndex, U1, U2 };
|
||||
|
||||
sess->updateFirst(edf.startdate);
|
||||
qint64 duration=edf.GetNumDataRecords()*edf.GetDuration();
|
||||
sess->updateFirst(edf.startdate);
|
||||
sess->updateLast(edf.startdate+duration);
|
||||
QString t;
|
||||
int emptycnt=0;
|
||||
@ -649,6 +693,7 @@ bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf)
|
||||
for (int s=0;s<edf.GetNumSignals();s++) {
|
||||
EDFSignal & es=*edf.edfsignals[s];
|
||||
recs=es.nr*edf.GetNumDataRecords();
|
||||
if (recs<=0) continue;
|
||||
rate=double(duration)/double(recs);
|
||||
//qDebug() << "EVE:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum << es.gain;
|
||||
if (es.label=="Snore Index") {
|
||||
@ -661,8 +706,8 @@ bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf)
|
||||
//a->setMax(1);
|
||||
//a->setMin(0);
|
||||
} else if (es.label=="Therapy Pres") {
|
||||
code=CPAP_IPAP; //TherapyPressure;
|
||||
sess->settings[CPAP_Mode]=MODE_APAP;
|
||||
code=CPAP_Pressure; //TherapyPressure;
|
||||
//sess->settings[CPAP_Mode]=MODE_APAP;
|
||||
//EventList *a=new EventList(code,EVL_Waveform,es.gain,es.offset,es.physical_minimum,es.physical_maximum,rate);
|
||||
//sess->eventlist[code].push_back(a);
|
||||
//a->AddWaveform(edf.startdate,es.data,recs,duration);
|
||||
@ -675,15 +720,14 @@ bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf)
|
||||
//a->AddWaveform(edf.startdate,es.data,recs,duration);
|
||||
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
||||
} else if (es.label=="MV") {
|
||||
code=CPAP_MinuteVentilation;
|
||||
code=CPAP_MinuteVent;
|
||||
//EventList *a=new EventList(code,EVL_Waveform,es.gain,es.offset,es.physical_minimum,es.physical_maximum,rate);
|
||||
//sess->eventlist[code].push_back(a);
|
||||
//a->AddWaveform(edf.startdate,es.data,recs,duration);
|
||||
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
||||
} else if (es.label=="RR") {
|
||||
code=CPAP_RespiratoryRate;
|
||||
a=new EventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
sess->eventlist[code].push_back(a);
|
||||
code=CPAP_RespRate;
|
||||
a=sess->AddEventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
a->AddWaveform(edf.startdate,es.data,recs,duration);
|
||||
//ToTimeDelta(sess,edf,es, code,recs,duration);
|
||||
} else if (es.label=="Vt") {
|
||||
@ -704,13 +748,13 @@ bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf)
|
||||
//a->setMax(1);
|
||||
//a->setMin(0);
|
||||
} else if (es.label=="FFL Index") {
|
||||
code=CPAP_FlowLimitGraph;
|
||||
code=CPAP_FLG;
|
||||
//es.gain=1;//10.0;
|
||||
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
||||
//a->setMax(1);
|
||||
//a->setMin(0);
|
||||
} else if (es.label.startsWith("Mask Pres")) {
|
||||
code=CPAP_Pressure;
|
||||
code=CPAP_MaskPressure;
|
||||
//es.gain=1;
|
||||
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
||||
} else if (es.label.startsWith("Exp Press")) {
|
||||
@ -718,28 +762,25 @@ bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf)
|
||||
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
||||
} else if (es.label.startsWith("I:E")) {
|
||||
code=CPAP_IE;//I:E;
|
||||
a=new EventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
sess->eventlist[code].push_back(a);
|
||||
a=sess->AddEventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
a->AddWaveform(edf.startdate,es.data,recs,duration);
|
||||
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
||||
} else if (es.label.startsWith("Ti")) {
|
||||
code=CPAP_Ti;//Ti;
|
||||
a=new EventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
sess->eventlist[code].push_back(a);
|
||||
a=sess->AddEventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
a->AddWaveform(edf.startdate,es.data,recs,duration);
|
||||
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
||||
} else if (es.label.startsWith("Te")) {
|
||||
code=CPAP_Te;//Te;
|
||||
a=new EventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
sess->eventlist[code].push_back(a);
|
||||
a=sess->AddEventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
||||
a->AddWaveform(edf.startdate,es.data,recs,duration);
|
||||
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
||||
} else if (es.label=="") {
|
||||
if (emptycnt==0) {
|
||||
code=RMS9_Empty1;
|
||||
code=RMS9_E01;
|
||||
a=ToTimeDelta(sess,edf,es, code,recs,duration);
|
||||
} else if (emptycnt==1) {
|
||||
code=RMS9_Empty2;
|
||||
code=RMS9_E02;
|
||||
a=ToTimeDelta(sess,edf,es, code,recs,duration);
|
||||
} else {
|
||||
qDebug() << "Unobserved Empty Signal " << es.label;
|
||||
@ -750,7 +791,7 @@ bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf)
|
||||
a=NULL;
|
||||
}
|
||||
if (a) {
|
||||
sess->machine()->registerChannel(code);
|
||||
//sess->machine()->registerChannel(code);
|
||||
sess->setMin(code,a->min());
|
||||
sess->setMax(code,a->max());
|
||||
a->setDimension(es.physical_dimension);
|
||||
|
@ -69,11 +69,13 @@ public:
|
||||
qint16 Read16();
|
||||
|
||||
QVector<EDFSignal *> edfsignals;
|
||||
QHash<QString,EDFSignal *> lookup;
|
||||
|
||||
long GetNumSignals() { return num_signals; }
|
||||
long GetNumDataRecords() { return num_data_records; }
|
||||
qint64 GetDuration() { return dur_data_record; }
|
||||
QString GetPatient() { return patientident; }
|
||||
|
||||
bool Parse();
|
||||
char *buffer;
|
||||
EDFHeader header;
|
||||
|
@ -16,291 +16,10 @@
|
||||
#include "machine.h"
|
||||
#include "profiles.h"
|
||||
#include <algorithm>
|
||||
|
||||
ChannelGroup CPAP_CODES;
|
||||
ChannelGroup PRS1_CODES;
|
||||
ChannelGroup RMS9_CODES;
|
||||
ChannelGroup OXI_CODES;
|
||||
ChannelGroup ZEO_CODES;
|
||||
ChannelGroup JOURNAL_CODES;
|
||||
|
||||
|
||||
ChannelID CPAP_Obstructive,CPAP_Hypopnea,CPAP_Apnea, CPAP_ClearAirway, CPAP_RERA, CPAP_FlowLimit,
|
||||
CPAP_CSR, CPAP_VSnore, CPAP_PressurePulse, CPAP_Mode, CPAP_FlowRate, CPAP_MaskPressure, CPAP_Pressure,
|
||||
CPAP_EPAP, CPAP_IPAP, CPAP_IPAP_Low, CPAP_IPAP_High, CPAP_PressureSupport, CPAP_Snore, CPAP_Leak,
|
||||
CPAP_RespiratoryRate, CPAP_TidalVolume, CPAP_MinuteVentilation, CPAP_PatientTriggeredBreaths,
|
||||
CPAP_FlowLimitGraph, CPAP_TherapyPressure, CPAP_ExpiratoryPressure, CPAP_AHI,CPAP_RespiratoryEvent,
|
||||
CPAP_IE,CPAP_Ti,CPAP_Te,
|
||||
CPAP_BrokenSummary,CPAP_BrokenWaveform, CPAP_Pulse, CPAP_SPO2, CPAP_Plethy;
|
||||
|
||||
ChannelID RMS9_PressureReliefType, RMS9_PressureReliefSetting, RMS9_Empty1, RMS9_Empty2;
|
||||
ChannelID PRS1_PressureMin,PRS1_PressureMax, PRS1_PressureMinAchieved, PRS1_PressureMaxAchieved,
|
||||
PRS1_PressureAvg, PRS1_Pressure90, PRS1_RampTime, PRS1_RampPressure, PRS1_HumidifierStatus,
|
||||
PRS1_HumidifierSetting, PRS1_PressureReliefType, PRS1_PressureReliefSetting, PRS1_SystemOneResistanceStatus,
|
||||
PRS1_SystemOneResistanceSetting, PRS1_SystemLockStatus, PRS1_HoseDiameter, PRS1_AutoOff, PRS1_AutoOn,
|
||||
PRS1_MaskAlert, PRS1_ShowAHI, PRS1_Unknown00, PRS1_Unknown01, PRS1_Unknown08, PRS1_Unknown09,
|
||||
PRS1_Unknown0A, PRS1_Unknown0B, PRS1_Unknown0C, PRS1_Unknown0E, PRS1_Unknown0F, PRS1_Unknown10,
|
||||
PRS1_Unknown12, PRS1_VSnore2;
|
||||
|
||||
ChannelID OXI_Pulse, OXI_SPO2, OXI_Plethysomogram, OXI_PulseChange, OXI_SPO2Drop, OXI_Error, OXI_SignalError;
|
||||
ChannelID ZEO_SleepStage, ZEO_Waveform;
|
||||
ChannelID JOURNAL_Notes, JOURNAL_Weight;
|
||||
|
||||
ChannelID ChannelGroup::Get(ChannelType ctype,QString description,QString label,QString lookup,QColor color)
|
||||
{
|
||||
ChannelID id=m_first+m_pos;
|
||||
if (++m_pos>=m_size) {
|
||||
qCritical("Not enough slots allocated for channel group");
|
||||
abort();
|
||||
}
|
||||
channel[id]=Channel(id,m_type,ctype,description,label,color);
|
||||
m_channelsbytype[ctype][id]=channel_lookup[lookup]=m_channel[id]=&channel[id];
|
||||
|
||||
return id;
|
||||
}
|
||||
ChannelGroup::ChannelGroup()
|
||||
{
|
||||
m_pos=0; m_first=0xf000, m_size=0x1000;
|
||||
}
|
||||
ChannelGroup::ChannelGroup(MachineType type, ChannelID first, ChannelID reserved)
|
||||
:m_type(type),m_first(first),m_size(reserved)
|
||||
{
|
||||
m_pos=0;
|
||||
}
|
||||
|
||||
#include "SleepLib/schema.h"
|
||||
|
||||
extern QProgressBar * qprogress;
|
||||
|
||||
// Master Lists..
|
||||
QHash<ChannelID, Channel> channel;
|
||||
QHash<QString,Channel *> channel_lookup;
|
||||
QHash<QString,ChannelGroup *> channel_group;
|
||||
|
||||
|
||||
void InitMapsWithoutAwesomeInitializerLists()
|
||||
{
|
||||
CPAP_CODES=ChannelGroup(MT_CPAP,0x1000,0x400);
|
||||
PRS1_CODES=ChannelGroup(MT_CPAP,0x1400,0x200);
|
||||
RMS9_CODES=ChannelGroup(MT_CPAP,0x1600,0x200);
|
||||
OXI_CODES=ChannelGroup(MT_OXIMETER,0x2000,0x800);
|
||||
ZEO_CODES=ChannelGroup(MT_SLEEPSTAGE,0x2800,0x800);
|
||||
JOURNAL_CODES=ChannelGroup(MT_JOURNAL,0x3000,0x800);
|
||||
|
||||
// ******************** IMPORTANT ********************
|
||||
// Add to the end of each group or be eaten by a Grue!
|
||||
// ******************** IMPORTANT ********************
|
||||
|
||||
// Flagable Events
|
||||
CPAP_Obstructive=CPAP_CODES.Get(CT_Event,QObject::tr("Obstructive Apnea"),QObject::tr("OA"),"OA"),
|
||||
CPAP_Hypopnea=CPAP_CODES.Get(CT_Event,QObject::tr("Hypopnea"),QObject::tr("H"),"OA"),
|
||||
CPAP_Apnea=CPAP_CODES.Get(CT_Event,QObject::tr("Unspecified Apnea"),QObject::tr("UA"),"UA"),
|
||||
CPAP_ClearAirway=CPAP_CODES.Get(CT_Event,QObject::tr("Clear Airway Apnea"),QObject::tr("CA"),"CA"),
|
||||
CPAP_RERA=CPAP_CODES.Get(CT_Event,QObject::tr("RERA"),QObject::tr("RE"),"RE"),
|
||||
CPAP_FlowLimit=CPAP_CODES.Get(CT_Event,QObject::tr("Flow Limitation"),QObject::tr("FL"),"FL"),
|
||||
CPAP_CSR=CPAP_CODES.Get(CT_Event,QObject::tr("Periodic Breathing"),QObject::tr("PB"),"PB"),
|
||||
CPAP_VSnore=CPAP_CODES.Get(CT_Event,QObject::tr("Vibratory Snore"),QObject::tr("VS"),"VS"),
|
||||
CPAP_PressurePulse=CPAP_CODES.Get(CT_Event,QObject::tr("Pressure Pulse"),QObject::tr("PP"),"PP"),
|
||||
CPAP_Mode=CPAP_CODES.Get(CT_Lookup,QObject::tr("CPAP Mode"),QObject::tr("CPAPMode"),"CPAPMode"),
|
||||
// Alternate names at the end of each section
|
||||
|
||||
// Graphable Events & Waveforms
|
||||
CPAP_FlowRate=CPAP_CODES.Get(CT_Graph,QObject::tr("Flow Rate Waveform"),QObject::tr("Flow Rate"),"FR"),
|
||||
CPAP_MaskPressure=CPAP_CODES.Get(CT_Graph,QObject::tr("Mask Pressure"),QObject::tr("Mask Pressure"),"MP"),
|
||||
CPAP_Pressure=CPAP_CODES.Get(CT_Graph,QObject::tr("Pressure"),QObject::tr("Pressure"),"P"),
|
||||
CPAP_EPAP=CPAP_CODES.Get(CT_Graph,QObject::tr("Expiratory Pressure"),QObject::tr("EPAP"),"EPAP"),
|
||||
CPAP_IPAP=CPAP_CODES.Get(CT_Graph,QObject::tr("Inhalation Pressure"),QObject::tr("IPAP"),"IPAP"),
|
||||
CPAP_IPAP_Low=CPAP_CODES.Get(CT_Graph,QObject::tr("IPAP Low"),QObject::tr("IPAP Low"),"IPAPL"),
|
||||
CPAP_IPAP_High=CPAP_CODES.Get(CT_Graph,QObject::tr("IPAP Low"),QObject::tr("IPAP High"),"IPAPH"),
|
||||
CPAP_PressureSupport=CPAP_CODES.Get(CT_Graph,QObject::tr("Pressure Support"),QObject::tr("Pressure Support"),"PS"),
|
||||
CPAP_Snore=CPAP_CODES.Get(CT_Graph,QObject::tr("Snore"),QObject::tr("Snore"),"Snore"),
|
||||
CPAP_Leak=CPAP_CODES.Get(CT_Graph,QObject::tr("Leak Rate"),QObject::tr("Leak"),"Leak"),
|
||||
CPAP_RespiratoryRate=CPAP_CODES.Get(CT_Graph,QObject::tr("Respiratory Rate"),QObject::tr("Resp. Rate"),"RR"),
|
||||
CPAP_TidalVolume=CPAP_CODES.Get(CT_Graph,QObject::tr("Tidal Volume"),QObject::tr("Tidal Volume"),"TV"),
|
||||
CPAP_MinuteVentilation=CPAP_CODES.Get(CT_Graph,QObject::tr("Minute Ventilation"),QObject::tr("Minute Vent."),"MV"),
|
||||
CPAP_PatientTriggeredBreaths=CPAP_CODES.Get(CT_Graph,QObject::tr("Patient Triggered Breaths"),QObject::tr("Patient Trig Breaths"),"PTB"),
|
||||
CPAP_FlowLimitGraph=CPAP_CODES.Get(CT_Graph,QObject::tr("Flow Limitation Graph"),QObject::tr("Flow Limit."),"FLG"),
|
||||
CPAP_TherapyPressure=CPAP_CODES.Get(CT_Graph,QObject::tr("Therapy Pressure"),QObject::tr("Therapy Pressure"),"TP"),
|
||||
CPAP_ExpiratoryPressure=CPAP_CODES.Get(CT_Graph,QObject::tr("Expiratory Pressure"),QObject::tr("Expiratory Pressure"),"EXP"),
|
||||
CPAP_RespiratoryEvent=CPAP_CODES.Get(CT_Graph,QObject::tr("Respiratory Event"),QObject::tr("Respiratory Event"),"RESPEv"),
|
||||
CPAP_IE=CPAP_CODES.Get(CT_Graph,QObject::tr("I:E"),QObject::tr("I:E"),"IE"),
|
||||
CPAP_Ti=CPAP_CODES.Get(CT_Graph,QObject::tr("Ti"),QObject::tr("Ti"),"Ti"),
|
||||
CPAP_Te=CPAP_CODES.Get(CT_Graph,QObject::tr("Te"),QObject::tr("Te"),"Te"),
|
||||
|
||||
|
||||
CPAP_AHI=CPAP_CODES.Get(CT_Calculation,QObject::tr("Apnea-Hypopnea Index"),QObject::tr("AHI"),"AHI"),
|
||||
/*CPAP_AI=CPAP_CODES.Get(CT_Calculation,QObject::tr("Apnea Index"),QObject::tr("AI"),"AI"),
|
||||
CPAP_HI=CPAP_CODES.Get(CT_Calculation,QObject::tr("Hypopnea Index"),QObject::tr("HI"),"HI"),
|
||||
CPAP_CAI=CPAP_CODES.Get(CT_Calculation,QObject::tr("Central Apnea Index"),QObject::tr("CAI"),"CAI"),
|
||||
CPAP_RAI=CPAP_CODES.Get(CT_Calculation,QObject::tr("RERA+AHI Index"),QObject::tr("RAI"),"RAI"),
|
||||
CPAP_RI=CPAP_CODES.Get(CT_Calculation,QObject::tr("RERA Index"),QObject::tr("RI"),"RI"),
|
||||
CPAP_FLI=CPAP_CODES.Get(CT_Calculation,QObject::tr("Flow Limitation Index"),QObject::tr("FLI"),"FLI"),
|
||||
CPAP_VSI=CPAP_CODES.Get(CT_Calculation,QObject::tr("Vibratory Snore Index"),QObject::tr("VSI"),"VSI"),
|
||||
CPAP_PBI=CPAP_CODES.Get(CT_Calculation,QObject::tr("% Night in Periodic Breathing"),QObject::tr("PBI"),"PBI"), */
|
||||
//CPAP_PressureReliefType=CPAP_CODES.Get(CT_Lookup,QObject::tr("Pressure Relief Type"),"","PRType"),
|
||||
//CPAP_PressureReliefSetting=CPAP_CODES.Get(CT_Lookup,QObject::tr("Pressure Relief Setting"),"","PRSet"),
|
||||
CPAP_BrokenSummary=CPAP_CODES.Get(CT_Boolean,QObject::tr("Broken Summary"),QObject::tr(""),"BrokenSummary"),
|
||||
CPAP_BrokenWaveform=CPAP_CODES.Get(CT_Boolean,QObject::tr("Broken Waveform"),QObject::tr(""),"BrokenWaveform");
|
||||
|
||||
|
||||
RMS9_PressureReliefType=RMS9_CODES.Get(CT_Lookup,QObject::tr("Pressure Relief Type"),"","RMS9_PRType"),
|
||||
RMS9_PressureReliefSetting=RMS9_CODES.Get(CT_Decimal,QObject::tr("Pressure Relief Setting"),"","RMS9_PRSet"),
|
||||
RMS9_Empty1=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown Empty 1"),"","RMS9_UE1"),
|
||||
RMS9_Empty2=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown Empty 2"),"","RMS9_UE2"),
|
||||
|
||||
PRS1_PressureMin=PRS1_CODES.Get(CT_Decimal,QObject::tr("Minimum Pressure"),"","PRS1_MinPres"),
|
||||
PRS1_PressureMax=PRS1_CODES.Get(CT_Decimal,QObject::tr("Maximum Pressure"),"","PRS1_MaxPres"),
|
||||
PRS1_PressureMinAchieved=PRS1_CODES.Get(CT_Decimal,QObject::tr("Minimum Pressure Achieved"),"","PRS1_MinPresAch"),
|
||||
PRS1_PressureMaxAchieved=PRS1_CODES.Get(CT_Decimal,QObject::tr("Maximum Pressure Achieved"),"","PRS1_MaxPresAch"),
|
||||
PRS1_PressureAvg=PRS1_CODES.Get(CT_Decimal,QObject::tr("Average Pressure"),"","PRS1_AvgPres"),
|
||||
PRS1_Pressure90=PRS1_CODES.Get(CT_Decimal,QObject::tr("90% Pressure"),"","PRS1_90Pres"),
|
||||
PRS1_RampTime=PRS1_CODES.Get(CT_Timespan,QObject::tr("Ramp Time"),"","PRS1_RampTime"),
|
||||
PRS1_RampPressure=PRS1_CODES.Get(CT_Decimal,QObject::tr("Ramp Pressure"),"","PRS1_RampPressure"),
|
||||
PRS1_HumidifierStatus=PRS1_CODES.Get(CT_Boolean,QObject::tr("Humidifier Status"),"","PRS1_HumiStat"),
|
||||
PRS1_HumidifierSetting=PRS1_CODES.Get(CT_Lookup,QObject::tr("Humidifier Setting"),"","PRS1_HumiSet"),
|
||||
PRS1_PressureReliefType=PRS1_CODES.Get(CT_Lookup,QObject::tr("Pressure Relief Type"),"","PRS1_PRType"),
|
||||
PRS1_PressureReliefSetting=PRS1_CODES.Get(CT_Lookup,QObject::tr("Pressure Relief Setting"),"","PRS1_PRSet"),
|
||||
PRS1_SystemOneResistanceStatus=PRS1_CODES.Get(CT_Boolean,QObject::tr("System-One Resistance Status"),"","PRS1_ResistStat"),
|
||||
PRS1_SystemOneResistanceSetting=PRS1_CODES.Get(CT_Lookup,QObject::tr("System-One Resistance Setting"),"","PRS1_ResistSet"),
|
||||
PRS1_SystemLockStatus=PRS1_CODES.Get(CT_Boolean,QObject::tr("System Lock Status"),"","PRS1_SysLockStat"),
|
||||
PRS1_HoseDiameter=PRS1_CODES.Get(CT_Lookup,QObject::tr("Hose Diameter"),"","PRS1_HoseDiam"),
|
||||
PRS1_AutoOff=PRS1_CODES.Get(CT_Boolean,QObject::tr("Auto Off"),"","PRS1_AutoOff"),
|
||||
PRS1_AutoOn=PRS1_CODES.Get(CT_Boolean,QObject::tr("Auto On"),"","PRS1_AutoOn"),
|
||||
PRS1_MaskAlert=PRS1_CODES.Get(CT_Boolean,QObject::tr("Mask Alert"),"","PRS1_MaskAlert"),
|
||||
PRS1_ShowAHI=PRS1_CODES.Get(CT_Boolean,QObject::tr("Show AHI"),"","PRS1_ShowAHI"),
|
||||
PRS1_Unknown00=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 00 Event"),QObject::tr("U00"),"PRS1_U00"),
|
||||
PRS1_Unknown01=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 01 Event"),QObject::tr("U01"),"PRS1_U01"),
|
||||
PRS1_Unknown08=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 08 Event"),QObject::tr("U08"),"PRS1_U08"),
|
||||
PRS1_Unknown09=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 09 Event"),QObject::tr("U09"),"PRS1_U09"),
|
||||
PRS1_Unknown0A=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 0A Event"),QObject::tr("U0A"),"PRS1_U0A"),
|
||||
PRS1_Unknown0B=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 0B Event"),QObject::tr("U0B"),"PRS1_U0B"),
|
||||
PRS1_Unknown0C=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 0C Event"),QObject::tr("U0C"),"PRS1_U0C"),
|
||||
PRS1_Unknown0E=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 0E Event"),QObject::tr("U0E"),"PRS1_U0E"),
|
||||
PRS1_Unknown0F=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 0F Event"),QObject::tr("U0F"),"PRS1_U0F"),
|
||||
PRS1_Unknown10=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 10 Event"),QObject::tr("U10"),"PRS1_U10"),
|
||||
PRS1_Unknown12=CPAP_CODES.Get(CT_Event,QObject::tr("Unknown 12 Event"),QObject::tr("U12"),"PRS1_U12"),
|
||||
PRS1_VSnore2=CPAP_CODES.Get(CT_Event,QObject::tr("Vibratory Snore (Type 2)"),QObject::tr("VS2"),"VS2");
|
||||
|
||||
// CPAP Integrated oximetery codes..
|
||||
CPAP_Pulse=CPAP_CODES.Get(CT_Graph,QObject::tr("Pulse Rate"),QObject::tr("Pulse"),"CPPR");
|
||||
CPAP_SPO2=CPAP_CODES.Get(CT_Graph,QObject::tr("SpO2"),QObject::tr("SpO2"),"CPSP");
|
||||
CPAP_Plethy=CPAP_CODES.Get(CT_Graph,QObject::tr("Plethysomagram"),QObject::tr("Plethysomagram"),"CPPL");
|
||||
|
||||
OXI_Pulse=OXI_CODES.Get(CT_Graph,QObject::tr("Pulse Rate"),QObject::tr("PR"),"PR"),
|
||||
OXI_SPO2=OXI_CODES.Get(CT_Graph,QObject::tr("Oxygen Saturation"),QObject::tr("SPO2"),"SPO2"),
|
||||
OXI_Plethysomogram=OXI_CODES.Get(CT_Graph,QObject::tr("Plethysomogram"),QObject::tr("PLE"),"PLE"),
|
||||
|
||||
OXI_PulseChange=OXI_CODES.Get(CT_Event,QObject::tr("Pulse Change"),QObject::tr("PC"),"PC"),
|
||||
OXI_SPO2Drop=OXI_CODES.Get(CT_Event,QObject::tr("Oxigen Saturation Drop"),QObject::tr("SD"),"SD"),
|
||||
OXI_Error=OXI_CODES.Get(CT_Event,QObject::tr("Oximeter Error"),QObject::tr("ER1"),"ER1"),
|
||||
OXI_SignalError=OXI_CODES.Get(CT_Event,QObject::tr("Oximeter Signal Error"),QObject::tr("ER2"),"ER2");
|
||||
|
||||
|
||||
ZEO_SleepStage=ZEO_CODES.Get(CT_Graph,QObject::tr("Sleep Stage"),QObject::tr("SS"),"SS"),
|
||||
ZEO_Waveform=ZEO_CODES.Get(CT_Graph,QObject::tr("Brainwaves"),QObject::tr("BW"),"BW");
|
||||
|
||||
|
||||
|
||||
JOURNAL_Notes=JOURNAL_CODES.Get(CT_Text,"Journal/Notes","JN"),
|
||||
JOURNAL_Weight=JOURNAL_CODES.Get(CT_Decimal,"Weight","W");
|
||||
|
||||
channel[CPAP_Mode].addOption("CPAP");
|
||||
channel[CPAP_Mode].addOption("Auto");
|
||||
channel[CPAP_Mode].addOption("BIPAP/VPAP");
|
||||
channel[CPAP_Mode].addOption("ASV");
|
||||
|
||||
channel[PRS1_HoseDiameter].addOption("22mm");
|
||||
channel[PRS1_HoseDiameter].addOption("15mm");
|
||||
channel[PRS1_HumidifierSetting].addOption("Off");
|
||||
channel[PRS1_HumidifierSetting].addOption("1");
|
||||
channel[PRS1_HumidifierSetting].addOption("2");
|
||||
channel[PRS1_HumidifierSetting].addOption("3");
|
||||
channel[PRS1_HumidifierSetting].addOption("4");
|
||||
channel[PRS1_HumidifierSetting].addOption("5");
|
||||
channel[PRS1_SystemOneResistanceSetting].addOption("x1");
|
||||
channel[PRS1_SystemOneResistanceSetting].addOption("x2");
|
||||
channel[PRS1_SystemOneResistanceSetting].addOption("x3");
|
||||
channel[PRS1_SystemOneResistanceSetting].addOption("x4");
|
||||
channel[PRS1_SystemOneResistanceSetting].addOption("x5");
|
||||
channel[PRS1_PressureReliefType].addOption("None");
|
||||
channel[PRS1_PressureReliefType].addOption("C-Flex");
|
||||
channel[PRS1_PressureReliefType].addOption("C-Flex+");
|
||||
channel[PRS1_PressureReliefType].addOption("A-Flex");
|
||||
channel[PRS1_PressureReliefType].addOption("Bi-Flex");
|
||||
channel[PRS1_PressureReliefType].addOption("Bi-Flex+");
|
||||
channel[PRS1_PressureReliefSetting].addOption("Off");
|
||||
channel[PRS1_PressureReliefSetting].addOption("1");
|
||||
channel[PRS1_PressureReliefSetting].addOption("2");
|
||||
channel[PRS1_PressureReliefSetting].addOption("3");
|
||||
channel[RMS9_PressureReliefType].addOption("None");
|
||||
channel[RMS9_PressureReliefType].addOption("EPR");
|
||||
channel[RMS9_PressureReliefSetting].addOption("0");
|
||||
channel[RMS9_PressureReliefSetting].addOption("0.5");
|
||||
channel[RMS9_PressureReliefSetting].addOption("1");
|
||||
channel[RMS9_PressureReliefSetting].addOption("1.5");
|
||||
channel[RMS9_PressureReliefSetting].addOption("2");
|
||||
channel[RMS9_PressureReliefSetting].addOption("2.5");
|
||||
channel[RMS9_PressureReliefSetting].addOption("3.0");
|
||||
|
||||
channel[ZEO_SleepStage].addOption("Unknown");
|
||||
channel[ZEO_SleepStage].addOption("Awake");
|
||||
channel[ZEO_SleepStage].addOption("REM");
|
||||
channel[ZEO_SleepStage].addOption("Light Sleep");
|
||||
channel[ZEO_SleepStage].addOption("Deep Sleep");
|
||||
|
||||
/* CPAPModeNames[MODE_UNKNOWN]=QObject::tr("Undetermined");
|
||||
CPAPModeNames[MODE_CPAP]=QObject::tr("CPAP");
|
||||
CPAPModeNames[MODE_APAP]=QObject::tr("Auto");
|
||||
CPAPModeNames[MODE_BIPAP]=QObject::tr("BIPAP");
|
||||
CPAPModeNames[MODE_ASV]=QObject::tr("ASV");
|
||||
|
||||
PressureReliefNames[PR_UNKNOWN]=QObject::tr("Undetermined");
|
||||
PressureReliefNames[PR_NONE]=QObject::tr("None");
|
||||
PressureReliefNames[PR_CFLEX]=QObject::tr("C-Flex");
|
||||
PressureReliefNames[PR_CFLEXPLUS]=QObject::tr("C-Flex+");
|
||||
PressureReliefNames[PR_AFLEX]=QObject::tr("A-Flex");
|
||||
PressureReliefNames[PR_BIFLEX]=QObject::tr("Bi-Flex");
|
||||
PressureReliefNames[PR_EPR]=QObject::tr("Exhalation Pressure Relief (EPR)");
|
||||
PressureReliefNames[PR_SMARTFLEX]=QObject::tr("SmartFlex");
|
||||
|
||||
DefaultMCShortNames[CPAP_Obstructive]=QObject::tr("OA");
|
||||
DefaultMCShortNames[CPAP_Apnea]=QObject::tr("A");
|
||||
DefaultMCShortNames[CPAP_Hypopnea]=QObject::tr("H");
|
||||
DefaultMCShortNames[CPAP_RERA]=QObject::tr("RE");
|
||||
DefaultMCShortNames[CPAP_ClearAirway]=QObject::tr("CA");
|
||||
DefaultMCShortNames[CPAP_CSR]=QObject::tr("CSR/PB");
|
||||
DefaultMCShortNames[CPAP_VSnore]=QObject::tr("VS");
|
||||
DefaultMCShortNames[PRS1_VSnore2]=QObject::tr("VS2");
|
||||
DefaultMCShortNames[CPAP_FlowLimit]=QObject::tr("FL");
|
||||
DefaultMCShortNames[CPAP_Pressure]=QObject::tr("P");
|
||||
DefaultMCShortNames[CPAP_Leak]=QObject::tr("LR");
|
||||
DefaultMCShortNames[CPAP_EAP]=QObject::tr("EPAP");
|
||||
DefaultMCShortNames[CPAP_IAP]=QObject::tr("IPAP");
|
||||
DefaultMCShortNames[PRS1_PressurePulse]=QObject::tr("PP");
|
||||
|
||||
DefaultMCLongNames[CPAP_Obstructive]=QObject::tr("Obstructive Apnea");
|
||||
DefaultMCLongNames[CPAP_Hypopnea]=QObject::tr("Hypopnea");
|
||||
DefaultMCLongNames[CPAP_Apnea]=QObject::tr("Apnea");
|
||||
DefaultMCLongNames[CPAP_RERA]=QObject::tr("RERA");
|
||||
DefaultMCLongNames[CPAP_ClearAirway]=QObject::tr("Clear Airway Apnea");
|
||||
DefaultMCLongNames[CPAP_CSR]=QObject::tr("Periodic Breathing");
|
||||
DefaultMCLongNames[CPAP_VSnore]=QObject::tr("Vibratory Snore"); // flags type
|
||||
DefaultMCLongNames[CPAP_FlowLimit]=QObject::tr("Flow Limitation");
|
||||
DefaultMCLongNames[CPAP_Pressure]=QObject::tr("Pressure");
|
||||
DefaultMCLongNames[CPAP_Leak]=QObject::tr("Leak Rate");
|
||||
DefaultMCLongNames[CPAP_PS]=QObject::tr("Pressure Support");
|
||||
DefaultMCLongNames[CPAP_EAP]=QObject::tr("BIPAP EPAP");
|
||||
DefaultMCLongNames[CPAP_IAP]=QObject::tr("BIPAP IPAP");
|
||||
DefaultMCLongNames[CPAP_Snore]=QObject::tr("Vibratory Snore"); // Graph data
|
||||
DefaultMCLongNames[PRS1_VSnore2]=QObject::tr("Vibratory Snore (Graph)");
|
||||
DefaultMCLongNames[PRS1_PressurePulse]=QObject::tr("Pressure Pulse");
|
||||
DefaultMCLongNames[PRS1_Unknown0E]=QObject::tr("Unknown 0E");
|
||||
DefaultMCLongNames[PRS1_Unknown00]=QObject::tr("Unknown 00");
|
||||
DefaultMCLongNames[PRS1_Unknown01]=QObject::tr("Unknown 01");
|
||||
DefaultMCLongNames[PRS1_Unknown0B]=QObject::tr("Unknown 0B");
|
||||
DefaultMCLongNames[PRS1_Unknown10]=QObject::tr("Unknown 10"); */
|
||||
}
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////////////////
|
||||
// Machine Base-Class implmementation
|
||||
//////////////////////////////////////////////////////////////////////////////////////////
|
||||
@ -374,7 +93,7 @@ Day *Machine::AddSession(Session *s,Profile *p)
|
||||
|
||||
sessionlist[s->session()]=s; // To make sure it get's saved later even if it's not wanted.
|
||||
|
||||
QDateTime d1,d2=QDateTime::fromTime_t(s->first()/1000);
|
||||
QDateTime d2=QDateTime::fromTime_t(s->first()/1000);
|
||||
|
||||
QDate date=d2.date();
|
||||
QTime time=d2.time();
|
||||
@ -605,8 +324,9 @@ bool Machine::Save()
|
||||
|
||||
QString fn=path+"/channels.dat";
|
||||
QFile cf(fn);
|
||||
if (cf.open(QIODevice::WriteOnly)) {
|
||||
int i=4;
|
||||
if (!cf.open(QIODevice::WriteOnly)) {
|
||||
qDebug() << "Couldn't write.. Permissions? Hard disk crashing?";
|
||||
return false;
|
||||
}
|
||||
QDataStream out(&cf);
|
||||
out.setVersion(QDataStream::Qt_4_6);
|
||||
@ -717,12 +437,6 @@ Session *Machine::popSaveList()
|
||||
CPAP::CPAP(Profile *p,MachineID id):Machine(p,id)
|
||||
{
|
||||
m_type=MT_CPAP;
|
||||
|
||||
registerChannel(CPAP_Obstructive);
|
||||
registerChannel(CPAP_Hypopnea);
|
||||
registerChannel(CPAP_ClearAirway);
|
||||
registerChannel(CPAP_Snore);
|
||||
registerChannel(CPAP_Leak);
|
||||
}
|
||||
|
||||
CPAP::~CPAP()
|
||||
@ -735,9 +449,6 @@ CPAP::~CPAP()
|
||||
Oximeter::Oximeter(Profile *p,MachineID id):Machine(p,id)
|
||||
{
|
||||
m_type=MT_OXIMETER;
|
||||
registerChannel(OXI_Pulse);
|
||||
registerChannel(OXI_SPO2);
|
||||
registerChannel(OXI_Plethysomogram);
|
||||
}
|
||||
|
||||
Oximeter::~Oximeter()
|
||||
|
@ -91,8 +91,8 @@ public:
|
||||
const MachineID & id() { return m_id; }
|
||||
const QDate & FirstDay() { return firstday; }
|
||||
const QDate & LastDay() { return lastday; }
|
||||
bool hasChannel(ChannelID id) { return m_channels.contains(id) && m_channels[id]; }
|
||||
void registerChannel(ChannelID id,bool b=true) { m_channels[id]=b; }
|
||||
bool hasChannel(QString id) { return m_channels.contains(id) && m_channels[id]; }
|
||||
void registerChannel(QString id,bool b=true) { m_channels[id]=b; }
|
||||
|
||||
protected:
|
||||
QDate firstday,lastday;
|
||||
@ -104,7 +104,7 @@ protected:
|
||||
Profile *profile;
|
||||
bool changed;
|
||||
bool firstsession;
|
||||
QHash<ChannelID,bool> m_channels;
|
||||
QHash<QString,bool> m_channels;
|
||||
};
|
||||
|
||||
class CPAP:public Machine
|
||||
|
@ -1,8 +1,8 @@
|
||||
/********************************************************************
|
||||
/*
|
||||
SleepLib Machine Loader Class Header
|
||||
Copyright (c)2011 Mark Watkins <jedimark@users.sourceforge.net>
|
||||
License: GPL
|
||||
*********************************************************************/
|
||||
*/
|
||||
|
||||
#ifndef MACHINE_COMMON_H
|
||||
#define MACHINE_COMMON_H
|
||||
@ -15,7 +15,7 @@
|
||||
#include <QDebug>
|
||||
using namespace std;
|
||||
|
||||
typedef quint32 ChannelID;
|
||||
typedef QString ChannelID;
|
||||
typedef long MachineID;
|
||||
typedef long SessionID;
|
||||
typedef float EventDataType;
|
||||
@ -32,163 +32,7 @@ const quint16 dbversion=6;
|
||||
|
||||
|
||||
enum MachineType { MT_UNKNOWN=0,MT_CPAP,MT_OXIMETER,MT_SLEEPSTAGE,MT_JOURNAL };
|
||||
enum ChannelType { CT_Unknown=0, CT_Event, CT_Graph, CT_Text, CT_Decimal, CT_Integer, CT_Boolean, CT_DateTime, CT_Date, CT_Time, CT_Timespan, CT_Calculation, CT_Lookup };
|
||||
|
||||
class Channel
|
||||
{
|
||||
protected:
|
||||
ChannelID m_id;
|
||||
MachineType m_mtype;
|
||||
ChannelType m_ctype;
|
||||
QString m_details;
|
||||
QString m_label;
|
||||
QColor m_color;
|
||||
QVector<QVariant> m_option;
|
||||
EventDataType m_min_value;
|
||||
EventDataType m_max_value;
|
||||
public:
|
||||
Channel()
|
||||
:m_id(0),m_mtype(MT_UNKNOWN), m_ctype(CT_Unknown), m_details(""), m_label(""),m_color(Qt::black) {}
|
||||
Channel(ChannelID id, MachineType mtype, ChannelType ctype, QString details, QString label, QColor color)
|
||||
:m_id(id),m_mtype(mtype),m_ctype(ctype),m_details(details),m_label(label),m_color(color) {}
|
||||
ChannelID & id() { return m_id; }
|
||||
MachineType & machinetype() { return m_mtype; }
|
||||
ChannelType & channeltype() { return m_ctype; }
|
||||
QString & details() { return m_details; }
|
||||
QString & label() { return m_label; }
|
||||
|
||||
EventDataType min() { return m_min_value; } // Min value allowed for this channel
|
||||
EventDataType max() { return m_max_value; } // Max value allowed for this channel
|
||||
void setMin(EventDataType v) { m_min_value=v; }
|
||||
void setMax(EventDataType v) { m_max_value=v; }
|
||||
|
||||
// add Option automatically sets the bounds
|
||||
void addOption(QVariant v) { m_option.push_back(v); m_min_value=0; m_max_value=m_option.size(); }
|
||||
int countOptions() { return m_option.count(); }
|
||||
|
||||
QVariant & option(int i) { if (m_option.contains(i)) return m_option[i]; }
|
||||
QVariant & operator [](int i) { if (m_option.contains(i)) return m_option[i]; }
|
||||
|
||||
QString optionString(int i) { if (m_option.contains(i)) return m_option[i].toString(); else return ""; }
|
||||
int optionInt(int i) { if (m_option.contains(i)) return m_option[i].toInt(); else return 0; }
|
||||
EventDataType optionFloat(int i) { if (m_option.contains(i)) return m_option[i].toFloat(); else return 0; }
|
||||
};
|
||||
|
||||
extern QHash<ChannelID, Channel> channel;
|
||||
extern QHash<QString,Channel *> channel_lookup;
|
||||
|
||||
struct ChannelGroup
|
||||
{
|
||||
protected:
|
||||
MachineType m_type;
|
||||
ChannelID m_first;
|
||||
ChannelID m_size;
|
||||
ChannelID m_pos;
|
||||
QHash<ChannelID, Channel *> m_channel;
|
||||
QHash<ChannelType, QHash<ChannelID, Channel *> > m_channelsbytype;
|
||||
QHash<QString, Channel *> m_channel_lookup;
|
||||
public:
|
||||
ChannelID first() { return m_first; }
|
||||
ChannelID count() { return m_pos; }
|
||||
Channel & operator [](ChannelID id) { return *m_channel[id]; }
|
||||
Channel & operator [](QString lookup) { return *m_channel_lookup[lookup]; }
|
||||
QHash<ChannelID, Channel *> & operator [](ChannelType type) { return m_channelsbytype[type]; }
|
||||
|
||||
ChannelID Get(ChannelType ctype,QString description="",QString label="",QString lookup="",QColor color=Qt::black);
|
||||
ChannelGroup();
|
||||
ChannelGroup(MachineType type, ChannelID first, ChannelID reserved=0x200);
|
||||
};
|
||||
extern QHash<QString,ChannelGroup *> channel_group;
|
||||
|
||||
extern ChannelGroup CPAP_CODES;
|
||||
extern ChannelGroup PRS1_CODES;
|
||||
extern ChannelGroup RMS9_CODES;
|
||||
extern ChannelGroup JOURNAL_CODES;
|
||||
extern ChannelGroup ZEO_CODES;
|
||||
extern ChannelGroup OXI_CODES;
|
||||
|
||||
// ******************** IMPORTANT ********************
|
||||
// Add to the end of each group or be eaten by a Grue!
|
||||
// Increment this version number if you mess with this
|
||||
// ******************** IMPORTANT ********************
|
||||
const quint32 ChannelVersionNumber=0;
|
||||
|
||||
const ChannelID EmptyChannel=0;
|
||||
extern ChannelID CPAP_Obstructive,CPAP_Hypopnea,CPAP_Apnea, CPAP_ClearAirway, CPAP_RERA, CPAP_FlowLimit,
|
||||
CPAP_CSR, CPAP_VSnore, CPAP_PressurePulse, CPAP_Mode, CPAP_FlowRate, CPAP_MaskPressure, CPAP_Pressure,
|
||||
CPAP_EPAP, CPAP_IPAP, CPAP_IPAP_Low, CPAP_IPAP_High, CPAP_PressureSupport, CPAP_Snore, CPAP_Leak,
|
||||
CPAP_RespiratoryRate, CPAP_TidalVolume, CPAP_MinuteVentilation, CPAP_PatientTriggeredBreaths,
|
||||
CPAP_FlowLimitGraph, CPAP_TherapyPressure, CPAP_ExpiratoryPressure, CPAP_AHI, CPAP_BrokenSummary,
|
||||
CPAP_BrokenWaveform, CPAP_Pulse, CPAP_SPO2, CPAP_Plethy,CPAP_RespiratoryEvent,CPAP_IE,CPAP_Ti,CPAP_Te;
|
||||
|
||||
extern ChannelID RMS9_PressureReliefType, RMS9_PressureReliefSetting, RMS9_Empty1, RMS9_Empty2;
|
||||
extern ChannelID PRS1_PressureMin,PRS1_PressureMax, PRS1_PressureMinAchieved, PRS1_PressureMaxAchieved,
|
||||
PRS1_PressureAvg, PRS1_Pressure90, PRS1_RampTime, PRS1_RampPressure, PRS1_HumidifierStatus,
|
||||
PRS1_HumidifierSetting, PRS1_PressureReliefType, PRS1_PressureReliefSetting, PRS1_SystemOneResistanceStatus,
|
||||
PRS1_SystemOneResistanceSetting, PRS1_SystemLockStatus, PRS1_HoseDiameter, PRS1_AutoOff, PRS1_AutoOn,
|
||||
PRS1_MaskAlert, PRS1_ShowAHI, PRS1_Unknown00, PRS1_Unknown01, PRS1_Unknown08, PRS1_Unknown09,
|
||||
PRS1_Unknown0A, PRS1_Unknown0B, PRS1_Unknown0C, PRS1_Unknown0E, PRS1_Unknown0F, PRS1_Unknown10,
|
||||
PRS1_Unknown12, PRS1_VSnore2;
|
||||
|
||||
extern ChannelID OXI_Pulse, OXI_SPO2, OXI_Plethysomogram, OXI_PulseChange, OXI_SPO2Drop, OXI_Error, OXI_SignalError;
|
||||
extern ChannelID ZEO_SleepStage, ZEO_Waveform;
|
||||
extern ChannelID JOURNAL_Notes, JOURNAL_Weight;
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
// Be cautious when extending these.. add to the end of each groups to preserve file formats.
|
||||
/*enum ChannelID//:qint16
|
||||
{
|
||||
// General Event Codes
|
||||
MC_UNKNOWN=0,
|
||||
|
||||
CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_ClearAirway, CPAP_RERA, CPAP_VSnore, CPAP_FlowLimit,
|
||||
CPAP_Leak, CPAP_Pressure, CPAP_EAP, CPAP_IAP, CPAP_CSR, CPAP_FlowRate, CPAP_MaskPressure,
|
||||
CPAP_Snore,CPAP_MinuteVentilation, CPAP_RespiratoryRate, CPAP_TidalVolume,CPAP_FlowLimitGraph,
|
||||
CPAP_PatientTriggeredBreaths, CPAP_PS, CPAP_IAPLO, CPAP_IAPHI,
|
||||
|
||||
// General CPAP Summary Information
|
||||
CPAP_PressureMin=0x80, CPAP_PressureMax, CPAP_RampTime, CPAP_RampStartingPressure, CPAP_Mode, CPAP_PressureReliefType,
|
||||
CPAP_PressureReliefSetting, CPAP_HumidifierSetting, CPAP_HumidifierStatus, CPAP_PressureMinAchieved,
|
||||
CPAP_PressureMaxAchieved, CPAP_PressurePercentValue, CPAP_PressurePercentName, CPAP_PressureAverage, CPAP_PressureMedian,
|
||||
CPAP_LeakMedian,CPAP_LeakMinimum,CPAP_LeakMaximum,CPAP_LeakAverage,CPAP_Duration,
|
||||
CPAP_SnoreMinimum, CPAP_SnoreMaximum, CPAP_SnoreAverage, CPAP_SnoreMedian, CPAP_TherapyPressure, CPAP_ExpPressure,
|
||||
|
||||
CPAP_Pressure90,CPAP_Leak90,CPAP_AI, CPAP_HI, CPAP_CAI, CPAP_AHI, CPAP_PBI,
|
||||
|
||||
BIPAP_EAPAverage,BIPAP_IAPAverage,BIPAP_EAPMin,BIPAP_EAPMax,BIPAP_IAPMin,BIPAP_IAPMax,
|
||||
BIPAP_PSAverage,BIPAP_PSMin, BIPAP_PSMax, BIPAP_PS90, BIPAP_EAP90, BIPAP_IAP90,
|
||||
|
||||
CPAP_PTBMin, CPAP_PTBMax, CPAP_PTBAverage, CPAP_PTB90,
|
||||
CPAP_RRMin, CPAP_RRMax, CPAP_RRAverage, CPAP_RR90,
|
||||
CPAP_MVMin, CPAP_MVMax, CPAP_MVAverage, CPAP_MV90,
|
||||
CPAP_TVMin, CPAP_TVMax, CPAP_TVAverage, CPAP_TV90,
|
||||
|
||||
CPAP_BrokenSummary, CPAP_BrokenWaveform,
|
||||
|
||||
// PRS1 Specific Codes
|
||||
PRS1_PressurePulse=0x1000,PRS1_VSnore2,
|
||||
PRS1_Unknown00, PRS1_Unknown01, PRS1_Unknown08, PRS1_Unknown09, PRS1_Unknown0B, PRS1_Unknown0E, PRS1_Unknown10, PRS1_Unknown12,
|
||||
PRS1_SystemLockStatus, PRS1_SystemResistanceStatus, PRS1_SystemResistanceSetting, PRS1_HoseDiameter, PRS1_AutoOff, PRS1_MaskAlert, PRS1_ShowAHI,
|
||||
|
||||
// ASV Unknown Codes
|
||||
PRS1_Unknown0A,PRS1_Unknown0C, PRS1_Unknown0F,
|
||||
|
||||
ResMed_Empty1, ResMed_Empty2,
|
||||
|
||||
// Oximeter Codes
|
||||
OXI_Pulse=0x2000, OXI_SPO2, OXI_Plethy, OXI_Signal2, OXI_SignalGood, OXI_PulseAverage, OXI_PulseMin, OXI_PulseMax, OXI_SPO2Average, OXI_SPO2Min, OXI_SPO2Max,
|
||||
OXI_PulseChange, OXI_SPO2Change,
|
||||
|
||||
ZEO_SleepStage=0x2800, ZEO_Waveform,
|
||||
|
||||
GEN_Weight=0x3000, GEN_Notes, GEN_Glucose, GEN_Calories, GEN_Energy, GEN_Mood, GEN_Excercise, GEN_Reminder
|
||||
|
||||
};*/
|
||||
void InitMapsWithoutAwesomeInitializerLists();
|
||||
//void InitMapsWithoutAwesomeInitializerLists();
|
||||
|
||||
|
||||
enum CPAPMode//:short
|
||||
@ -206,8 +50,65 @@ enum PRTypes//:short
|
||||
//extern map<CPAPMode,QString> CPAPModeNames;
|
||||
|
||||
// These are types supported by wxVariant class. To retain compatability, add to the end of this list only..
|
||||
enum MCDataType//:wxInt8
|
||||
enum MCDataType
|
||||
{ MC_bool=0, MC_int, MC_long, MC_float, MC_double, MC_string, MC_datetime };
|
||||
|
||||
const QString CPAP_IPAP="IPAP";
|
||||
const QString CPAP_EPAP="EPAP";
|
||||
const QString CPAP_Pressure="Pressure";
|
||||
const QString CPAP_PS="PS";
|
||||
const QString PRS1_FlexMode="FlexMode";
|
||||
const QString PRS1_FlexSet="FlexSet";
|
||||
const QString CPAP_Mode="PAPMode";
|
||||
const QString CPAP_BrokenSummary="BrokenSummary";
|
||||
const QString CPAP_PressureMin="PressureMin";
|
||||
const QString CPAP_PressureMax="PressureMin";
|
||||
const QString CPAP_RampTime="RampTime";
|
||||
const QString CPAP_RampPressure="RampPressure";
|
||||
const QString CPAP_Obstructive="Obstructive";
|
||||
const QString CPAP_Hypopnea="Hypopnea";
|
||||
const QString CPAP_ClearAirway="ClearAirway";
|
||||
const QString CPAP_Apnea="Apnea";
|
||||
const QString CPAP_CSR="CSR";
|
||||
const QString CPAP_VSnore="VSnore";
|
||||
const QString CPAP_VSnore2="VSnore2";
|
||||
const QString CPAP_RERA="RERA";
|
||||
const QString CPAP_PressurePulse="PressurePulse";
|
||||
const QString CPAP_FlowLimit="FlowLimit";
|
||||
const QString CPAP_FlowRate="FlowRate";
|
||||
const QString CPAP_MaskPressure="MaskPressure";
|
||||
const QString CPAP_MaskPressureHi="MaskPressureHi";
|
||||
const QString CPAP_RespEvent="RespEvent";
|
||||
const QString CPAP_Snore="Snore";
|
||||
const QString CPAP_MinuteVent="MinuteVent";
|
||||
const QString CPAP_RespRate="RespRate";
|
||||
const QString CPAP_TidalVolume="TidalVolume";
|
||||
const QString CPAP_PTB="PTB";
|
||||
const QString CPAP_Leak="Leak";
|
||||
const QString CPAP_FLG="FLG";
|
||||
const QString CPAP_IE="IE";
|
||||
const QString CPAP_Te="Te";
|
||||
const QString CPAP_Ti="Ti";
|
||||
const QString RMS9_E01="RMS9_E01";
|
||||
const QString RMS9_E02="RMS9_E02";
|
||||
const QString PRS1_00="PRS1_00";
|
||||
const QString PRS1_01="PRS1_01";
|
||||
const QString PRS1_08="PRS1_08";
|
||||
const QString PRS1_0A="PRS1_0A";
|
||||
const QString PRS1_0B="PRS1_0B";
|
||||
const QString PRS1_0C="PRS1_0C";
|
||||
const QString PRS1_0E="PRS1_0E";
|
||||
const QString PRS1_0F="PRS1_0F";
|
||||
const QString PRS1_10="PRS1_10";
|
||||
const QString PRS1_12="PRS1_12";
|
||||
|
||||
const QString OXI_Pulse="Pulse";
|
||||
const QString OXI_SPO2="SPO2";
|
||||
const QString OXI_PulseChange="PulseChange";
|
||||
const QString OXI_SPO2Drop="SPO2Drop";
|
||||
const QString OXI_Plethy="Plethy";
|
||||
|
||||
const QString CPAP_AHI="AHI";
|
||||
|
||||
|
||||
#endif // MACHINE_COMMON_H
|
||||
|
@ -358,8 +358,7 @@ Profile *Get()
|
||||
*/
|
||||
void Scan()
|
||||
{
|
||||
InitMapsWithoutAwesomeInitializerLists();
|
||||
|
||||
//InitMapsWithoutAwesomeInitializerLists();
|
||||
p_pref=new Preferences("Preferences");
|
||||
p_layout=new Preferences("Layout");
|
||||
|
||||
|
251
SleepLib/schema.cpp
Normal file
251
SleepLib/schema.cpp
Normal file
@ -0,0 +1,251 @@
|
||||
#include <QFile>
|
||||
#include <QDebug>
|
||||
#include <QDomDocument>
|
||||
#include <QDomElement>
|
||||
#include <QDomNode>
|
||||
#include "schema.h"
|
||||
|
||||
|
||||
|
||||
namespace schema {
|
||||
|
||||
ChannelList channel;
|
||||
Channel EmptyChannel;
|
||||
|
||||
QHash<QString,ChanType> ChanTypes;
|
||||
QHash<QString,DataType> DataTypes;
|
||||
QHash<QString,ScopeType> Scopes;
|
||||
|
||||
bool schema_initialized=false;
|
||||
|
||||
void init()
|
||||
{
|
||||
if (schema_initialized) return;
|
||||
schema_initialized=true;
|
||||
|
||||
EmptyChannel=Channel(0,DATA,DAY,"Empty","Empty Channel","Empty","");
|
||||
|
||||
ChanTypes["data"]=DATA;
|
||||
//Types["waveform"]=WAVEFORM;
|
||||
ChanTypes["setting"]=SETTING;
|
||||
|
||||
Scopes["session"]=SESSION;
|
||||
Scopes["day"]=DAY;
|
||||
Scopes["machine"]=MACHINE;
|
||||
Scopes["global"]=GLOBAL;
|
||||
|
||||
DataTypes[""]=DEFAULT;
|
||||
DataTypes["bool"]=BOOL;
|
||||
DataTypes["double"]=DOUBLE;
|
||||
DataTypes["integer"]=INTEGER;
|
||||
DataTypes["string"]=STRING;
|
||||
DataTypes["richtext"]=RICHTEXT;
|
||||
DataTypes["date"]=DATE;
|
||||
DataTypes["datetime"]=DATETIME;
|
||||
DataTypes["time"]=TIME;
|
||||
|
||||
schema::channel.Load(":/docs/channels.xml");
|
||||
|
||||
}
|
||||
|
||||
Channel::Channel(int id, ChanType type, ScopeType scope, QString name, QString description, QString label, QString unit, DataType datatype, QColor color, int link):
|
||||
m_id(id),
|
||||
m_type(type),
|
||||
m_scope(scope),
|
||||
m_name(name),
|
||||
m_description(description),
|
||||
m_label(label),
|
||||
m_unit(unit),
|
||||
m_datatype(datatype),
|
||||
m_defaultcolor(color),
|
||||
m_link(link)
|
||||
{
|
||||
}
|
||||
bool Channel::isNull()
|
||||
{
|
||||
return (this==&EmptyChannel);
|
||||
}
|
||||
|
||||
ChannelList::ChannelList()
|
||||
:m_doctype("channels")
|
||||
{
|
||||
}
|
||||
ChannelList::~ChannelList()
|
||||
{
|
||||
for (QHash<int,Channel *>::iterator i=channels.begin();i!=channels.end();i++) {
|
||||
delete i.value();
|
||||
}
|
||||
}
|
||||
bool ChannelList::Load(QString filename)
|
||||
{
|
||||
QDomDocument doc(m_doctype);
|
||||
QFile file(filename);
|
||||
qDebug() << "Opening " << filename;
|
||||
if (!file.open(QIODevice::ReadOnly)) {
|
||||
qWarning() << "Could not open" << filename;
|
||||
return false;
|
||||
}
|
||||
QString errorMsg;
|
||||
int errorLine;
|
||||
if (!doc.setContent(&file,false,&errorMsg,&errorLine)) {
|
||||
qWarning() << "Invalid XML Content in" << filename;
|
||||
qWarning() << "Error line" << errorLine <<":" << errorMsg;
|
||||
return false;
|
||||
}
|
||||
file.close();
|
||||
|
||||
|
||||
QDomElement root=doc.documentElement();
|
||||
if (root.tagName().toLower() != "channels") {
|
||||
return false;
|
||||
}
|
||||
QString language=root.attribute("language","en");
|
||||
QString version=root.attribute("version","");
|
||||
|
||||
if (version.isEmpty()) {
|
||||
qWarning() << "No Version Field in" << m_doctype << "Schema, assuming 1.0" << filename;
|
||||
version="1.0";
|
||||
}
|
||||
|
||||
qDebug() << "Processing xml file:" << m_doctype << language << version;
|
||||
QDomNodeList grp=root.elementsByTagName("group");
|
||||
QDomNode node,n,ch;
|
||||
QDomElement e;
|
||||
|
||||
bool ok;
|
||||
int id,linkid;
|
||||
QString chantype,scopestr,typestr,name,group,idtxt,details,label,unit,datatypestr,defcolor,link;
|
||||
ChanType type;
|
||||
DataType datatype;
|
||||
Channel *chan;
|
||||
QColor color;
|
||||
bool multi;
|
||||
ScopeType scope;
|
||||
int line;
|
||||
for (int i=0;i<grp.size();i++) {
|
||||
node=grp.at(i);
|
||||
group=node.toElement().attribute("name");
|
||||
//qDebug() << "Group Name" << group;
|
||||
// Why do I have to skip the first node here? (shows up empty)
|
||||
n=node.firstChild().nextSibling();
|
||||
|
||||
while (!n.isNull()) {
|
||||
line=n.lineNumber();
|
||||
e=n.toElement();
|
||||
|
||||
if (e.nodeName().toLower()!="channel") {
|
||||
qWarning() << "Ignoring unrecognized schema type "<< e.nodeName() << "in" << filename << "line" << line;
|
||||
continue;
|
||||
}
|
||||
|
||||
ch=n.firstChild();
|
||||
n=n.nextSibling();
|
||||
|
||||
idtxt=e.attribute("id");
|
||||
id=idtxt.toInt(&ok,16);
|
||||
if (!ok) {
|
||||
qWarning() << "Dodgy ID number "<< e.nodeName() << "in" << filename << "line" << line;
|
||||
continue;
|
||||
}
|
||||
|
||||
chantype=e.attribute("class","data").toLower();
|
||||
if (!ChanTypes.contains(chantype)) {
|
||||
qWarning() << "Dodgy class "<< chantype << "in" << filename << "line" << line;
|
||||
continue;
|
||||
}
|
||||
type=ChanTypes[chantype];
|
||||
|
||||
scopestr=e.attribute("scope","session");
|
||||
if (scopestr.at(0)==QChar('!')) {
|
||||
scopestr=scopestr.mid(1);
|
||||
multi=true;
|
||||
} multi=false;
|
||||
if (!Scopes.contains(scopestr)) {
|
||||
qWarning() << "Dodgy Scope "<< scopestr << "in" << filename << "line" << line;
|
||||
continue;
|
||||
}
|
||||
scope=Scopes[scopestr];
|
||||
name=e.attribute("name","").toLower();
|
||||
details=e.attribute("details","");
|
||||
label=e.attribute("label","");
|
||||
|
||||
if (name.isEmpty() || details.isEmpty() || label.isEmpty()) {
|
||||
qWarning() << "Missing name,details or label attribute in" << filename << "line" << line;
|
||||
continue;
|
||||
}
|
||||
unit=e.attribute("unit");
|
||||
|
||||
defcolor=e.attribute("color","black");
|
||||
color=QColor(defcolor);
|
||||
if (!color.isValid()) {
|
||||
qWarning() << "Invalid Color "<< defcolor<< "in" << filename << "line" << line;
|
||||
color=Qt::black;
|
||||
}
|
||||
|
||||
datatypestr=e.attribute("type","").toLower();
|
||||
|
||||
link=e.attribute("link","");
|
||||
if (!link.isEmpty()) {
|
||||
linkid=link.toInt(&ok,16);
|
||||
if (!ok) {
|
||||
qWarning() << "Dodgy Link ID number "<< e.nodeName() << "in" << filename << " line" << line;
|
||||
}
|
||||
} else linkid=0;
|
||||
|
||||
if (DataTypes.contains(datatypestr)) {
|
||||
datatype=DataTypes[typestr];
|
||||
} else {
|
||||
qWarning() << "Ignoring unrecognized schema datatype in" << filename <<"line" << line;
|
||||
continue;
|
||||
}
|
||||
|
||||
if (channels.contains(id)) {
|
||||
qWarning() << "Schema already contains id" << id << "in" << filename <<"line" << line;
|
||||
continue;
|
||||
}
|
||||
if (names.contains(name)) {
|
||||
qWarning() << "Schema already contains name" << name << "in" << filename <<"line" << line;
|
||||
continue;
|
||||
}
|
||||
chan=new Channel(id,type,scope,name,details,label,unit,datatype,color,linkid);
|
||||
channels[id]=chan;
|
||||
names[name]=chan;
|
||||
groups[group][name]=chan;
|
||||
if (linkid>0) {
|
||||
if (channels.contains(linkid)) {
|
||||
Channel *it=channels[id];
|
||||
chan->m_links.push_back(it);
|
||||
} else {
|
||||
qWarning() << "Linked channel must be defined first in" << filename <<"line" << line;
|
||||
}
|
||||
}
|
||||
|
||||
// process children
|
||||
while(!ch.isNull()) {
|
||||
e=ch.toElement();
|
||||
QString sub=ch.nodeName().toLower();
|
||||
QString id2str,name2str;
|
||||
int id2;
|
||||
if (sub=="option") {
|
||||
id2str=e.attribute("id");
|
||||
id2=id2str.toInt(&ok,10);
|
||||
name2str=e.attribute("value");
|
||||
//qDebug() << sub << id2 << name2str;
|
||||
chan->addOption(id2,name2str);
|
||||
} else if (sub=="color") {
|
||||
}
|
||||
ch=ch.nextSibling();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
bool ChannelList::Save(QString filename)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
typedef schema::Channel * ChannelID;
|
93
SleepLib/schema.h
Normal file
93
SleepLib/schema.h
Normal file
@ -0,0 +1,93 @@
|
||||
#ifndef SCHEMA_H
|
||||
#define SCHEMA_H
|
||||
|
||||
#include <QColor>
|
||||
#include <QHash>
|
||||
#include <QVariant>
|
||||
#include <QString>
|
||||
|
||||
namespace schema {
|
||||
|
||||
enum Function {
|
||||
NONE=0, AVG, WAVG, MIN, MAX, SUM, CNT, P90, CPH, SPH, HOURS, SET
|
||||
};
|
||||
|
||||
enum ChanType {
|
||||
DATA=0, SETTING
|
||||
};
|
||||
|
||||
enum DataType {
|
||||
DEFAULT=0, INTEGER, BOOL, DOUBLE, STRING, RICHTEXT, DATE, TIME, DATETIME
|
||||
};
|
||||
enum ScopeType {
|
||||
GLOBAL=0, MACHINE, DAY, SESSION
|
||||
};
|
||||
class Channel;
|
||||
extern Channel EmptyChannel;
|
||||
|
||||
// this is really a channel definition.
|
||||
class Channel
|
||||
{
|
||||
public:
|
||||
Channel() { m_id=0; }
|
||||
Channel(int id, ChanType type, ScopeType scope, QString name, QString description, QString label, QString unit,DataType datatype=DEFAULT, QColor=Qt::black, int link=0);
|
||||
void addColor(Function f, QColor color) { m_colors[f]=color; }
|
||||
void addOption(int i, QString option) { m_options[i]=option; }
|
||||
|
||||
const int & id() { return m_id; }
|
||||
const ChanType & type() { return m_type; }
|
||||
const QString & name() { return m_name; }
|
||||
const QString & description() { return m_description; }
|
||||
const QString & label() { return m_label; }
|
||||
QHash<int,QString> m_options;
|
||||
QHash<Function,QColor> m_colors;
|
||||
QList<Channel *> m_links; // better versions of this data type
|
||||
bool isNull();
|
||||
protected:
|
||||
int m_id;
|
||||
ChanType m_type;
|
||||
ScopeType m_scope;
|
||||
QString m_name;
|
||||
QString m_description;
|
||||
QString m_label;
|
||||
QString m_unit;
|
||||
DataType m_datatype;
|
||||
QColor m_defaultcolor;
|
||||
|
||||
int m_link;
|
||||
};
|
||||
|
||||
class ChannelList
|
||||
{
|
||||
public:
|
||||
ChannelList();
|
||||
virtual ~ChannelList();
|
||||
bool Load(QString filename);
|
||||
bool Save(QString filename);
|
||||
Channel & operator[](int i) {
|
||||
if (channels.contains(i))
|
||||
return *channels[i];
|
||||
else
|
||||
return EmptyChannel;
|
||||
}
|
||||
Channel & operator[](QString name) {
|
||||
QString tmp=name.toLower();
|
||||
if (names.contains(tmp))
|
||||
return *names[tmp];
|
||||
else
|
||||
return EmptyChannel;
|
||||
}
|
||||
|
||||
QHash<int,Channel *> channels;
|
||||
QHash<QString,Channel *> names;
|
||||
QHash<QString,QHash<QString,Channel *> > groups;
|
||||
QString m_doctype;
|
||||
};
|
||||
|
||||
extern ChannelList channel;
|
||||
|
||||
void init();
|
||||
|
||||
} // namespace
|
||||
|
||||
#endif // SCHEMA_H
|
@ -254,7 +254,7 @@ bool Session::StoreEvents(QString filename)
|
||||
QHash<ChannelID,QVector<EventList *> >::iterator i;
|
||||
|
||||
for (i=eventlist.begin(); i!=eventlist.end(); i++) {
|
||||
out << (qint16)i.key(); // ChannelID
|
||||
out << i.key(); // ChannelID
|
||||
out << (qint16)i.value().size();
|
||||
for (int j=0;j<i.value().size();j++) {
|
||||
EventList &e=*i.value()[j];
|
||||
@ -350,8 +350,7 @@ bool Session::LoadEvents(QString filename)
|
||||
QVector<qint16> sizevec;
|
||||
QString dim;
|
||||
for (int i=0;i<mcsize;i++) {
|
||||
in >> t16;
|
||||
code=(ChannelID)t16;
|
||||
in >> code;
|
||||
mcorder.push_back(code);
|
||||
in >> size2;
|
||||
sizevec.push_back(size2);
|
||||
@ -367,10 +366,10 @@ bool Session::LoadEvents(QString filename)
|
||||
in >> mn;
|
||||
in >> mx;
|
||||
in >> dim;
|
||||
EventList *elist=new EventList(code,elt,gain,offset,mn,mx,rate);
|
||||
EventList *elist=AddEventList(code,elt,gain,offset,mn,mx,rate);
|
||||
elist->setDimension(dim);
|
||||
|
||||
eventlist[code].push_back(elist);
|
||||
//eventlist[code].push_back(elist);
|
||||
elist->m_count=evcount;
|
||||
elist->m_first=ts1;
|
||||
elist->m_last=ts2;
|
||||
@ -411,7 +410,7 @@ void Session::UpdateSummaries()
|
||||
QHash<ChannelID,QVector<EventList *> >::iterator c;
|
||||
for (c=eventlist.begin();c!=eventlist.end();c++) {
|
||||
id=c.key();
|
||||
if ((channel[id].channeltype()==CT_Event) || (channel[id].channeltype()==CT_Graph)) {
|
||||
if (schema::channel[id].type()==schema::DATA) {
|
||||
//sum(id); // avg calculates this and cnt.
|
||||
min(id);
|
||||
max(id);
|
||||
@ -540,7 +539,7 @@ qint64 Session::last(ChannelID id)
|
||||
m_lastchan[id]=max;
|
||||
return max;
|
||||
}
|
||||
bool Session::channelExists(ChannelID id)
|
||||
/*bool Session::channelExists(ChannelID id)
|
||||
{
|
||||
if (s_events_loaded) {
|
||||
QHash<ChannelID,QVector<EventList *> >::iterator j=eventlist.find(id);
|
||||
@ -552,7 +551,7 @@ bool Session::channelExists(ChannelID id)
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
}*/
|
||||
|
||||
int Session::count(ChannelID id)
|
||||
{
|
||||
@ -763,3 +762,15 @@ EventDataType Session::wavg(ChannelID id)
|
||||
return v;
|
||||
}
|
||||
|
||||
EventList * Session::AddEventList(QString chan, EventListType et,EventDataType gain,EventDataType offset,EventDataType min, EventDataType max,EventDataType rate)
|
||||
{
|
||||
schema::Channel * channel=&schema::channel[chan];
|
||||
if (!channel) {
|
||||
qWarning() << "Channel" << chan << "does not exist!";
|
||||
//return NULL;
|
||||
}
|
||||
EventList * el=new EventList(et,gain,offset,min,max,rate);
|
||||
eventlist[chan].push_back(el);
|
||||
s_machine->registerChannel(chan);
|
||||
return el;
|
||||
}
|
||||
|
@ -13,8 +13,9 @@
|
||||
#include <QVector>
|
||||
|
||||
#include "SleepLib/machine.h"
|
||||
#include "SleepLib/schema.h"
|
||||
#include "SleepLib/event.h"
|
||||
class EventList;
|
||||
//class EventList;
|
||||
class Machine;
|
||||
const quint32 magic=0xC73216AB;
|
||||
|
||||
@ -117,7 +118,7 @@ public:
|
||||
|
||||
EventDataType percentile(ChannelID id,EventDataType percentile);
|
||||
|
||||
bool channelExists(ChannelID id);
|
||||
bool channelExists(QString name) { return (schema::channel.names.contains(name));}
|
||||
|
||||
bool IsLoneSession() { return s_lonesession; }
|
||||
void SetLoneSession(bool b) { s_lonesession=b; }
|
||||
@ -130,6 +131,7 @@ public:
|
||||
qint64 last(ChannelID code);
|
||||
|
||||
void UpdateSummaries();
|
||||
EventList * AddEventList(QString chan, EventListType et, EventDataType gain=1.0, EventDataType offset=0.0, EventDataType min=0.0, EventDataType max=0.0, EventDataType rate=0.0);
|
||||
Machine * machine() { return s_machine; }
|
||||
protected:
|
||||
SessionID s_session;
|
||||
|
@ -46,14 +46,15 @@ SOURCES += main.cpp\
|
||||
Graphs/gYAxis.cpp \
|
||||
Graphs/gFlagsLine.cpp \
|
||||
Graphs/glcommon.cpp \
|
||||
Graphs/gBarChart.cpp \
|
||||
Graphs/gSegmentChart.cpp \
|
||||
Graphs/gSessionTime.cpp \
|
||||
qextserialport/qextserialport.cpp \
|
||||
preferencesdialog.cpp \
|
||||
Graphs/gGraphView.cpp \
|
||||
Graphs/gStatsLine.cpp \
|
||||
report.cpp
|
||||
report.cpp \
|
||||
Graphs/gSummaryChart.cpp \
|
||||
SleepLib/schema.cpp
|
||||
|
||||
unix:SOURCES += qextserialport/posix_qextserialport.cpp
|
||||
unix:!macx:SOURCES += qextserialport/qextserialenumerator_unix.cpp
|
||||
@ -93,7 +94,6 @@ HEADERS += \
|
||||
Graphs/gYAxis.h \
|
||||
Graphs/gFlagsLine.h \
|
||||
Graphs/glcommon.h \
|
||||
Graphs/gBarChart.h \
|
||||
Graphs/gSegmentChart.h\
|
||||
Graphs/gSessionTime.h \
|
||||
SleepLib/loader_plugins/resmed_loader.h \
|
||||
@ -103,7 +103,9 @@ HEADERS += \
|
||||
preferencesdialog.h \
|
||||
Graphs/gGraphView.h \
|
||||
Graphs/gStatsLine.h \
|
||||
report.h
|
||||
report.h \
|
||||
Graphs/gSummaryChart.h \
|
||||
SleepLib/schema.h
|
||||
|
||||
|
||||
FORMS += \
|
||||
@ -120,4 +122,7 @@ RESOURCES += \
|
||||
OTHER_FILES += \
|
||||
docs/index.html \
|
||||
docs/usage.html \
|
||||
docs/template_overview.sht
|
||||
docs/template_overview.sht \
|
||||
docs/schema.xml \
|
||||
docs/graphs.xml \
|
||||
docs/channels.xml
|
||||
|
58
daily.cpp
58
daily.cpp
@ -125,11 +125,10 @@ Daily::Daily(QWidget *parent,Profile * _profile,gGraphView * shared, MainWindow
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Hypopnea,QColor("blue"),"H")));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_PressurePulse,QColor("red"),"PR",FT_Dot)));
|
||||
//FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Pressure,QColor("white"),"P",FT_Dot)));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_Unknown0B,QColor("blue"),"0B",FT_Dot)));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_Unknown10,QColor("orange"),"10",FT_Dot)));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_Unknown0E,QColor("dark red"),"0E",FT_Dot)));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0B,QColor("blue"),"0B",FT_Dot)));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_10,QColor("orange"),"10",FT_Dot)));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0E,QColor("dark red"),"0E",FT_Dot)));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_RERA,QColor("gold"),"RE")));
|
||||
//FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Unknown0E,QColor("dark green"),"U0E")));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Apnea,QColor("dark green"),"A")));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_VSnore,QColor("red"),"VS")));
|
||||
FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_FlowLimit,QColor("black"),"FL")));
|
||||
@ -160,24 +159,25 @@ Daily::Daily(QWidget *parent,Profile * _profile,gGraphView * shared, MainWindow
|
||||
PRD->AddLayer(AddCPAP(new gLineChart(CPAP_Pressure,QColor("dark green"),true)));
|
||||
PRD->AddLayer(AddCPAP(new gLineChart(CPAP_EPAP,Qt::blue,true)));
|
||||
PRD->AddLayer(AddCPAP(new gLineChart(CPAP_IPAP,Qt::red,true)));
|
||||
|
||||
LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_Leak,Qt::darkYellow,true)));
|
||||
SNORE->AddLayer(AddCPAP(new gLineChart(CPAP_Snore,Qt::darkGray,true)));
|
||||
|
||||
PTB->AddLayer(AddCPAP(new gLineChart(CPAP_PatientTriggeredBreaths,Qt::gray,true)));
|
||||
PTB->AddLayer(AddCPAP(new gLineChart(CPAP_PTB,Qt::gray,true)));
|
||||
MP->AddLayer(AddCPAP(new gLineChart(CPAP_MaskPressure,Qt::blue,false)));
|
||||
RR->AddLayer(AddCPAP(new gLineChart(CPAP_RespiratoryRate,Qt::darkMagenta,true)));
|
||||
MV->AddLayer(AddCPAP(new gLineChart(CPAP_MinuteVentilation,Qt::darkCyan,true)));
|
||||
RR->AddLayer(AddCPAP(new gLineChart(CPAP_RespRate,Qt::darkMagenta,true)));
|
||||
MV->AddLayer(AddCPAP(new gLineChart(CPAP_MinuteVent,Qt::darkCyan,true)));
|
||||
TV->AddLayer(AddCPAP(new gLineChart(CPAP_TidalVolume,Qt::magenta,true)));
|
||||
FLG->AddLayer(AddCPAP(new gLineChart(CPAP_FlowLimitGraph,Qt::darkBlue,true)));
|
||||
FLG->AddLayer(AddCPAP(new gLineChart(CPAP_FLG,Qt::darkBlue,true)));
|
||||
//RE->AddLayer(AddCPAP(new gLineChart(CPAP_RespiratoryEvent,Qt::magenta,true)));
|
||||
IE->AddLayer(AddCPAP(new gLineChart(CPAP_IE,Qt::darkRed,true)));
|
||||
TE->AddLayer(AddCPAP(new gLineChart(CPAP_Te,Qt::darkGreen,true)));
|
||||
TI->AddLayer(AddCPAP(new gLineChart(CPAP_Ti,Qt::darkBlue,true)));
|
||||
INTPULSE->AddLayer(AddCPAP(new gLineChart(CPAP_Pulse,Qt::red,true)));
|
||||
INTSPO2->AddLayer(AddCPAP(new gLineChart(CPAP_SPO2,Qt::blue,true)));
|
||||
INTPULSE->AddLayer(AddCPAP(new gLineChart(OXI_Pulse,Qt::red,true)));
|
||||
INTSPO2->AddLayer(AddCPAP(new gLineChart(OXI_SPO2,Qt::blue,true)));
|
||||
PULSE->AddLayer(AddOXI(new gLineChart(OXI_Pulse,Qt::red,true)));
|
||||
SPO2->AddLayer(AddOXI(new gLineChart(OXI_SPO2,Qt::blue,true)));
|
||||
PLETHY->AddLayer(AddOXI(new gLineChart(OXI_Plethysomogram,Qt::darkBlue,false)));
|
||||
PLETHY->AddLayer(AddOXI(new gLineChart(OXI_Plethy,Qt::darkBlue,false)));
|
||||
|
||||
for (int i=0;i<ng;i++){
|
||||
graphs[i]->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin);
|
||||
@ -607,7 +607,7 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day)
|
||||
if ((code!=CPAP_Obstructive)
|
||||
&& (code!=CPAP_Hypopnea)
|
||||
&& (code!=CPAP_Apnea)
|
||||
&& (code!=PRS1_Unknown0B)
|
||||
&& (code!=PRS1_0B)
|
||||
&& (code!=CPAP_ClearAirway)
|
||||
&& (code!=CPAP_CSR)
|
||||
&& (code!=CPAP_RERA)
|
||||
@ -618,9 +618,9 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day)
|
||||
if (mcroot.find(code)==mcroot.end()) {
|
||||
int cnt=day->count(code);
|
||||
total_events+=cnt;
|
||||
QString st=channel[m.key()].details();
|
||||
QString st=schema::channel[m.key()].description();
|
||||
if (st.isEmpty()) {
|
||||
st="Fixme "+QString::number((int)code);
|
||||
st="Fixme "+code;
|
||||
}
|
||||
st+=" ("+QString::number(cnt)+" event"+((cnt>1)?"s":"")+")";
|
||||
QStringList l(st);
|
||||
@ -772,13 +772,13 @@ void Daily::Load(QDate date)
|
||||
QString a;
|
||||
if (cpap) {
|
||||
mode=(CPAPMode)cpap->settings_max(CPAP_Mode);
|
||||
pr=(PRTypes)cpap->settings_max(PRS1_PressureReliefType);
|
||||
pr=(PRTypes)cpap->settings_max(PRS1_FlexMode);
|
||||
if (pr==PR_NONE)
|
||||
epr=tr(" No Pressure Relief");
|
||||
else {
|
||||
epr=channel[PRS1_PressureReliefSetting].optionString(pr)+QString(" x%1").arg((int)cpap->settings_max(PRS1_PressureReliefSetting));
|
||||
//epr=schema::channel[PRS1_FlexSet].optionString(pr)+QString(" x%1").arg((int)cpap->settings_max(PRS1_FlexSet));
|
||||
}
|
||||
modestr=channel[CPAP_Mode].optionString(mode);
|
||||
modestr=schema::channel[CPAP_Mode].m_options[mode];
|
||||
|
||||
float ahi=(cpap->count(CPAP_Obstructive)+cpap->count(CPAP_Hypopnea)+cpap->count(CPAP_ClearAirway)+cpap->count(CPAP_Apnea))/cpap->hours();
|
||||
float csr=(100.0/cpap->hours())*(cpap->sum(CPAP_CSR)/3600.0);
|
||||
@ -870,24 +870,24 @@ void Daily::Load(QDate date)
|
||||
|
||||
html+=("<tr><td> </td><td><b>Min</b></td><td><b>Avg</b></td><td><b>90%</b></td><td><b>Max</b></td></tr>");
|
||||
ChannelID chans[]={
|
||||
CPAP_Pressure,CPAP_EPAP,CPAP_IPAP,CPAP_PressureSupport,CPAP_PatientTriggeredBreaths,
|
||||
CPAP_MinuteVentilation,CPAP_RespiratoryRate,CPAP_RespiratoryEvent,CPAP_FlowLimitGraph,
|
||||
CPAP_Pressure,CPAP_EPAP,CPAP_IPAP,CPAP_PS,CPAP_PTB,
|
||||
CPAP_MinuteVent,CPAP_RespRate,CPAP_RespEvent,CPAP_FLG,
|
||||
CPAP_Leak,CPAP_Snore,CPAP_IE,CPAP_Ti,CPAP_Te,CPAP_TidalVolume,
|
||||
CPAP_Pulse,CPAP_SPO2,OXI_Pulse,OXI_SPO2
|
||||
OXI_Pulse,OXI_SPO2
|
||||
};
|
||||
int numchans=sizeof(chans)/sizeof(ChannelID);
|
||||
for (int i=0;i<numchans;i++) {
|
||||
ChannelID code=chans[i];
|
||||
if (cpap && cpap->channelExists(code)) {
|
||||
html+="<tr><td align=left>"+channel[code].label();
|
||||
if (cpap && cpap->channelHasData(code)) {
|
||||
html+="<tr><td align=left>"+schema::channel[code].label();
|
||||
html+="</td><td>"+a.sprintf("%.2f",cpap->min(code));
|
||||
html+="</td><td>"+a.sprintf("%.2f",cpap->wavg(code));
|
||||
html+="</td><td>"+a.sprintf("%.2f",cpap->p90(code));
|
||||
html+="</td><td>"+a.sprintf("%.2f",cpap->max(code));
|
||||
html+="</td><tr>";
|
||||
}
|
||||
if (oxi && oxi->channelExists(code)) {
|
||||
html+="<tr><td align=left>"+channel[code].label();
|
||||
if (oxi && oxi->channelHasData(code)) {
|
||||
html+="<tr><td align=left>"+schema::channel[code].label();
|
||||
html+="</td><td>"+a.sprintf("%.2f",oxi->min(code));
|
||||
html+="</td><td>"+a.sprintf("%.2f",oxi->wavg(code));
|
||||
html+="</td><td>"+a.sprintf("%.2f",oxi->p90(code));
|
||||
@ -946,7 +946,7 @@ void Daily::Load(QDate date)
|
||||
for (QVector<Session *>::iterator s=cpap->begin();s!=cpap->end();s++) {
|
||||
fd=QDateTime::fromTime_t((*s)->first()/1000L);
|
||||
ld=QDateTime::fromTime_t((*s)->last()/1000L);
|
||||
QHash<ChannelID,QVariant>::iterator i=(*s)->settings.find(CPAP_BrokenWaveform);
|
||||
QHash<ChannelID,QVariant>::iterator i=(*s)->settings.find("BrokenWaveform");
|
||||
if ((i!=(*s)->settings.end()) && i.value().toBool()) corrupted_waveform=true;
|
||||
tmp.sprintf(("<tr><td align=center>%08i</td><td align=center>"+fd.date().toString(Qt::SystemLocaleShortDate)+"</td><td align=center>"+fd.toString("HH:mm ")+"</td><td align=center>"+ld.toString("HH:mm")+"</td></tr>").toLatin1(),(*s)->session());
|
||||
html+=tmp;
|
||||
@ -960,16 +960,16 @@ void Daily::Load(QDate date)
|
||||
|
||||
ui->webView->setHtml(html);
|
||||
|
||||
ui->JournalNotes->clear();
|
||||
/*ui->JournalNotes->clear();
|
||||
Session *journal=GetJournalSession(date);
|
||||
if (journal) {
|
||||
ui->JournalNotes->setHtml(journal->settings[JOURNAL_Notes].toString());
|
||||
}
|
||||
}*/
|
||||
|
||||
}
|
||||
void Daily::Unload(QDate date)
|
||||
{
|
||||
Session *journal=GetJournalSession(date);
|
||||
/*Session *journal=GetJournalSession(date);
|
||||
if (!ui->JournalNotes->toPlainText().isEmpty()) {
|
||||
QString jhtml=ui->JournalNotes->toHtml();
|
||||
if (journal) {
|
||||
@ -988,7 +988,7 @@ void Daily::Unload(QDate date)
|
||||
if (journal) {
|
||||
Machine *jm=profile->GetMachine(MT_JOURNAL);
|
||||
if (jm) jm->SaveSession(journal);
|
||||
}
|
||||
} */
|
||||
UpdateCalendarDay(date);
|
||||
}
|
||||
|
||||
|
2
daily.h
2
daily.h
@ -16,7 +16,7 @@
|
||||
#include <QLabel>
|
||||
#include <QtOpenGL/QGLContext>
|
||||
#include <QScrollBar>
|
||||
#include "Graphs/gBarChart.h"
|
||||
#include "Graphs/gSummaryChart.h"
|
||||
|
||||
#include <SleepLib/profiles.h>
|
||||
#include "mainwindow.h"
|
||||
|
195
docs/channels.xml
Normal file
195
docs/channels.xml
Normal file
@ -0,0 +1,195 @@
|
||||
<!DOCTYPE Channels>
|
||||
<!-- Copy this to SleepApp folder on app start once completed.
|
||||
Metric units please.. A conversion system will deal with the other measurement systems
|
||||
English only.. A different translation table will be used..
|
||||
It's only details & label that will be translated. name is used internally.
|
||||
One id code per item
|
||||
-->
|
||||
<channels language="en" version="1.0">
|
||||
<group name="CPAP">
|
||||
<!-- Channel List -->
|
||||
<channel id="0x1000" class="data" name="CSR" details="Cheyne Stokes Respiration" label="CSR" color="light green"/>
|
||||
<channel id="0x1001" class="data" name="ClearAirway" details="Clear Airway Apnea" label="CA" color="purple"/>
|
||||
<channel id="0x1002" class="data" name="Obstructive" details="Obstructive Apnea" label="O" color="#40c0ff"/>
|
||||
<channel id="0x1003" class="data" name="Hypopnea" details="Hypopnea" label="H" color="blue"/>
|
||||
<channel id="0x1004" class="data" name="Apnea" details="Unspecified Apnea" label="A" color="dark green"/>
|
||||
<channel id="0x1005" class="data" name="FlowLimit" details="Flow Limitation" label="FL" color="dark grey"/>
|
||||
<channel id="0x1006" class="data" name="RERA" details="Respiratory Effort Related Arousal" label="RERA" color="gold"/>
|
||||
<channel id="0x1007" class="data" name="VSnore" details="Vibratory Snore" label="VS" unit="" color="red"/>
|
||||
<channel id="0x1008" class="data" name="VSnore2" details="Vibratory Snore 2" label="VS2" unit="" color="orange"/>
|
||||
<channel id="0x1009" class="data" name="PressurePulse" details="Pressure Pulse" label="PP" color="dark red"/>
|
||||
|
||||
<channel id="0x1020" class="data" name="PressureMin" details="Min Therapy Pressure" label="PMin" color="black"/>
|
||||
<channel id="0x1021" class="data" name="PressureMax" details="Max Therapy Pressure" label="PMax" color="black"/>
|
||||
<channel id="0x1022" class="data" name="RampTime" details="Ramp Time" label="Ramp Time" color="black"/>
|
||||
<channel id="0x1023" class="data" name="RampPressure" details="Ramp Starting Pressure" label="Ramp Pr." color="black"/>
|
||||
|
||||
<channel id="0x1100" class="data" name="Flow" details="Flow Rate" label="Flow Rate" unit="L/min" color="black"/>
|
||||
<channel id="0x1101" class="data" name="MaskPres" details="Mask Pressure" label="Mask Pressure" unit="cmH20" color="blue"/>
|
||||
<channel id="0x1102" class="data" name="MaskPresHi" details="Mask Pressure" label="Mask Pressure" unit="cmH20" color="blue" link="0x1101"/>
|
||||
<channel id="0x1103" class="data" name="TidalVolume" details="Tidal Volume" label="Tidal Volume" unit="" color="magenta"/>
|
||||
<channel id="0x1104" class="data" name="Snore" details="Snore" label="Snore" unit="" color="grey"/>
|
||||
<channel id="0x1105" class="data" name="MinuteVent" details="Minute Ventilation" label="Minute Vent." unit="" color="dark cyan"/>
|
||||
<channel id="0x1106" class="data" name="RespRate" details="Respiratory Rate" label="Resp. Rate" unit="Breaths/Min" color="dark magenta"/>
|
||||
<channel id="0x1107" class="data" name="PTB" details="Patient Triggered Breaths" label="Pat. Trig. Breaths" unit="%" color="dark grey"/>
|
||||
<channel id="0x1108" class="data" name="Leak" details="Leak Rate" label="Leaks" unit="L/min" color="dark green"/>
|
||||
<channel id="0x1109" class="data" name="I:E" details="Inspiratory:Expiratory" label="I:E" unit="ratio" color="dark red"/>
|
||||
<channel id="0x110a" class="data" name="Te" details="Te" label="Te" unit="" color="dark green"/>
|
||||
<channel id="0x110b" class="data" name="Ti" details="Ti" label="Ti" unit="" color="dark blue"/>
|
||||
<channel id="0x110c" class="data" name="Pressure" details="Pressure" label="P" unit="cmH20" color="dark green"/>
|
||||
<channel id="0x110d" class="data" name="IPAP" details="Inspiratory Pressure" label="IPAP" unit="cmH20" color="orange"/>
|
||||
<channel id="0x110e" class="data" name="EPAP" details="Expiratory Pressure" label="EPAP" unit="cmH20" color="light blue"/>
|
||||
<channel id="0x110f" class="data" name="PS" details="Pressure Support" label="PS" unit="cmH20" color="dark blue"/>
|
||||
<channel id="0x1110" class="data" name="IPAPLo" details="Inspiratory Pressure Lo" label="IPAP Lo" unit="cmH20" color="grey"/>
|
||||
<channel id="0x1111" class="data" name="IPAPHi" details="Inspiratory Pressure Hi" label="IPAP Hi" unit="cmH20" color="grey"/>
|
||||
<channel id="0x1112" class="data" name="RespEvent" details="Respiratory Events" label="Resp Events" unit="" color="black"/>
|
||||
<channel id="0x1113" class="data" name="FLG" details="Flow Limit Graph" label="Flow Limit" unit="" color="dark grey"/>
|
||||
<channel id="0x1150" class="data" name="PRS1_00" details="Unknown 00" label="U00" unit="" color="black"/>
|
||||
<channel id="0x1151" class="data" name="PRS1_01" details="Unknown 01" label="U01" unit="" color="black"/>
|
||||
<channel id="0x1152" class="data" name="PRS1_08" details="Unknown 08" label="U08" unit="" color="black"/>
|
||||
<channel id="0x1153" class="data" name="PRS1_09" details="Unknown 09" label="U09" unit="" color="black"/>
|
||||
<channel id="0x1154" class="data" name="PRS1_0A" details="Unknown 0A" label="U0A" unit="" color="black"/>
|
||||
<channel id="0x1155" class="data" name="PRS1_0B" details="Unknown 0B" label="U0B" unit="" color="light blue"/>
|
||||
<channel id="0x1156" class="data" name="PRS1_0C" details="Unknown 0C" label="U0C" unit="" color="black"/>
|
||||
<channel id="0x1157" class="data" name="PRS1_0E" details="Unknown 0E" label="U0E" unit="" color="dark green"/>
|
||||
<channel id="0x1158" class="data" name="PRS1_10" details="Unknown 10" label="U10" unit="" color="black"/>
|
||||
<channel id="0x1159" class="data" name="PRS1_12" details="PRS1 Unknown 12" label="U12" unit="" color="black"/>
|
||||
<channel id="0x1160" class="data" name="RMS9_E01" details="RMS9 Empty 1" label="E01" unit="" color="black"/>
|
||||
<channel id="0x1161" class="data" name="RMS9_E02" details="RMS9 Empty 2" label="U02" unit="" color="black"/>
|
||||
|
||||
<channel id="0x1200" class="setting" scope="!session" name="PAPMode" details="PAP Mode" label="PAP Mode" type="integer">
|
||||
<option id="0" value="CPAP"/>
|
||||
<option id="1" value="Auto"/>
|
||||
<option id="2" value="Bi-Level"/>
|
||||
<option id="3" value="ASV"/>
|
||||
</channel>
|
||||
</group>
|
||||
<group name="OXI">
|
||||
<channel id="0x1800" class="data" name="Pulse" details="Pulse Rate" label="Pulse" color="red"/>
|
||||
<channel id="0x1801" class="data" name="SPO2" details="SPO2" label="SPO2" color="blue"/>
|
||||
<channel id="0x1802" class="data" name="Plethy" details="Plethysomogram" label="Plethy" color="black"/>
|
||||
<channel id="0x1803" class="data" name="PulseChange" details="Pulse Change" label="Pulse Change" color="red"/>
|
||||
<channel id="0x1804" class="data" name="SPO2Drop" details="SPO2Drop" label="SPO2 Drop" color="blue"/>
|
||||
</group>
|
||||
<group name="SLEEP">
|
||||
<channel id="0x2000" class="data" name="SleepStage" details="Sleep Stage" label="Sleep Stage" color="dark grey"/>
|
||||
<channel id="0x2001" class="data" name="ZEOBW" details="Zeo Brainwave" label="ZeoWave" color="black"/>
|
||||
</group>
|
||||
<group name="GENERAL">
|
||||
<channel id="0x0800" class="data" name="BPSys" details="Blood Pressure Systolic" label="BPS" unit="mmHg" color="red"/>
|
||||
<channel id="0x0801" class="data" name="BPDia" details="Blood Pressure Diastolic" label="BPD" unit="mmHg" color="blue"/>
|
||||
<channel id="0x0802" class="data" name="Glucose" details="Blood Glucose" label="BGL" unit="mmol/L" color="black"/>
|
||||
<channel id="0x0803" class="data" scope="!day" name="Weight" details="Weight" label="Weight" unit="Kg" color="black"/>
|
||||
<channel id="0x0804" class="data" scope="!day" name="Height" details="Height" label="Height" unit="cm" color="blue"/>
|
||||
<channel id="0x0805" class="data" name="Bookmark" details="Session Bookmark" label="Bookmark" unit="duration" color="orange"/>
|
||||
<channel id="0xd000" class="data" scope="!day" unique="true" name="Journal" details="Journal Notes" label="Journal" type="richtext"/>
|
||||
</group>
|
||||
<group name="PRS1">
|
||||
<!-- PRS1 Settings -->
|
||||
<channel id="0xe101" class="setting" scope="!session" name="HumidStat" details="Humidifier Status" label="Hum. Status" type="bool">
|
||||
<option id="0" value="Off"/>
|
||||
<option id="1" value="On"/>
|
||||
</channel>
|
||||
<channel id="0xe102" class="setting" scope="!session" name="HumidSet" details="Humidifier Setting" label="Hum. Setting" type="integer">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="x1"/>
|
||||
<Option id="2" value="x2"/>
|
||||
<Option id="3" value="x3"/>
|
||||
<Option id="4" value="x4"/>
|
||||
<Option id="5" value="x5"/>
|
||||
</channel>
|
||||
<channel id="0xe103" class="setting" scope="!session" name="SysOneResistStat" details="System One Resistance Status" label="S1 Resist. Status" type="bool">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="On"/>
|
||||
</channel>
|
||||
<channel id="0xe104" class="setting" scope="!session" name="SysOneResistSet" details="System One Resistance Setting" label="S1 Resist. Setting" type="integer">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="x1"/>
|
||||
<Option id="2" value="x2"/>
|
||||
<Option id="3" value="x3"/>
|
||||
<Option id="4" value="x4"/>
|
||||
<Option id="5" value="x5"/>
|
||||
</channel>
|
||||
<channel id="0xe105" class="setting" scope="!session" name="FlexMode" details="Pressure Relief Mode" label="Flex Mode" type="integer">
|
||||
<Option id="0" value="None"/>
|
||||
<Option id="1" value="C-Flex"/>
|
||||
<Option id="2" value="C-Flex+"/>
|
||||
<Option id="3" value="A-Flex"/>
|
||||
</channel>
|
||||
<channel id="0xe106" class="setting" scope="!session" name="FlexSet" details="Pressure Relief Setting" label="Flex Set." type="integer">
|
||||
<Option id="0" value="0"/>
|
||||
<Option id="1" value="1"/>
|
||||
<Option id="2" value="2"/>
|
||||
<Option id="3" value="3"/>
|
||||
</channel>
|
||||
<channel id="0xe107" class="setting" scope="!session" name="HoseDiam" details="Hose Diameter" label="Hose Diameter" type="bool">
|
||||
<Option id="0" value="22mm"/>
|
||||
<Option id="1" value="15mm"/>
|
||||
</channel>
|
||||
<channel id="0xe108" class="setting" scope="!session" name="SysLock" details="System Lock Status" label="Sys Lock" type="bool">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="On"/>
|
||||
</channel>
|
||||
<channel id="0xe109" class="setting" scope="!session" name="AutoOn" details="Auto On" label="Auto On" type="bool">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="On"/>
|
||||
</channel>
|
||||
<channel id="0xe10a" class="setting" scope="!session" name="AutoOff" details="Auto Off" label="Auto Off" type="bool">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="On"/>
|
||||
</channel>
|
||||
<channel id="0xe10b" class="setting" scope="!session" name="MaskAlert" details="Mask Alert" label="Mask Alert" type="bool">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="On"/>
|
||||
</channel>
|
||||
<channel id="0xe10c" class="setting" scope="!session" name="ShowAHI" details="Show AHI" label="Show AHI" type="bool">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="On"/>
|
||||
</channel>
|
||||
|
||||
<channel id="0xe10d" class="setting" scope="!session" name="PRS1Mode" details="PAP Mode" label="PAP Mode" type="integer" link="0x1200">
|
||||
<Option id="0" value="CPAP"/>
|
||||
<Option id="1" value="Auto"/>
|
||||
<Option id="2" value="BIPAP"/>
|
||||
<Option id="3" value="AutoSV"/>
|
||||
</channel>
|
||||
</group>
|
||||
<group name="RMS9">
|
||||
<!-- RESMED Settings -->
|
||||
<channel id="0xe200" class="setting" scope="!session" name="RMS9Mode" details="PAP Mode" label="PAP Mode" type="integer" link="0x1200">
|
||||
<Option id="0" value="CPAP"/>
|
||||
<Option id="1" value="Auto"/>
|
||||
<Option id="2" value="VPAP"/>
|
||||
<Option id="3" value="ASV"/>
|
||||
</channel>
|
||||
<channel id="0xe201" class="setting" scope="!session" name="EPR" details="EPR Mode" label="EPR Mode" type="integer">
|
||||
<Option id="0" value="Off"/>
|
||||
<Option id="1" value="EPR"/>
|
||||
<Option id="2" value="EPR?"/>
|
||||
<Option id="3" value="EPR?"/>
|
||||
</channel>
|
||||
<channel id="0xe202" class="setting" scope="!session" name="EPRSet" details="EPR Setting" label="EPR Setting" type="integer">
|
||||
<Option id="0" value="0"/>
|
||||
<Option id="1" value="1"/>
|
||||
<Option id="2" value="2"/>
|
||||
<Option id="3" value="3"/>
|
||||
</channel>
|
||||
</group>
|
||||
<group name="MACHINE">
|
||||
<!-- General Per Machine Settings -->
|
||||
<channel id="0xf000" class="setting" scope="!machine" name="Type" details="Type" label="Type" type="integer">
|
||||
<option id="0" value="Unknown"/>
|
||||
<option id="1" value="CPAP"/>
|
||||
<option id="2" value="Oximeter"/>
|
||||
<option id="3" value="Sleep"/>
|
||||
<option id="4" value="Journal"/>
|
||||
<option id="5" value="EEG"/>
|
||||
</channel>
|
||||
<channel id="0xf001" class="setting" scope="!machine" name="Brand" details="Brand" label="Brand" type="string"/>
|
||||
<channel id="0xf002" class="setting" scope="!machine" name="Model" details="Model" label="Model" type="string"/>
|
||||
<channel id="0xf003" class="setting" scope="!machine" name="ModelNumber" details="Model Number" label="Model Number" type="string"/>
|
||||
<channel id="0xf004" class="setting" scope="!machine" name="SubModel" details="Sub-Model" label="Sub Model" type="string"/>
|
||||
<channel id="0xf005" class="setting" scope="!machine" name="Serial" details="Serial" label="Serial" type="string"/>
|
||||
<channel id="0xf006" class="setting" scope="!machine" name="Notes" details="Machine Notes" label="Notes" type="richtext"/>
|
||||
</group>
|
||||
</channels>
|
47
docs/graphs.xml
Normal file
47
docs/graphs.xml
Normal file
@ -0,0 +1,47 @@
|
||||
<!DOCTYPE Graphs>
|
||||
<Schema language="en" version="1.0">
|
||||
<Graph name="Event Flags" group="Daily" master="true">
|
||||
<layer type="flaglines">
|
||||
<line name="CSR" func="data" type="span">
|
||||
<line name="ClearAirway" func="data" type="bar">
|
||||
<line name="Obstructive" func="data" type="bar">
|
||||
<line name="Apnea" func="data" type="bar">
|
||||
<line name="Hypopnea" func="data" type="bar">
|
||||
<line name="FlowLimit" func="data" type="bar">
|
||||
<line name="RERA" func="data" type="bar">
|
||||
<line name="VSnore" func="data" type="bar">
|
||||
<line name="VSnore2" func="data" type="bar">
|
||||
</layer>
|
||||
</Graph>
|
||||
<Graph name="Flow Rate" group="Daily">
|
||||
<layer type="overlay" data="CSR" visual="span"/>
|
||||
<layer type="linechart" data="FlowRate"/>
|
||||
<layer type="overlay" data="Obstructive" visual="bar"/>
|
||||
<layer type="overlay" data="Hypopnea" visual="bar"/>
|
||||
<layer type="overlay" data="Apnea" visual="bar"/>
|
||||
<layer type="overlay" data="ClearAirway" visual="bar"/>
|
||||
<layer type="overlay" data="FlowLimit" visual="bar"/>
|
||||
<layer type="overlay" data="RERA" visual="bar"/>
|
||||
<layer type="overlay" data="VSnore" visual="bar"/>
|
||||
<layer type="overlay" data="PressurePulse" visual="dot"/>
|
||||
</Graph>
|
||||
<Graph name="AHI" group="Overview">
|
||||
<layer type="barchart">
|
||||
<slice name="Hypopnea" func="cph">
|
||||
<slice name="Apnea" func="cph">
|
||||
<slice name="Obstructive" func="cph">
|
||||
<slice name="ClearAirway" func="cph">
|
||||
</layer>
|
||||
</Graph>
|
||||
<Graph name="Pressure" visual="linechart">
|
||||
<layer name="Pressure" func="wavg" color="dark green">
|
||||
<layer name="Pressure" func="min" color="dark red">
|
||||
<layer name="Pressure" func="max" color="dark blue">
|
||||
<layer name="IPAP" func="wavg" color="red">
|
||||
<layer name="EPAP" func="wavg" color="blue">
|
||||
</Graph>
|
||||
|
||||
|
||||
|
||||
</Graphs>
|
||||
</Schema>
|
68
docs/schema.xml
Normal file
68
docs/schema.xml
Normal file
@ -0,0 +1,68 @@
|
||||
<!DOCTYPE Schema>
|
||||
<!-- Schema List Notes: -->
|
||||
<Schema language="en" version="1.0">
|
||||
<enum name="class"/>
|
||||
<item id="0" value="event"/>
|
||||
<item id="1" value="waveform"/>
|
||||
<item id="2" value="setting"/>
|
||||
<default id="0">
|
||||
</enum>
|
||||
<enum name="function">
|
||||
<item id="0" value="data"/>
|
||||
<item id="1" value="avg"/>
|
||||
<item id="2" value="wavg"/>
|
||||
<item id="3" value="min"/>
|
||||
<item id="4" value="max"/>
|
||||
<item id="5" value="sum"/>
|
||||
<item id="6" value="cnt"/>
|
||||
<item id="7" value="p90"/>
|
||||
<item id="8" value="cph"/>
|
||||
<item id="9" value="sph"/>
|
||||
<item id="10" value="hours"/>
|
||||
<item id="11" value="set"/>
|
||||
<default id="0">
|
||||
</enum>
|
||||
<enum name="scope">
|
||||
<item id="0" value="preference">
|
||||
<item id="1" value="machine">
|
||||
<item id="2" value="day">
|
||||
<item id="3" value="session">
|
||||
<default id="3">
|
||||
</enum>
|
||||
<enum name="datatype">
|
||||
<item id="0" value=""/>
|
||||
<item id="1" value=""/>
|
||||
<item id="2" value=""/>
|
||||
<item id="3" value=""/>
|
||||
<item id="4" value=""/>
|
||||
<item id="5" value=""/>
|
||||
<item id="6" value=""/>
|
||||
<item id="7" value=""/>
|
||||
<item id="8" value=""/>
|
||||
<item id="9" value=""/>
|
||||
<item id="10" value=""/>
|
||||
<item id="11" value=""/>
|
||||
<default id="0">
|
||||
</enum>
|
||||
<object name="color">
|
||||
<property name="func" type="function"/>
|
||||
<property name="color" type="qcolor"/>
|
||||
</object>
|
||||
<object name="option">
|
||||
<property name="id" type="integer"/>
|
||||
<property name="type" type="datatype"/>
|
||||
<property name="value" type="variant"/>
|
||||
</object>
|
||||
<object name="channel">
|
||||
<property name="id" type="number"/>
|
||||
<property name="class" type="class"/>
|
||||
<property name="scope" type="scope"/>
|
||||
<property name="name" type="string"/>
|
||||
<property name="details" type="string"/>
|
||||
<property name="label" type="string"/>
|
||||
<property name="unit" type="string"/>
|
||||
<property name="type" type="datatype"/>
|
||||
<array name="colors" type="color" index="func"/>
|
||||
<array name="options" type="option" index="id"/>
|
||||
</object>
|
||||
</Schema>
|
9
main.cpp
9
main.cpp
@ -10,7 +10,7 @@
|
||||
#include <QStringList>
|
||||
#include <QDebug>
|
||||
|
||||
|
||||
#include "SleepLib/schema.h"
|
||||
#include "mainwindow.h"
|
||||
#include "SleepLib/profiles.h"
|
||||
|
||||
@ -45,6 +45,11 @@ void MyOutputHandler(QtMsgType type, const char *msg) {
|
||||
//loglock.unlock();
|
||||
}
|
||||
|
||||
void initialize()
|
||||
{
|
||||
schema::init();
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
#ifdef Q_WS_X11
|
||||
@ -52,8 +57,8 @@ int main(int argc, char *argv[])
|
||||
#endif
|
||||
QGL::setPreferredPaintEngine(QPaintEngine::OpenGL);
|
||||
QApplication a(argc, argv);
|
||||
|
||||
a.setApplicationName("SleepyHead");
|
||||
initialize();
|
||||
|
||||
/*int id=QFontDatabase::addApplicationFont(":/fonts/FreeSans.ttf");
|
||||
QStringList ffam=QFontDatabase::applicationFontFamilies(id);
|
||||
|
@ -20,6 +20,7 @@
|
||||
#include "SleepLib/loader_plugins/zeo_loader.h"
|
||||
#include "SleepLib/loader_plugins/resmed_loader.h"
|
||||
#include "preferencesdialog.h"
|
||||
#include "SleepLib/schema.h"
|
||||
|
||||
|
||||
#include "Graphs/glcommon.h"
|
||||
@ -34,9 +35,10 @@ void MainWindow::Log(QString s)
|
||||
static QMutex loglock,strlock;
|
||||
static int start=QDateTime::currentDateTime().toTime_t();
|
||||
static QStringList slist;
|
||||
if (!loglock.tryLock()) {
|
||||
return;
|
||||
}
|
||||
//if (!loglock.tryLock()) {
|
||||
//return;
|
||||
//}
|
||||
loglock.lock();
|
||||
|
||||
strlock.lock();
|
||||
QString tmp=QString("%1: %2").arg(QDateTime::currentDateTime().toTime_t()-start,5,10,QChar('0')).arg(s);
|
||||
@ -150,6 +152,7 @@ MainWindow::MainWindow(QWidget *parent) :
|
||||
connect(netmanager, SIGNAL(finished(QNetworkReply*)), this, SLOT(replyFinished(QNetworkReply*)));
|
||||
|
||||
connect(ui->webView, SIGNAL(statusBarMessage(QString)), this, SLOT(updatestatusBarMessage(QString)));
|
||||
|
||||
}
|
||||
extern MainWindow *mainwin;
|
||||
MainWindow::~MainWindow()
|
||||
@ -197,6 +200,12 @@ void MainWindow::Startup()
|
||||
if (overview) overview->ReloadGraphs();
|
||||
qprogress->hide();
|
||||
qstatus->setText("");
|
||||
schema::Channel & item=schema::channel["SysOneResistSet"];
|
||||
if (!item.isNull()) {
|
||||
for (QHash<int,QString>::iterator i=item.m_options.begin();i!=item.m_options.end();i++) {
|
||||
qDebug() << i.key() << i.value();
|
||||
}
|
||||
}
|
||||
//qstatusbar->clearMessage();
|
||||
|
||||
}
|
||||
|
@ -57,7 +57,7 @@ Overview::Overview(QWidget *parent,Profile * _profile,gGraphView * shared) :
|
||||
LK=new gGraph(GraphView,"Leaks",default_height,0);
|
||||
|
||||
uc=new SummaryChart(profile,"Hours",GT_BAR);
|
||||
uc->addSlice(EmptyChannel,QColor("green"),ST_HOURS);
|
||||
uc->addSlice("",QColor("green"),ST_HOURS);
|
||||
UC->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin);
|
||||
gXAxis *gx=new gXAxis();
|
||||
gx->setUtcFix(true);
|
||||
|
@ -13,7 +13,7 @@
|
||||
#include <QDateEdit>
|
||||
#include "SleepLib/profiles.h"
|
||||
#include "Graphs/gGraphView.h"
|
||||
#include "Graphs/gBarChart.h"
|
||||
#include "Graphs/gSummaryChart.h"
|
||||
#include "report.h"
|
||||
|
||||
namespace Ui {
|
||||
|
67
oximetry.cpp
67
oximetry.cpp
@ -9,7 +9,7 @@
|
||||
#include "SleepLib/loader_plugins/cms50_loader.h"
|
||||
#include "SleepLib/event.h"
|
||||
#include "Graphs/gXAxis.h"
|
||||
#include "Graphs/gBarChart.h"
|
||||
#include "Graphs/gSummaryChart.h"
|
||||
#include "Graphs/gLineChart.h"
|
||||
#include "Graphs/gYAxis.h"
|
||||
|
||||
@ -66,7 +66,7 @@ Oximetry::Oximetry(QWidget *parent,Profile * _profile,gGraphView * shared) :
|
||||
SPO2=new gGraph(GraphView,tr("SPO2"),120);
|
||||
foobar=new gShadowArea();
|
||||
CONTROL->AddLayer(foobar);
|
||||
Layer *cl=new gLineChart(OXI_Plethysomogram);
|
||||
Layer *cl=new gLineChart(OXI_Plethy);
|
||||
CONTROL->AddLayer(cl);
|
||||
cl->SetDay(day);
|
||||
CONTROL->setBlockZoom(true);
|
||||
@ -79,16 +79,13 @@ Oximetry::Oximetry(QWidget *parent,Profile * _profile,gGraphView * shared) :
|
||||
}
|
||||
|
||||
// Create the Event Lists to store / import data
|
||||
ev_plethy=new EventList(OXI_Plethysomogram,EVL_Waveform,1,0,0,0,1000.0/50.0);
|
||||
session->eventlist[OXI_Plethysomogram].push_back(ev_plethy);
|
||||
ev_plethy=session->AddEventList(OXI_Plethy,EVL_Waveform,1,0,0,0,1000.0/50.0);
|
||||
|
||||
ev_pulse=new EventList(OXI_Pulse,EVL_Event,1);
|
||||
session->eventlist[OXI_Pulse].push_back(ev_pulse);
|
||||
ev_pulse=session->AddEventList(OXI_Pulse,EVL_Event,1);
|
||||
|
||||
ev_spo2=new EventList(OXI_SPO2,EVL_Event,1);
|
||||
session->eventlist[OXI_SPO2].push_back(ev_spo2);
|
||||
ev_spo2=session->AddEventList(OXI_SPO2,EVL_Event,1);
|
||||
|
||||
plethy=new gLineChart(OXI_Plethysomogram,Qt::black,false,true);
|
||||
plethy=new gLineChart(OXI_Plethy,Qt::black,false,true);
|
||||
plethy->SetDay(day);
|
||||
|
||||
CONTROL->AddLayer(plethy); //new gLineChart(OXI_Plethysomogram));
|
||||
@ -290,7 +287,7 @@ void Oximetry::on_RunButton_toggled(bool checked)
|
||||
|
||||
sess->eventlist[OXI_SPO2].push_back(ev_spo2);
|
||||
sess->eventlist[OXI_Pulse].push_back(ev_pulse);
|
||||
sess->eventlist[OXI_Plethysomogram].push_back(ev_plethy);
|
||||
sess->eventlist[OXI_Plethy].push_back(ev_plethy);
|
||||
//Session *sess=session;
|
||||
sess->SetSessionID(starttime/1000L);
|
||||
|
||||
@ -310,36 +307,31 @@ void Oximetry::on_RunButton_toggled(bool checked)
|
||||
sess->wavg(OXI_SPO2);
|
||||
sess->p90(OXI_SPO2);
|
||||
|
||||
sess->avg(OXI_Plethysomogram);
|
||||
sess->wavg(OXI_Plethysomogram);
|
||||
sess->p90(OXI_Plethysomogram);
|
||||
sess->setMin(OXI_Plethysomogram,ev_plethy->min());
|
||||
sess->setMax(OXI_Plethysomogram,ev_plethy->max());
|
||||
sess->avg(OXI_Plethy);
|
||||
sess->wavg(OXI_Plethy);
|
||||
sess->p90(OXI_Plethy);
|
||||
sess->setMin(OXI_Plethy,ev_plethy->min());
|
||||
sess->setMax(OXI_Plethy,ev_plethy->max());
|
||||
|
||||
sess->setFirst(OXI_Plethysomogram,ev_plethy->first());
|
||||
sess->setLast(OXI_Plethysomogram,ev_plethy->last());
|
||||
sess->setFirst(OXI_Plethy,ev_plethy->first());
|
||||
sess->setLast(OXI_Plethy,ev_plethy->last());
|
||||
|
||||
sess->updateFirst(sess->first(OXI_Pulse));
|
||||
sess->updateLast(sess->last(OXI_Pulse));
|
||||
sess->updateFirst(sess->first(OXI_SPO2));
|
||||
sess->updateLast(sess->last(OXI_SPO2));
|
||||
sess->updateFirst(sess->first(OXI_Plethysomogram));
|
||||
sess->updateLast(sess->last(OXI_Plethysomogram));
|
||||
sess->updateFirst(sess->first(OXI_Plethy));
|
||||
sess->updateLast(sess->last(OXI_Plethy));
|
||||
|
||||
sess->SetChanged(true);
|
||||
mach->AddSession(sess,profile);
|
||||
mach->Save();
|
||||
|
||||
ev_plethy=new EventList(OXI_Plethysomogram,EVL_Waveform,1,0,0,0,1000.0/50.0);
|
||||
session->eventlist[OXI_Plethysomogram].push_back(ev_plethy);
|
||||
ev_plethy=session->AddEventList(OXI_Plethy,EVL_Waveform,1,0,0,0,1000.0/50.0);
|
||||
ev_pulse=session->AddEventList(OXI_Pulse,EVL_Event,1);
|
||||
ev_spo2=session->AddEventList(OXI_SPO2,EVL_Event,1);
|
||||
|
||||
ev_pulse=new EventList(OXI_Pulse,EVL_Event,1);
|
||||
session->eventlist[OXI_Pulse].push_back(ev_pulse);
|
||||
|
||||
ev_spo2=new EventList(OXI_SPO2,EVL_Event,1);
|
||||
session->eventlist[OXI_SPO2].push_back(ev_spo2);
|
||||
|
||||
session->setCount(OXI_Plethysomogram,0);
|
||||
session->setCount(OXI_Plethy,0);
|
||||
session->setCount(OXI_Pulse,0);
|
||||
session->setCount(OXI_SPO2,0);
|
||||
|
||||
@ -367,7 +359,7 @@ void Oximetry::UpdatePlethy(qint8 d)
|
||||
if (d>ev_plethy->max()) ev_plethy->setMax(d);
|
||||
int i=ev_plethy->count()+1;
|
||||
ev_plethy->setCount(i);
|
||||
session->setCount(OXI_Plethysomogram,i); // update the cache
|
||||
session->setCount(OXI_Plethy,i); // update the cache
|
||||
//ev_plethy->AddEvent(lasttime,d);
|
||||
lasttime+=20; // 50 samples per second
|
||||
PLETHY->SetMinY(ev_plethy->min());
|
||||
@ -679,8 +671,7 @@ void Oximetry::on_ImportButton_clicked()
|
||||
int drop=max-min;
|
||||
if (drop>6) {
|
||||
if (!oxf1) {
|
||||
oxf1=new EventList(OXI_PulseChange,EVL_Event);
|
||||
session->eventlist[OXI_PulseChange].push_back(oxf1);
|
||||
oxf1=session->AddEventList(OXI_PulseChange,EVL_Event);
|
||||
}
|
||||
oxf1->AddEvent(tt,drop);
|
||||
}
|
||||
@ -696,15 +687,12 @@ void Oximetry::on_ImportButton_clicked()
|
||||
int drop=max-min;
|
||||
if (drop>4) {
|
||||
if (!oxf1) {
|
||||
oxf2=new EventList(OXI_SPO2Drop,EVL_Event);
|
||||
session->eventlist[OXI_SPO2Drop].push_back(oxf2);
|
||||
oxf2=session->AddEventList(OXI_SPO2Drop,EVL_Event);
|
||||
}
|
||||
oxf2->AddEvent(tt,drop);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
++rb_pos;
|
||||
rb_pos=rb_pos % rb_size;
|
||||
|
||||
@ -743,14 +731,11 @@ void Oximetry::on_ImportButton_clicked()
|
||||
day->AddSession(session);
|
||||
|
||||
// As did these
|
||||
ev_plethy=new EventList(OXI_Plethysomogram,EVL_Waveform,1,0,0,0,1000.0/50.0);
|
||||
session->eventlist[OXI_Plethysomogram].push_back(ev_plethy);
|
||||
ev_plethy=session->AddEventList(OXI_Plethy,EVL_Waveform,1,0,0,0,1000.0/50.0);
|
||||
|
||||
ev_pulse=new EventList(OXI_Pulse,EVL_Event,1);
|
||||
session->eventlist[OXI_Pulse].push_back(ev_pulse);
|
||||
ev_pulse=session->AddEventList(OXI_Pulse,EVL_Event,1);
|
||||
|
||||
ev_spo2=new EventList(OXI_SPO2,EVL_Event,1);
|
||||
session->eventlist[OXI_SPO2].push_back(ev_spo2);
|
||||
ev_spo2=session->AddEventList(OXI_SPO2,EVL_Event,1);
|
||||
}
|
||||
delete port;
|
||||
port=NULL;
|
||||
|
@ -84,11 +84,11 @@ PreferencesDialog::PreferencesDialog(QWidget *parent,Profile * _profile) :
|
||||
ui->eventTable->setColumnWidth(1,55);
|
||||
int row=0;
|
||||
QTableWidgetItem *item;
|
||||
QHash<ChannelID, Channel>::iterator ci;
|
||||
for (ci=channel.begin();ci!=channel.end();ci++) {
|
||||
if ((ci.value().channeltype()==CT_Event) || (ci.value().channeltype()==CT_Graph)) {
|
||||
QHash<QString, schema::Channel *>::iterator ci;
|
||||
for (ci=schema::channel.names.begin();ci!=schema::channel.names.end();ci++) {
|
||||
if (ci.value()->type()==schema::DATA) {
|
||||
ui->eventTable->insertRow(row);
|
||||
item=new QTableWidgetItem(ci.value().details());
|
||||
item=new QTableWidgetItem(ci.value()->description());
|
||||
ui->eventTable->setItem(row,2,item);
|
||||
QCheckBox *c=new QCheckBox(ui->eventTable);
|
||||
c->setChecked(true);
|
||||
|
Loading…
Reference in New Issue
Block a user