mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
PRS1 Respiratory Rate Calculations and Graph. Temporarily overlayed over ASV's real RespRate graph..
This commit is contained in:
parent
d8e24607da
commit
c7a15f6d36
@ -38,11 +38,7 @@ extern QProgressBar *qprogress;
|
||||
|
||||
QHash<int,QString> ModelMap;
|
||||
|
||||
const quint16 CRC16_INIT_VALUE=0xffff;
|
||||
const quint16 CRC16_XOR_VALUE=0x0000;
|
||||
|
||||
typedef quint16 crc_t;
|
||||
//quint16 crctable[256];
|
||||
|
||||
static const crc_t crc_table[256] = {
|
||||
0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf,
|
||||
@ -1166,9 +1162,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);
|
||||
|
||||
QString FlowRate="FlowRate";
|
||||
QString MaskPressure="MaskPressure";
|
||||
QString wc[2]={FlowRate,MaskPressure};
|
||||
//QString FlowRate=CPAP_FlowRate;
|
||||
//QString MaskPressure="MaskPressure";
|
||||
QString wc[2]={CPAP_FlowRate,CPAP_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;
|
||||
@ -1221,10 +1217,10 @@ bool PRS1Loader::OpenWaveforms(Session *session,QString filename)
|
||||
else {
|
||||
a->AddWaveform(qint64(start)*1000L,(char *)waveform[i],wlength[i],qint64(wdur[i])*1000L);
|
||||
}
|
||||
if (wc[i]==FlowRate) {
|
||||
if (wc[i]==CPAP_FlowRate) {
|
||||
a->setMax(120);
|
||||
a->setMin(-120);
|
||||
} else if (wc[i]==MaskPressure) {
|
||||
} else if (wc[i]==CPAP_MaskPressure) {
|
||||
/* int v=ceil(a->max()/5);
|
||||
a->setMax(v*5);
|
||||
v=floor(a->min()/5);
|
||||
@ -1275,15 +1271,211 @@ bool PRS1Loader::OpenWaveforms(Session *session,QString filename)
|
||||
else {
|
||||
a->AddWaveform(qint64(start)*1000L,(char *)waveform[i],wlength[i],qint64(wdur[i])*1000L);
|
||||
}
|
||||
if (wc[i]==FlowRate) {
|
||||
if (wc[i]==CPAP_FlowRate) {
|
||||
a->setMax(120);
|
||||
a->setMin(-120);
|
||||
} else if (wc[i]==MaskPressure) {
|
||||
} else if (wc[i]==CPAP_MaskPressure) {
|
||||
}
|
||||
session->updateLast(start+qint64(wdur[i])*1000L);
|
||||
}
|
||||
// if (!session->eventlist.contains(CPAP_RespRate)) // only ASV machines have it..
|
||||
CalcRespiratoryRate(session);
|
||||
return true;
|
||||
}
|
||||
// Generate RespiratoryRate graph
|
||||
void PRS1Loader::CalcRespiratoryRate(Session *session)
|
||||
{
|
||||
EventList *flow, *rr;
|
||||
for (int ws=0; ws < session->eventlist[CPAP_FlowRate].size(); ws++) {
|
||||
flow=session->eventlist[CPAP_FlowRate][ws];
|
||||
if (flow->count() > 5) {
|
||||
rr=new EventList(EVL_Event);//EVL_Waveform,1,0,0,0,60000);
|
||||
ChannelID resp="RespRate";
|
||||
if (session->eventlist.contains(resp)) resp+="2";
|
||||
session->eventlist[resp].push_back(rr);
|
||||
filterFlow(flow,rr);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void PRS1Loader::filterFlow(EventList *in, EventList *out)
|
||||
{
|
||||
int size=in->count();
|
||||
EventDataType *stage1=new EventDataType [size];
|
||||
EventDataType *stage2=new EventDataType [size];
|
||||
|
||||
QVector<EventDataType> med;
|
||||
med.reserve(8);
|
||||
|
||||
EventDataType r;
|
||||
int cnt;
|
||||
|
||||
// Anti-Alias the flow waveform to get rid of jagged edges.
|
||||
EventDataType c;
|
||||
double avg;
|
||||
int i;
|
||||
|
||||
|
||||
/*i=2;
|
||||
stage1[0]=in->data(0);
|
||||
stage1[1]=in->data(1);
|
||||
//stage1[2]=in->data(2);
|
||||
for (;i<size-2;i++) {
|
||||
med.clear();
|
||||
for (quint32 k=0;k<5;k++) {
|
||||
med.push_back(in->data(i-2+k));
|
||||
}
|
||||
qSort(med);
|
||||
stage1[i]=med[3];
|
||||
}
|
||||
stage1[i]=in->data(i);
|
||||
i++;
|
||||
stage1[i]=in->data(i); */
|
||||
|
||||
//i++;
|
||||
//stage1[i]=in->data(i);
|
||||
|
||||
stage2[0]=stage1[0];
|
||||
stage2[1]=stage1[1];
|
||||
stage2[2]=stage1[2];
|
||||
|
||||
i=3;
|
||||
for (;i<size-3;i++) {
|
||||
cnt=0;
|
||||
r=0;
|
||||
for (quint32 k=0;k<7;k++) {
|
||||
//r+=stage1[i-3+k];
|
||||
r+=in->data(i-3+k);
|
||||
cnt++;
|
||||
}
|
||||
c=r/float(cnt);
|
||||
stage2[i]=c;
|
||||
}
|
||||
stage2[i]=in->data(i);
|
||||
i++;
|
||||
stage2[i]=in->data(i);
|
||||
i++;
|
||||
stage2[i]=in->data(i);
|
||||
//i++;
|
||||
//stage2[i]=in->data(i);
|
||||
|
||||
float weight=0.6;
|
||||
//stage2[0]=in->data(0);
|
||||
stage1[0]=stage2[0];
|
||||
for (int i=1;i<size;i++) {
|
||||
//stage2[i]=in->data(i);
|
||||
stage1[i]=weight*stage2[i]+(1.0-weight)*stage1[i-1];
|
||||
}
|
||||
|
||||
|
||||
qint64 time=in->first();
|
||||
qint64 u1=0,u2=0,len,l1=0,l2=0;
|
||||
EventDataType lastc=0,thresh=0;
|
||||
QVector<int> breaths;
|
||||
QVector<qint64> breaths_start;
|
||||
|
||||
for (i=0;i<size;i++) {
|
||||
c=stage1[i];
|
||||
if (c>thresh) {
|
||||
if (lastc<=thresh) {
|
||||
u2=u1;
|
||||
u1=time;
|
||||
if (u2>0) {
|
||||
len=abs(u2-u1);
|
||||
//if (len>1500) {
|
||||
breaths_start.push_back(time);
|
||||
breaths.push_back(len);
|
||||
//}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (lastc>thresh) {
|
||||
l2=l1;
|
||||
l1=time;
|
||||
if (l2>0) {
|
||||
len=abs(l2-l1);
|
||||
//if (len>1500) {
|
||||
// breaths2_start.push_back(time);
|
||||
// breaths2.push_back(len);
|
||||
//}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
lastc=c;
|
||||
time+=200;
|
||||
}
|
||||
|
||||
qint64 window=60000;
|
||||
qint64 t1=in->first()-window/2;
|
||||
qint64 t2=in->first()+window/2;
|
||||
qint64 t;
|
||||
EventDataType br,q;
|
||||
int z=0;
|
||||
int l;
|
||||
|
||||
QVector<int> breaths2;
|
||||
QVector<qint64> breaths2_start;
|
||||
|
||||
int fir=0;
|
||||
do {
|
||||
br=0;
|
||||
bool first=true;
|
||||
bool cont=false;
|
||||
for (int i=fir;i<breaths.size();i++) {
|
||||
t=breaths_start[i];
|
||||
l=breaths[i];
|
||||
if (t+l < t1) continue;
|
||||
if (t > t2) break;
|
||||
|
||||
if (first) {
|
||||
first=false;
|
||||
fir=i;
|
||||
}
|
||||
//q=1;
|
||||
if (t<t1) {
|
||||
// move to start of previous breath
|
||||
t1=breaths_start[++i];
|
||||
t2=t1+window;
|
||||
fir=i;
|
||||
cont=true;
|
||||
break;
|
||||
//q=(t+l)-t1;
|
||||
//br+=(1.0/double(l))*double(q);
|
||||
|
||||
} else if (t+l>t2) {
|
||||
q=t2-t;
|
||||
br+=(1.0/double(l))*double(q);
|
||||
continue;
|
||||
} else
|
||||
br+=1.0;
|
||||
}
|
||||
if (cont) continue;
|
||||
breaths2.push_back(br);
|
||||
breaths2_start.push_back(t1+window/2);
|
||||
//out->AddEvent(t,br);
|
||||
//stage2[z++]=br;
|
||||
|
||||
t1+=window/2.0;
|
||||
t2+=window/2.0;
|
||||
} while (t2<in->last());
|
||||
|
||||
|
||||
for (int i=1;i<breaths2.size()-2;i++) {
|
||||
t=breaths2_start[i];
|
||||
med.clear();
|
||||
for (int j=0;j<4;j++) {
|
||||
med.push_back(breaths2[i+j-1]);
|
||||
}
|
||||
qSort(med);
|
||||
br=med[2];
|
||||
out->AddEvent(t,br);
|
||||
}
|
||||
|
||||
delete [] stage2;
|
||||
delete [] stage1;
|
||||
|
||||
}
|
||||
|
||||
void InitModelMap()
|
||||
{
|
||||
@ -1305,3 +1497,5 @@ void PRS1Loader::Register()
|
||||
initialized=true;
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
@ -59,9 +59,9 @@ protected:
|
||||
bool OpenWaveforms(Session *session,QString filename);
|
||||
bool Parse002(Session *session,unsigned char *buffer,int size,qint64 timestamp);
|
||||
bool Parse002ASV(Session *session,unsigned char *buffer,int size,qint64 timestamp);
|
||||
|
||||
void CalcRespiratoryRate(Session *);
|
||||
void filterFlow(EventList *in, EventList *out);
|
||||
unsigned char * m_buffer;
|
||||
};
|
||||
|
||||
|
||||
#endif // PRS1LOADER_H
|
||||
|
@ -806,13 +806,14 @@ void ResInitModelMap()
|
||||
RMS9ModelMap[36006]="ResMed S9 VPAP Auto";
|
||||
RMS9ModelMap[36007]="ResMed S9 VPAP Adapt";
|
||||
RMS9ModelMap[36008]="ResMed S9 VPAP ST";
|
||||
// S8 Series
|
||||
/* S8 Series
|
||||
RMS9ModelMap[33007]="ResMed S8 Escape";
|
||||
RMS9ModelMap[33039]="ResMed S8 Elite II";
|
||||
RMS9ModelMap[33051]="ResMed S8 Escape II";
|
||||
RMS9ModelMap[33064]="ResMed S8 Escape II AutoSet";
|
||||
RMS9ModelMap[33064]="ResMed S8 Escape II AutoSet";
|
||||
RMS9ModelMap[33129]="ResMed S8 AutoSet II";
|
||||
*/
|
||||
|
||||
resmed_codes[CPAP_FlowRate].push_back("Flow");
|
||||
resmed_codes[CPAP_MaskPressureHi].push_back("Mask Pres");
|
||||
|
15
daily.cpp
15
daily.cpp
@ -188,6 +188,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared, MainWindow *mw)
|
||||
|
||||
PTB->AddLayer(AddCPAP(new gLineChart(CPAP_PTB,Qt::gray,square)));
|
||||
MP->AddLayer(AddCPAP(new gLineChart(CPAP_MaskPressure,Qt::blue,false)));
|
||||
RR->AddLayer(AddCPAP(new gLineChart("RespRate2",Qt::red,false)));
|
||||
RR->AddLayer(AddCPAP(new gLineChart(CPAP_RespRate,Qt::darkMagenta,square)));
|
||||
MV->AddLayer(AddCPAP(new gLineChart(CPAP_MinuteVent,Qt::darkCyan,square)));
|
||||
TV->AddLayer(AddCPAP(new gLineChart(CPAP_TidalVolume,Qt::magenta,square)));
|
||||
@ -465,7 +466,7 @@ void Daily::Load(QDate date)
|
||||
"</style>"
|
||||
"</head>"
|
||||
"<body leftmargin=0 rightmargin=0 topmargin=0 marginwidth=0 marginheight=0>"
|
||||
"<table cellspacing=0 cellpadding=2 border=0 width='100%'>\n";
|
||||
"<table cellspacing=0 cellpadding=1 border=0 width='100%'>\n";
|
||||
QString tmp;
|
||||
//const int gwwidth=240;
|
||||
//const int gwheight=100;
|
||||
@ -554,8 +555,8 @@ void Daily::Load(QDate date)
|
||||
if (cpap->machine->GetClass()!="PRS1") {
|
||||
cs="4 width='100%' align=center>";
|
||||
} else cs="2 width='50%'>";
|
||||
html+="<tr><td colspan="+cs+"<table cellspacing=0 cellpadding=2 border=0 width='100%'>"
|
||||
"<tr><td align='right' bgcolor='#F88017'><b><font color='black'>"+tr("AHI")+"</font></b></td><td bgcolor='#F88017'><b><font color='black'>"+QString().sprintf("%.2f",ahi)+"</font></b></td></tr>\n"
|
||||
html+="<tr><td colspan="+cs+"<table cellspacing=0 cellpadding=1 border=0 width='100%'>"
|
||||
"<tr><td align='right' bgcolor='#F88017'><b><font color='black'>"+tr("AHI")+"</font></b></td><td width=20% bgcolor='#F88017'><b><font color='black'>"+QString().sprintf("%.2f",ahi)+"</font></b></td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#4040ff'><b><font color='white'> <a href='event=Hypopnea'>"+tr("Hypopnea")+"</a></font></b></td><td bgcolor='#4040ff'><font color='white'>"+QString().sprintf("%.2f",hi)+"</font></td></tr>\n";
|
||||
if (cpap->machine->GetClass()=="ResMed") {
|
||||
html+="<tr><td align='right' bgcolor='#208020'><b> <a href='event=Apnea'>"+tr("Unspecified Apnea")+"</a></b></td><td bgcolor='#208020'>"+QString().sprintf("%.2f",uai)+"</td></tr>\n";
|
||||
@ -565,15 +566,15 @@ void Daily::Load(QDate date)
|
||||
"</table></td>";
|
||||
|
||||
if (cpap->machine->GetClass()=="PRS1") {
|
||||
html+="<td colspan=2><table cellspacing=0 cellpadding=2 border=0 width='100%'>"
|
||||
"<tr><td align='right' bgcolor='#ffff80'><b> <a href='event=Respiratory Effort'>"+tr("RERA")+"</a></b></td><td bgcolor='#ffff80'>"+QString().sprintf("%.2f",rei)+"</td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#404040'><b> <font color='white'><a href='event=Flow Limit'>"+tr("FlowLimit")+"</a></font></b></td><td bgcolor='#404040'><font color='white'>"+a.sprintf("%.2f",fli)+"</font></td></tr>\n"
|
||||
html+="<td colspan=2><table cellspacing=0 cellpadding=1 border=0 width='100%'>"
|
||||
"<tr><td align='right' bgcolor='#ffff80'><b> <a href='event=Respiratory Effort'>"+tr("RERA")+"</a></b></td><td width=20% bgcolor='#ffff80'>"+QString().sprintf("%.2f",rei)+"</td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#404040'><b> <font color='white'><a href='event=Flow Limit'>"+tr("Flow Limit")+"</a></font></b></td><td bgcolor='#404040'><font color='white'>"+a.sprintf("%.2f",fli)+"</font></td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#ff4040'><b> <a href='event=Vibratory snore'>"+tr("Vsnore")+"</a></b></td><td bgcolor='#ff4040'>"+QString().sprintf("%.2f",vsi)+"</td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#80ff80'><b> <a href='event=Cheyne Stokes'>"+tr("PB/CSR")+"</a></b></td><td bgcolor='#80ff80'>"+QString().sprintf("%.2f",csr)+"%</td></tr>\n"
|
||||
"</table></td>";
|
||||
} else if (cpap->machine->GetClass()=="Intellipap") {
|
||||
html+="<td colspan=2><table cellspacing=0 cellpadding=2 border=0 width='100%'>"
|
||||
"<tr><td align='right' bgcolor='#ffff80'><b><a href='event=NRI'>"+tr("NRI")+"</a></b></td><td bgcolor='#ffff80'>"+QString().sprintf("%.2f",nri)+"</td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#ffff80'><b><a href='event=NRI'>"+tr("NRI")+"</a></b></td><td width=20% bgcolor='#ffff80'>"+QString().sprintf("%.2f",nri)+"</td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#404040'><b><font color='white'><a href='event=Leak'>"+tr("Leak Idx")+"</a></font></b></td><td bgcolor='#404040'><font color='white'>"+a.sprintf("%.2f",lki)+"</font></td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#ff4040'><b><a href='event=VSnore'>"+tr("Vibratory Snore")+"</a></b></td><td bgcolor='#ff4040'>"+QString().sprintf("%.2f",vsi)+"</td></tr>\n"
|
||||
"<tr><td align='right' bgcolor='#80ff80'><b><a href='event=ExP'>"+tr("Exhalation Puff")+"</a></b></td><td bgcolor='#80ff80'>"+QString().sprintf("%.2f",exp)+"%</td></tr>\n"
|
||||
|
@ -18,7 +18,7 @@ p,a,td,body { font-size: 14px }
|
||||
<li>ResMed S9 models</li>
|
||||
<li>DeVilbiss Intellipap models (*new)</li>
|
||||
<b>Oximetry</b>
|
||||
<li>Contec CMS50 Oximeters (rather poorly still I'm afraid)</li>
|
||||
<li>Contec CMS50 Oximeters (*improved)</li>
|
||||
<p>I don't recommend using this built in "web browser" to do any major surfing in, it will work, but it's mainly meant as a help browser.
|
||||
(It doesn't support SSL encryption.)</p>
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user