diff --git a/Graphs/gSummaryChart.cpp b/Graphs/gSummaryChart.cpp index c0e1d1fa..08b1dae7 100644 --- a/Graphs/gSummaryChart.cpp +++ b/Graphs/gSummaryChart.cpp @@ -167,7 +167,7 @@ void SummaryChart::SetDay(Day * nullday) if (code==CPAP_Pressure) { if ((cpapmode>MODE_CPAP) && (mode==MODE_CPAP)) { hascode=false; - if ((type==ST_PERC) && (m_typeval[j]==0.5)) { + if ((type==ST_PERC) && (typeval==0.5)) { type=ST_SETWAVG; hascode=true; } @@ -177,7 +177,7 @@ void SummaryChart::SetDay(Day * nullday) } if (hascode) { m_days[dn]=day; - switch(m_type[j]) { + switch(type) { case ST_AVG: tmp=day->avg(code); break; case ST_SUM: tmp=day->sum(code); break; case ST_WAVG: tmp=day->wavg(code); break; diff --git a/SleepLib/calcs.cpp b/SleepLib/calcs.cpp index 55ee5e1c..e5849c2a 100644 --- a/SleepLib/calcs.cpp +++ b/SleepLib/calcs.cpp @@ -90,10 +90,10 @@ void xpassFilter(EventDataType * input, EventDataType * output, int samples, Eve // prime the first value output[0]=input[0]; - for (int i=1;i3) && ((max-min) > 8) && (len>sps) && (middle > start)) { - breaths.push_back(BreathPeak(min, max, start, peakmax, middle, peakmin, k)); + // peak detection may not be needed.. + breaths.push_back(BreathPeak(min, max, start, middle, k)); //, peakmin, peakmax)); //EventDataType g0=(0-lastc) / (c-lastc); //double d=(m_rate*g0); //double d1=flowstart+ (start*rate); @@ -365,11 +369,11 @@ void FlowParser::calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool EventList * Te=NULL, * Ti=NULL; if (calcTi) { Ti=m_session->AddEventList(CPAP_Ti,EVL_Event); - Ti->setGain(0.1); + Ti->setGain(0.02); } if (calcTe) { Te=m_session->AddEventList(CPAP_Te,EVL_Event); - Te->setGain(0.1); + Te->setGain(0.02); } @@ -425,7 +429,7 @@ void FlowParser::calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool // Calculate Inspiratory Time (Ti) for this breath ///////////////////////////////////////////////////////////////////// if (calcTi) { - ti=(mt-st)/100.0; + ti=((mt-st)/1000.0)*50.0; ti1=(lastti2+lastti+ti)/3.0; Ti->AddEvent(mt,ti1); lastti2=lastti; @@ -435,7 +439,7 @@ void FlowParser::calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool // Calculate Expiratory Time (Te) for this breath ///////////////////////////////////////////////////////////////////// if (calcTe) { - te=(et-mt)/100.0; // (/1000 * 10) + te=((et-mt)/1000.0)*50.0; // Average last three values.. te1=(lastte2+lastte+te)/3.0; Te->AddEvent(mt,te1); @@ -478,42 +482,43 @@ void FlowParser::calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool if (stmin < start) stmin=start; len=et-stmin; - if (len < minute) - continue; - rr=0; - //et2=et; + if (len >= minute) { + //et2=et; - // Step back through last minute and count breaths - for (int i=idx;i>=0;i--) { - st2=start + double(breaths[i].start) * m_rate; - et2=start + double(breaths[i].end) * m_rate; - if (et2 < stmin) - break; + // Step back through last minute and count breaths + for (int i=idx;i>=0;i--) { + st2=start + double(breaths[i].start) * m_rate; + et2=start + double(breaths[i].end) * m_rate; + if (et2 < stmin) + break; - len=et2-st2; - if (st2 < stmin) { - // Partial breath - st2=stmin; - adj=et2 - st2; - b=(1.0 / len) * adj; - } else b=1; - rr+=b; + len=et2-st2; + if (st2 < stmin) { + // Partial breath + st2=stmin; + adj=et2 - st2; + b=(1.0 / len) * adj; + } else b=1; + rr+=b; + } + + // Calculate min & max + if (rr < minrr) + minrr=rr; + if (rr > maxrr) + maxrr=rr; + + // Add manually.. (much quicker) + *rr_tptr++ = timeval; + + // Use the same gains as ResMed.. + + *rr_dptr++ = rr * 5.0; + rr_count++; + + //rr->AddEvent(et,br * 50.0); } - - // Calculate min & max - if (rr < minrr) minrr=rr; - if (rr > maxrr) maxrr=rr; - - // Add manually.. (much quicker) - *rr_tptr++ = timeval; - - // Use the same gains as ResMed.. - - *rr_dptr++ = rr * 5.0; - rr_count++; - - //rr->AddEvent(et,br * 50.0); } if (calcMv && calcResp && calcTv) { mv=(tv/1000.0) * rr; @@ -659,129 +664,6 @@ void FlowParser::flagEvents() } } } - - - -/* QVector good; - - - good.reserve(numbreaths); - bool bad=false; - int bs,bm,be; - - for (int idx=0;idx cutoffval) continue; - - int j=bs; - for (;j>0;j--) { - if (qAbs(m_filtered[j]) > cutoffval) { - bs=j; - break; - } - } - - if (bs==be) continue; - j=be; - for (;j cutoffval) { - be=j; - break; - } - } - - - st=start + bs * m_rate; - mt=start + bm * m_rate; - et=start + be * m_rate; - - len=et-st; - dur=len/1000.0; - if (dur>=duration) { - //if (!SearchApnea(m_session,st-len/2,15000)) { - if (!uf1) { - uf1=m_session->AddEventList(CPAP_UserFlag1,EVL_Event); - } - uf1->AddEvent(et-len/2,dur); - //} - } - - // Uncomment to use UserFlags to show waveform crossover points - // Good for debugging this stuff. (Make sure to add the EventLists up above) - - //if (val > cutoffval) { - //uf2->AddEvent(st,0); - //uf2->AddEvent(mt,0); - //uf3->AddEvent(et,0); - lastet=et; - lastst=st; - //} - - - } - return; - -*/ - //EventList *uf1=NULL; - -// int lastbad=-1; -// qint64 firstbad=0; - -// bool fr=false; // flow restriction -// for (int i=0;i cutoffval) -// break; -// fr=true; -// et=start + breaths[j].end * m_rate; -// } - - -// if (fr) { -// i=j-1; // rewind - -// len=et-st; -// dur=(len) / 1000.0; - -// if (dur >= duration) { -// if (!uf1) { -// uf1=m_session->AddEventList(CPAP_UserFlag1,EVL_Event); -// } -// uf1->AddEvent(et-(len/2),dur); -// } -// } -// } - - - } void calcRespRate(Session *session, FlowParser * flowparser) @@ -812,8 +694,11 @@ void calcRespRate(Session *session, FlowParser * flowparser) bool calcMv=!session->eventlist.contains(CPAP_MinuteVent); + int z=(calcResp ? 1 : 0) + (calcTv ? 1 : 0) + (calcMv ? 1 : 0); + // If any of these three missing, remove all, and switch all on - if (!(calcResp & calcTv & calcMv)) { + if (z>0 && z<3) { + if (!calcResp && !calcTv && !calcMv) calcTv=calcMv=calcResp=true; QVector & list=session->eventlist[CPAP_RespRate]; @@ -904,6 +789,8 @@ int calcAHIGraph(Session *session) f; EventList *AHI=new EventList(EVL_Event); + + AHI->setGain(0.02); session->eventlist[CPAP_AHI].push_back(AHI); EventDataType ahi; @@ -931,7 +818,7 @@ int calcAHIGraph(Session *session) ahi = events / hours; - AHI->AddEvent(t,ahi); + AHI->AddEvent(t,ahi * 50); avg+=ahi; cnt++; } @@ -951,7 +838,7 @@ int calcAHIGraph(Session *session) ahi=calcAHI(session,f,ti); avg+=ahi; cnt++; - AHI->AddEvent(ti,ahi); + AHI->AddEvent(ti,ahi * 50); lastti=ti; ti+=window_step; } diff --git a/SleepLib/calcs.h b/SleepLib/calcs.h index 9e619f9d..a34cb054 100644 --- a/SleepLib/calcs.h +++ b/SleepLib/calcs.h @@ -43,24 +43,24 @@ struct Filter { }; struct BreathPeak { - BreathPeak() { min=0; max=0; start=0; peakmax=0; middle=0; peakmin=0; end=0; } - BreathPeak(EventDataType _min, EventDataType _max, qint32 _start, qint64 _peakmax, qint32 _middle, qint64 _peakmin, qint32 _end) { + BreathPeak() { min=0; max=0; start=0; middle=0; end=0; } // peakmin=0; peakmax=0; } + BreathPeak(EventDataType _min, EventDataType _max, qint32 _start, qint32 _middle, qint32 _end) {//, qint64 _peakmin, qint64 _peakmax) { min=_min; max=_max; start=_start; middle=_middle; end=_end; - peakmax=_peakmax; - peakmin=_peakmin; + //peakmax=_peakmax; + //peakmin=_peakmin; } BreathPeak(const BreathPeak & copy) { min=copy.min; max=copy.max; start=copy.start; - peakmax=copy.peakmax; middle=copy.middle; - peakmin=copy.peakmin; end=copy.end; + //peakmin=copy.peakmin; + //peakmax=copy.peakmax; } int samplelength() { return end-start; } int upperLength() { return middle-start; } @@ -69,10 +69,10 @@ struct BreathPeak { EventDataType min; // peak value EventDataType max; // peak value qint32 start; // beginning zero cross - qint64 peakmax; // max peak index qint32 middle; // ending zero cross - qint64 peakmin; // min peak index qint32 end; // ending zero cross + //qint64 peakmin; // min peak index + //qint64 peakmax; // max peak index }; bool operator<(const BreathPeak & p1, const BreathPeak & p2); diff --git a/SleepLib/event.cpp b/SleepLib/event.cpp index 5f985086..717b5a57 100644 --- a/SleepLib/event.cpp +++ b/SleepLib/event.cpp @@ -47,6 +47,41 @@ EventDataType EventList::data2(quint32 i) return EventDataType(m_data2[i]); } +void EventList::AddEvent(qint64 time, EventStoreType data) +{ + m_data.push_back(data); + + // Apply gain & offset + EventDataType val=EventDataType(data)*m_gain; // ignoring m_offset + + if (m_update_minmax) { + if (m_min>val) m_min=val; + else if (m_maxtime) { + // Crud.. Update all the previous records + // This really shouldn't happen. + + qint32 t=(m_first-time); + for (quint32 i=0;ival) m_min=val; - if (m_max val) min=val; + if (max < val) max=val; } + m_min=min; + m_max=max; } else { - for (int i=0;ival) m_min=val; if (m_maxval) m_min=val; if (m_maxupdateFirst(starttime); diff --git a/SleepLib/loader_plugins/prs1_loader.cpp b/SleepLib/loader_plugins/prs1_loader.cpp index 20b6b267..5d41fcac 100644 --- a/SleepLib/loader_plugins/prs1_loader.cpp +++ b/SleepLib/loader_plugins/prs1_loader.cpp @@ -40,6 +40,7 @@ extern QProgressBar *qprogress; QHash ModelMap; #define PRS1_CRC_CHECK + #ifdef PRS1_CRC_CHECK typedef quint16 crc_t; @@ -359,9 +360,10 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile) } // sessions are fully loaded here.. - cnt++; - if (qprogress) qprogress->setValue(0.0+(float(cnt)/float(size)*100.0)); - QApplication::processEvents(); + if ((++cnt % 10)==0) { + if (qprogress) qprogress->setValue(0.0+(float(cnt)/float(size)*100.0)); + QApplication::processEvents(); + } } // strictly can do this in the above loop, but this is cautionary cnt=0; @@ -417,9 +419,10 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile) sess->SetChanged(true); m->AddSession(sess,profile); - cnt++; - if (qprogress) qprogress->setValue(33.0+(float(cnt)/float(size)*33.0)); - QApplication::processEvents(); + if ((++cnt % 10) ==0) { + //if (qprogress) qprogress->setValue(33.0+(float(cnt)/float(size)*33.0)); + QApplication::processEvents(); + } } @@ -1029,34 +1032,38 @@ bool PRS1Loader::Parse002(qint32 sequence, quint32 timestamp, unsigned char *buf } -bool PRS1Loader::ParseWaveform(qint32 sequence, quint32 timestamp, unsigned char *data, quint16 size, quint16 duration, quint16 num_signals, quint16 interleave, quint8 sample_format) -{ - if (!new_sessions.contains(sequence)) - return false; +//bool PRS1Loader::ParseWaveform(qint32 sequence, quint32 timestamp, unsigned char *data, quint16 size, quint16 duration, quint16 num_signals, quint16 interleave, quint8 sample_format) +//{ +// Q_UNUSED(interleave) +// Q_UNUSED(sample_format) +// // this whole function is currently unused.. - qint64 t=qint64(timestamp)*1000L; +// if (!new_sessions.contains(sequence)) +// return false; - Session *session=new_sessions[sequence]; +// qint64 t=qint64(timestamp)*1000L; - if (num_signals==1) { - session->updateFirst(t); - double d=duration*1000; - EventDataType rate=d / EventDataType(size); - EventList *ev=session->AddEventList(CPAP_FlowRate,EVL_Waveform,1.0,0.00,0,0,rate); - ev->AddWaveform(t,(char *)data,size,qint64(duration)*1000L); - session->updateLast(t+qint64(duration)*1000L); +// Session *session=new_sessions[sequence]; - } +// if (num_signals==1) { +// session->updateFirst(t); +// double d=duration*1000; +// EventDataType rate=d / EventDataType(size); +// EventList *ev=session->AddEventList(CPAP_FlowRate,EVL_Waveform,1.0,0.00,0,0,rate); +// ev->AddWaveform(t,(char *)data,size,qint64(duration)*1000L); +// session->updateLast(t+qint64(duration)*1000L); - return true; -} +// } + +// return true; +//} bool PRS1Loader::OpenFile(Machine *mach, QString filename) { int sequence,version; quint32 timestamp; qint64 pos; - unsigned char ext,htype,sum; + unsigned char ext,sum, htype; unsigned char *header,*data; int chunk,hl; quint16 size,datasize,c16,crc; @@ -1090,6 +1097,7 @@ bool PRS1Loader::OpenFile(Machine *mach, QString filename) size=(header[2] << 8) | header[1]; htype=header[3]; // 00 = normal // 01=waveform // could be a bool? + Q_UNUSED(htype); version=header[4]; // == 5 ext=header[6]; sequence=(header[10] << 24) | (header[9] << 16) | (header[8] << 8) | header[7]; @@ -1097,6 +1105,7 @@ bool PRS1Loader::OpenFile(Machine *mach, QString filename) if (ext==5) { duration=header[0xf] | header[0x10] << 8; // block duration in seconds + Q_UNUSED(duration); num_signals=header[0x12] | header[0x13] << 8; if (num_signals>2) { qWarning() << "More than 2 Waveforms in " << filename; @@ -1265,6 +1274,8 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename) } length=m_buffer[pos+0x1] | m_buffer[pos+0x2] << 8; // block length in bytes duration=m_buffer[pos+0xf] | m_buffer[pos+0x10] << 8; // block duration in seconds + + if (diff<0) { qDebug() << "Padding waveform to keep sync" << block; //diff=qAbs(diff); diff --git a/SleepLib/loader_plugins/prs1_loader.h b/SleepLib/loader_plugins/prs1_loader.h index e3989ea8..da68a18f 100644 --- a/SleepLib/loader_plugins/prs1_loader.h +++ b/SleepLib/loader_plugins/prs1_loader.h @@ -79,8 +79,8 @@ protected: //! \brief Parse a .005 waveform file, extracting Flow Rate waveform (and Mask Pressure data if available) bool OpenWaveforms(SessionID sid, QString filename); - //! \brief ParseWaveform chunk.. Currently unused, as the old one works fine. - bool ParseWaveform(qint32 sequence, quint32 timestamp, unsigned char *data, quint16 size, quint16 duration, quint16 num_signals, quint16 interleave, quint8 sample_format); + // //! \brief ParseWaveform chunk.. Currently unused, as the old one works fine. + //bool ParseWaveform(qint32 sequence, quint32 timestamp, unsigned char *data, quint16 size, quint16 duration, quint16 num_signals, quint16 interleave, quint8 sample_format); //! \brief Parse a data chunk from the .000 (brick) and .001 (summary) files. bool ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, unsigned char *data, quint16 size, char version); diff --git a/SleepLib/loader_plugins/resmed_loader.cpp b/SleepLib/loader_plugins/resmed_loader.cpp index f108f912..287f715b 100644 --- a/SleepLib/loader_plugins/resmed_loader.cpp +++ b/SleepLib/loader_plugins/resmed_loader.cpp @@ -584,9 +584,11 @@ int ResmedLoader::Open(QString & path,Profile *profile) // Push current filename to ordered-by-sessionid list sessfiles[sessionid].push_back(filename); - // Update the progress bar - if (qprogress) qprogress->setValue((float(i+1)/float(size)*10.0)); - QApplication::processEvents(); + if ((i%10) ==0) { + // Update the progress bar + if (qprogress) qprogress->setValue((float(i+1)/float(size)*10.0)); + QApplication::processEvents(); + } } QString fn; @@ -932,6 +934,7 @@ int ResmedLoader::Open(QString & path,Profile *profile) ///////////////////////////////////////////////////////////////////// // Duration and Event Indices ///////////////////////////////////////////////////////////////////// + dur=0; if ((sig=stredf.lookupName("Mask Dur"))) { dur=sig->data[dn]*sig->gain; dur/=60.0f; // convert to hours. @@ -1044,8 +1047,10 @@ int ResmedLoader::Open(QString & path,Profile *profile) else if (fn=="brp") LoadBRP(sess,edf); else if (fn=="sad") LoadSAD(sess,edf); } - if (qprogress) qprogress->setValue(10.0+(float(++cnt)/float(size)*90.0)); - QApplication::processEvents(); + if ((++cnt%10) ==0) { + if (qprogress) qprogress->setValue(10.0+(float(cnt)/float(size)*90.0)); + QApplication::processEvents(); + } if (!sess) continue; if (!sess->first()) { @@ -1342,9 +1347,6 @@ bool ResmedLoader::LoadBRP(Session *sess,EDFParser &edf) //qDebug() << "BRP:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum; long recs=es.nr*edf.GetNumDataRecords(); ChannelID code; - if (es.offset>0) { - int i=5; - } if (es.label=="Flow") { es.gain*=60; es.physical_dimension="L/M"; @@ -1372,37 +1374,43 @@ EventList * ResmedLoader::ToTimeDelta(Session *sess,EDFParser &edf, EDFSignal & QElapsedTimer time; time.start(); #endif - bool first=true; + + double rate=(duration/recs); // milliseconds per record double tt=edf.startdate; //sess->UpdateFirst(tt); EventDataType c,last; - 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)) { startpos=20; // Shave the first 20 seconds of pressure data tt+=rate*startpos; } - for (int i=startpos;istartpos+1) { + el=sess->AddEventList(code,EVL_Event,es.gain,es.offset,min,max); + c=last=*sptr++; + el->AddEvent(tt,last); + + for (; sptr < eptr; sptr++) { //int i=startpos;iAddEvent(tt,c); - first=false; - } else { if (last!=c) { if (square) el->AddEvent(tt,last); // square waves look better on some charts. el->AddEvent(tt,c); } - } - tt+=rate; + tt+=rate; - last=c; + last=c; + } + el->AddEvent(tt,c); + sess->updateLast(tt); } - el->AddEvent(tt,c); - sess->updateLast(tt); #ifdef DEBUG_EFFICIENCY @@ -1472,15 +1480,12 @@ bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf) sess->updateLast(edf.startdate+duration); QString t; int emptycnt=0; - EventList *a; + EventList *a=NULL; double rate; long recs; ChannelID code; for (int s=0;s0) { - int i=5; - } recs=es.nr*edf.GetNumDataRecords(); if (recs<=0) continue; rate=double(duration)/double(recs); diff --git a/SleepLib/machine.cpp b/SleepLib/machine.cpp index 2ef04553..a6f17278 100644 --- a/SleepLib/machine.cpp +++ b/SleepLib/machine.cpp @@ -268,6 +268,7 @@ bool Machine::Load() ext_s=fi.fileName().section(".",-1); ext=ext_s.toInt(&ok,10); if (!ok) continue; + sesstr=fi.fileName().section(".",0,-2); sessid=sesstr.toLong(&ok,16); if (!ok) continue; @@ -278,12 +279,10 @@ bool Machine::Load() int size=sessfiles.size(); int cnt=0; for (s=sessfiles.begin(); s!=sessfiles.end(); s++) { - cnt++; - if ((cnt % 10)==0) { + if ((++cnt % 50)==0) { // This is slow.. :-/ if (qprogress) qprogress->setValue((float(cnt)/float(size)*100.0)); - + QApplication::processEvents(); } - QApplication::processEvents(); Session *sess=new Session(this,s.key()); @@ -330,8 +329,10 @@ bool Machine::Save() savelistSize=m_savelist.size(); if (!PROFILE.session->multithreading()) { for (int i=0;isetValue(0+(float(savelistCnt)/float(savelistSize)*100.0)); - QApplication::processEvents(); + if ((i % 10) ==0) { + qprogress->setValue(0+(float(savelistCnt)/float(savelistSize)*100.0)); + QApplication::processEvents(); + } Session *s=m_savelist.at(i); s->UpdateSummaries(); s->Store(path); diff --git a/SleepLib/session.cpp b/SleepLib/session.cpp index 5422cc27..b5505724 100644 --- a/SleepLib/session.cpp +++ b/SleepLib/session.cpp @@ -215,6 +215,7 @@ bool Session::LoadSummary(QString filename) QHash cruft; if (version<7) { + // This code is deprecated.. just here incase anyone tries anything crazy... QHash v1; in >> v1; settings.clear(); @@ -320,7 +321,8 @@ bool Session::LoadSummary(QString filename) } - if (version= first) { if (t<=last) { @@ -977,7 +988,7 @@ double Session::rangeSum(ChannelID id, qint64 first,qint64 last) double sum=0,gain; qint64 t,start; - EventStoreType * dptr; + EventStoreType * dptr, * eptr; quint32 * tptr; int cnt,idx; @@ -989,6 +1000,7 @@ double Session::rangeSum(ChannelID id, qint64 first,qint64 last) start=ev.first(); dptr=ev.rawData(); cnt=ev.count(); + eptr=dptr+cnt; gain=ev.gain(); rate=ev.rate(); if (ev.type()==EVL_Waveform) { @@ -996,24 +1008,26 @@ double Session::rangeSum(ChannelID id, qint64 first,qint64 last) // Skip the samples before first idx=(first - ev.first()) / rate; } + + dptr+=idx; //???? foggy. + t=start; - for (int j=idx;j= first) { if (t <= last) { sum+=EventDataType(*dptr) * gain; } else break; } - dptr++; } } } @@ -1029,7 +1043,7 @@ EventDataType Session::rangeMin(ChannelID id, qint64 first,qint64 last) EventDataType gain,v,min=999999999; qint64 t,start,rate; - EventStoreType * dptr; + EventStoreType * dptr, * eptr; quint32 * tptr; int cnt,idx; @@ -1041,6 +1055,7 @@ EventDataType Session::rangeMin(ChannelID id, qint64 first,qint64 last) dptr=ev.rawData(); start=ev.first(); cnt=ev.count(); + eptr=dptr+cnt; gain=ev.gain(); if (ev.type()==EVL_Waveform) { @@ -1052,19 +1067,19 @@ EventDataType Session::rangeMin(ChannelID id, qint64 first,qint64 last) // Skip the samples before first idx=(first - ev.first())/rate; } + dptr+=idx; - for (int j=idx;j= first) { if (t <= last) { @@ -1073,7 +1088,6 @@ EventDataType Session::rangeMin(ChannelID id, qint64 first,qint64 last) min=v; } else break; } - dptr++; } } } @@ -1089,7 +1103,7 @@ EventDataType Session::rangeMax(ChannelID id, qint64 first,qint64 last) EventDataType gain,v,max=-999999999; qint64 t,start,rate; - EventStoreType * dptr; + EventStoreType * dptr, * eptr; quint32 * tptr; int cnt,idx; @@ -1101,6 +1115,7 @@ EventDataType Session::rangeMax(ChannelID id, qint64 first,qint64 last) start=ev.first(); dptr=ev.rawData(); cnt=ev.count(); + eptr=dptr + cnt; gain=ev.gain(); if (ev.type()==EVL_Waveform) { rate=ev.rate(); @@ -1111,17 +1126,17 @@ EventDataType Session::rangeMax(ChannelID id, qint64 first,qint64 last) // Skip the samples before first idx=(first - ev.first())/rate; } - for (int j=idx;jmax) max=v; } else break; - dptr++; t+=rate; } } else { tptr=ev.rawTime(); - for (int j=0;j=first) { if (t<=last) { @@ -1129,7 +1144,6 @@ EventDataType Session::rangeMax(ChannelID id, qint64 first,qint64 last) if (v>max) max=v; } else break; } - dptr++; } } } @@ -1171,15 +1185,16 @@ double Session::sum(ChannelID id) QVector & evec=j.value(); double gain,sum=0; - EventStoreType * dptr; + EventStoreType * dptr, * eptr; int cnt; for (int i=0;igain(); - EventStoreType * dptr, * sptr; + EventStoreType * dptr, * sptr, *eptr; int tt=0,cnt; for (int i=0;iraw(j)); + eptr=sptr+cnt; + for (; sptr < eptr; sptr++) { + *dptr++ = * sptr; } } diff --git a/daily.cpp b/daily.cpp index 590a4029..e6794b73 100644 --- a/daily.cpp +++ b/daily.cpp @@ -105,13 +105,13 @@ Daily::Daily(QWidget *parent,gGraphView * shared) RR=new gGraph(GraphView,tr("Resp. Rate"),schema::channel[CPAP_RespRate].description()+"\n("+schema::channel[CPAP_RespRate].units()+")",default_height); TV=new gGraph(GraphView,tr("Tidal Volume"),schema::channel[CPAP_TidalVolume].description()+"\n("+schema::channel[CPAP_TidalVolume].units()+")",default_height); MV=new gGraph(GraphView,tr("Minute Vent."),schema::channel[CPAP_MinuteVent].description()+"\n("+schema::channel[CPAP_MinuteVent].units()+")",default_height); + TgMV=new gGraph(GraphView,tr("Tgt. Min. Vent"),schema::channel[CPAP_TgMV].description()+"\n("+schema::channel[CPAP_TgMV].units()+")",default_height); FLG=new gGraph(GraphView,tr("Flow Limitation"),schema::channel[CPAP_FLG].description()+"\n("+schema::channel[CPAP_FLG].units()+")",default_height); PTB=new gGraph(GraphView,tr("Pat. Trig. Breath"),schema::channel[CPAP_PTB].description()+"\n("+schema::channel[CPAP_PTB].units()+")",default_height); RE=new gGraph(GraphView,tr("Resp. Event"),schema::channel[CPAP_RespEvent].description()+"\n("+schema::channel[CPAP_RespEvent].units()+")",default_height); + TI=new gGraph(GraphView,tr("Insp. Time"),schema::channel[CPAP_Ti].description()+"\n("+schema::channel[CPAP_Ti].units()+")",default_height); + TE=new gGraph(GraphView,tr("Exp. Time"),schema::channel[CPAP_Te].description()+"\n("+schema::channel[CPAP_Te].units()+")",default_height); IE=new gGraph(GraphView,tr("IE"),schema::channel[CPAP_IE].description()+"\n("+schema::channel[CPAP_IE].units()+")",default_height); - TE=new gGraph(GraphView,tr("Te"),schema::channel[CPAP_Te].description()+"\n("+schema::channel[CPAP_Te].units()+")",default_height); - TI=new gGraph(GraphView,tr("Ti"),schema::channel[CPAP_Ti].description()+"\n("+schema::channel[CPAP_Ti].units()+")",default_height); - TgMV=new gGraph(GraphView,tr("Tgt. Min. Vent"),schema::channel[CPAP_TgMV].description()+"\n("+schema::channel[CPAP_TgMV].units()+")",default_height); int oxigrp=PROFILE.ExistsAndTrue("SyncOximetry") ? 0 : 1; PULSE=new gGraph(GraphView,STR_TR_PulseRate,schema::channel[OXI_Pulse].description()+"\n("+schema::channel[OXI_Pulse].units()+")",default_height,oxigrp); @@ -136,6 +136,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared) evseg->AddSlice(CPAP_RERA,QColor(0xff,0xff,0x80,0xff),tr("RE")); evseg->AddSlice(CPAP_NRI,QColor(0x00,0x80,0x40,0xff),tr("NR")); evseg->AddSlice(CPAP_FlowLimit,QColor(0x40,0x40,0x40,0xff),tr("FL")); + //evseg->AddSlice(CPAP_UserFlag1,QColor(0x40,0x40,0x40,0xff),tr("UF")); GAHI->AddLayer(AddCPAP(evseg)); GAHI->setMargins(0,0,0,0); @@ -254,6 +255,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared) // lc->addPlot(CPAP_Test1,Qt::darkRed,square); MV->AddLayer(AddCPAP(new gLineChart(CPAP_MinuteVent,Qt::darkCyan,square))); + TV->AddLayer(AddCPAP(lc=new gLineChart(CPAP_TidalVolume,Qt::magenta,square))); //lc->addPlot(CPAP_Test2,Qt::darkYellow,square); @@ -825,10 +827,6 @@ void Daily::Load(QDate date) html+=QString("%3%4%5%\n") .arg(schema::channel[CPAP_NRI].defaultColor().name()).arg("black").arg(tr("NRI")).arg(schema::channel[CPAP_NRI].description()).arg(nri,0,'f',2).arg(CPAP_NRI); } - if (cpap->machine->GetClass()==STR_MACH_PRS1) { - html+=QString("%3%4%5%\n") - .arg("#80ff80").arg("black").arg(tr("PB/CSR")).arg(schema::channel[CPAP_CSR].description()).arg(csr,0,'f',2).arg(CPAP_CSR); - } if (PROFILE.cpap->userEventFlagging()) { EventDataType uf1=cpap->count(CPAP_UserFlag1) / cpap->hours(); if (uf1>0) @@ -852,10 +850,13 @@ void Daily::Load(QDate date) html+=QString("%3%4%5\n") .arg("#ff4040").arg("black").arg(tr("VSnore")).arg(schema::channel[CPAP_VSnore].description()).arg(cpap->count(CPAP_VSnore)/cpap->hours(),0,'f',2).arg(CPAP_VSnore); } else { - html+=" "; + //html+=" "; html+=QString("%3%4%5\n") .arg("#ff4040").arg("black").arg(tr("VSnore2")).arg(schema::channel[CPAP_VSnore2].description()).arg(cpap->count(CPAP_VSnore2)/cpap->hours(),0,'f',2).arg(CPAP_VSnore2); } + html+=QString("%3%4%5%\n") + .arg("#80ff80").arg("black").arg(tr("PB/CSR")).arg(schema::channel[CPAP_CSR].description()).arg(csr,0,'f',2).arg(CPAP_CSR); + html+=""; } else if (cpap->machine->GetClass()==STR_MACH_Intellipap) { html+=""; diff --git a/docs/channels.xml b/docs/channels.xml index 8e8bbdaf..3fd91bce 100644 --- a/docs/channels.xml +++ b/docs/channels.xml @@ -42,8 +42,8 @@ Important: One id code per item, DO NOT CHANGE ID NUMBERS!!! - - + + diff --git a/docs/release_notes.html b/docs/release_notes.html index 212f8fda..ea0225b6 100644 --- a/docs/release_notes.html +++ b/docs/release_notes.html @@ -24,6 +24,7 @@
  • Option to automatically maintain backup folder for ResMed users. (on by default)
  • Compression options to save disk space for SleepyHead data and backups.
  • Rewritten Flow Rate Calculations Module gives MinuteVent, RespiratoryRate, TidalVolume, Te & Ti to machines where it's missing, (provided Flow Waveform is available) and is much more accurate than before.
  • +
  • User Event flagging now works on ResMed machines too. It must be switched on before import.
  • New Context cube to make empty pages more attractive.. Yes you can switch it off. No it doesn't take much resources.
  • Plenty of other bug fixes, including more oximetry fixes.
  • @@ -36,7 +37,8 @@ The hard stuffs done, and everything needs time to settle and have the bugs knocked out of it, however there still are some minor tweaks and features in the pipeline...

  • Lots more Performance Optimizations
  • -
  • Graphical tweaks, animations & improvements
  • +
  • Better custom event flagging (including a workaround for the dreaded ResMed S9 drift bug)
  • +
  • More graphical tweaks, animations & improvements
  • Improving the quality of printed report data, and showing more data and statistics.
  • Improving the CSV export feature, and making it more customizable
  • diff --git a/preferencesdialog.ui b/preferencesdialog.ui index 28069f55..72c0103e 100644 --- a/preferencesdialog.ui +++ b/preferencesdialog.ui @@ -10,7 +10,7 @@ 0 0 640 - 407 + 414 @@ -42,7 +42,7 @@ - 0 + 1 @@ -312,24 +312,26 @@ p, li { white-space: pre-wrap; } This maintains a backup of SD-card data for ResMed machines, -ResMed machines delete high resolution data older than 7 days, and graph data older than 30 days.. +ResMed machines delete high resolution data older than 7 days, +and graph data older than 30 days.. Sleepyhead can keep a copy of this data if you ever need to reinstall. -(Highly recomended, unless your short on disk space and don't care about the graph data) +(Highly recomended, unless your short on disk space or don't care about the graph data) - Create SD Card Backups during Import (only works with ResMed for now) + Create SD Card Backups during Import (only for ResMed so far, highly recommended) - This makes SleepyHead's data take less disk space (about half as much), -but it makes import take longer, and also makes switching between days a little slower. + This makes SleepyHead's data take around half as much space. +But it makes import and day changing take longer.. +If you've got a new computer with a small solid state disk, this is a good option. - Compress Session Data (slower, but makes SleepyHead data smaller) + Compress Session Data (makes SleepyHead data smaller, but day changing slower.) @@ -344,7 +346,7 @@ SleepyHead can import from this compressed backup directory natively.. To use with ResScan will require the .gz files to be uncompressed first.. - Compress SD Card Backups (slower, but makes backups smaller) + Compress SD Card Backups (slower first import, but makes backups smaller) @@ -370,7 +372,7 @@ To use with ResScan will require the .gz files to be uncompressed first.. +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Note: <span style=" font-style:italic;">Compression options don't automatically recompress already saved data. (yet)</span></p></body></html> @@ -690,9 +692,8 @@ p, li { white-space: pre-wrap; } Enable/disable experimental event flagging enhancements. -It allows detecting borderline events on PRS1 machines.. - -Has no effect on other machines (yet). +It allows detecting borderline events, and some the machine missed. +This option must be enabled before import, otherwise a purge is required. Custom PRS1 Event Flagging @@ -729,7 +730,8 @@ Has no effect on other machines (yet). - Percentage of restriction in airflow + Percentage of restriction in airflow from the median value. +A value of 20% works well for detecting apneas. % @@ -833,7 +835,8 @@ p, li { white-space: pre-wrap; } - Adjusts the amount of data considered for each point in the AHI/Hour graph. + Adjusts the amount of data considered for each point in the AHI/Hour graph. +Defaults to 60 minutes.. Highly recommend it's left at this value. minutes @@ -852,7 +855,8 @@ p, li { white-space: pre-wrap; } - Reset the counter to zero at beginning of each (time) window + Reset the counter to zero at beginning of each (time) window. +Note: Unless you purge and reimport after changing this, you won't see the changes. Zero Reset @@ -1089,6 +1093,9 @@ p, li { white-space: pre-wrap; } 0 + + Tries to forces the oximetry data to link with CPAP when possible. + Link Oximetry and CPAP graphs @@ -1433,6 +1440,9 @@ Mainly affects the importer. + + Sourceforge hosts this project for free.. Please be considerate of their resources.. + 90 @@ -1613,7 +1623,8 @@ Mainly affects the importer. - The visual method of displaying waveform overlay flags. + The visual method of displaying waveform overlay flags. + @@ -1725,7 +1736,8 @@ this application to be unstable with this feature enabled. - Turn on/off the spinning "context" cube. + Turn on/off the spinning "context" cube. +It really doesn't use that much resources.. :) Animations && Fancy Stuff