diff --git a/SleepLib/calcs.cpp b/SleepLib/calcs.cpp index d05788be..d87fcc54 100644 --- a/SleepLib/calcs.cpp +++ b/SleepLib/calcs.cpp @@ -12,7 +12,7 @@ extern double round(double number); -bool SearchApnea(Session *session, qint64 time, qint64 dist=15000) +bool SearchApnea(Session *session, qint64 time, qint64 dist) { if (session->SearchEvent(CPAP_Obstructive,time,dist)) return true; if (session->SearchEvent(CPAP_Apnea,time,dist)) return true; @@ -22,15 +22,537 @@ bool SearchApnea(Session *session, qint64 time, qint64 dist=15000) return false; } +// Sort BreathPeak by peak index +bool operator<(const BreathPeak & p1, const BreathPeak & p2) +{ + return p1.start < p2.start; +} + +//! \brief Filters input to output with a percentile filter with supplied width. +//! \param samples Number of samples +//! \param width number of surrounding samples to consider +//! \param percentile fractional percentage, between 0 and 1 +void percentileFilter(EventDataType * input, EventDataType * output, int samples, int width, EventDataType percentile) +{ + if (samples<=0) + return; + + if (percentile>1) + percentile=1; + + QVector buf(width); + int s,e; + int z1=width/2; + int z2=z1+(width % 2); + int nm1=samples-1; + //int j; + + // Scan through all of input + for (int k=0;k < samples;k++) { + + s=k-z1; + e=k+z2; + + // Cap bounds + if (s < 0) s=0; + if (e > nm1) e=nm1; + + // + int j=0; + for (int i=s; i < e; i++) { + buf[j++]=input[i]; + } + j--; + + EventDataType val=j * percentile; + EventDataType fl=floor(val); + + // If even percentile, or already max value.. + if ((val==fl) || (j>=width-1)) { + nth_element(buf.begin(),buf.begin()+j,buf.begin()+width-1); + val=buf[j]; + } else { + // Percentile lies between two points, interpolate. + double v1,v2; + nth_element(buf.begin(),buf.begin()+j,buf.begin()+width-1); + v1=buf[j]; + nth_element(buf.begin(),buf.begin()+j+1,buf.begin()+width-1); + v2=buf[j+1]; + + val = v1 + (v2-v1)*(val-fl); + } + output[k]=val; + } +} + +void xpassFilter(EventDataType * input, EventDataType * output, int samples, EventDataType weight) +{ + // prime the first value + output[0]=input[0]; + + for (int i=1;igain(); + m_rate=flow->rate(); + m_samples=flow->count(); + EventStoreType *inraw=flow->rawData(); + + // Make sure we won't overflow internal buffers + if (m_samples > max_filter_buf_size) { + qDebug() << "Error: Sample size exceeds max_filter_buf_size in FlowParser::openFlow().. Capping!!!"; + m_samples=max_filter_buf_size; + } + + // Begin with the second internal buffer + EventDataType * buf=m_filtered; + EventDataType c; + // Apply gain to waveform + for (int i=0;irate(); + + double flowstart=m_flow->first(); + double lasttime,time; + + double peakmax=flowstart, peakmin=flowstart; + + time=lasttime=flowstart; + + // Estimate storage space needed using typical average breaths per minute. + m_minutes=double(m_flow->last() - m_flow->first()) / 60000.0; + const double avgbpm=20; + int guestimate=m_minutes*avgbpm; + breaths_lower.reserve(guestimate); + breaths_upper.reserve(guestimate); + + // Prime min & max, and see which side of the zero line we are starting from. + c=input[0]; + min=max=c; + lastc=c; + m_startsUpper=(c >= zeroline); + + qint32 start=0,middle=0;//,end=0; + + breaths.clear(); + + int sps=1000/m_rate; + // For each samples, find the peak upper and lower breath components + //bool dirty=false; + int len=0, lastk=0; //lastlen=0 + + EventList *uf1=m_session->AddEventList(CPAP_UserFlag1,EVL_Event); + //EventList *uf2=m_session->AddEventList(CPAP_UserFlag2,EVL_Event); + + qint64 sttime=time;//, ettime=time; + + for (int k=0; k < samples; k++) { + c=input[k]; + // dirty=false; + + if (c >= zeroline) { + + // Did we just cross the zero line going up? + if (lastc < zeroline) { + // This helps filter out dirty breaths.. + len=k-start; + if ((max>3) && ((max-min) > 8) && (len>sps) && (middle > start)) { + breaths.push_back(BreathPeak(min, max, start, peakmax, middle, peakmin, k)); + //EventDataType g0=(0-lastc) / (c-lastc); + //double d=(m_rate*g0); + double d1=flowstart+ (start*rate); + //double d2=flowstart+ (k*rate); + + uf1->AddEvent(d1,0); + //uf2->AddEvent(d2,0); + //lastlen=k-start; + // Set max for start of the upper breath cycle + max=c; + peakmax=time; + // Starting point of next breath cycle + start=k; + sttime=time; + } //else { +// dirty=true; +// lastc=-1; +// } + } else if (c > max) { + // Update upper breath peak + max=c; + peakmax=time; + } + } + + if (c < zeroline) { + // Did we just cross the zero line going down? + + if (lastc >= zeroline) { + // Set min for start of the lower breath cycle + min=c; + peakmin=time; + middle=k; + + } else if (c < min) { + // Update lower breath peak + min=c; + peakmin=time; + } + + } + lasttime=time; + time+=rate; + //if (!dirty) + lastc=c; + lastk=k; + } +} + + +//! \brief Calculate Respiratory Rate, TidalVolume, Minute Ventilation, Ti & Te.. +// These are grouped together because, a) it's faster, and b) some of these calculations rely on others. +void FlowParser::calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool calcMv) +{ + if (!m_session) { + return; + } + + // Don't even bother if only a few breaths in this chunk + const int lowthresh=4; + int nm=breaths.size(); + if (nm < lowthresh) + return; + + const qint64 minute=60000; + + + double start=m_flow->first(); + double time=start; + + int bs,be,bm; + double st,et,mt; + + ///////////////////////////////////////////////////////////////////////////////// + // Respiratory Rate setup + ///////////////////////////////////////////////////////////////////////////////// + EventDataType minrr,maxrr; + EventList * RR=NULL; + quint32 * rr_tptr=NULL; + EventStoreType * rr_dptr=NULL; + if (calcResp) { + RR=m_session->AddEventList(CPAP_RespRate,EVL_Event); + minrr=RR->Min(), maxrr=RR->Max(); + RR->setFirst(time+minute); + RR->getData().resize(nm); + RR->getTime().resize(nm); + rr_tptr=RR->rawTime(); + rr_dptr=RR->rawData(); + } + + int rr_count=0; + + double len, st2,et2,adj, stmin, b, rr=0; + int idx; + + + + ///////////////////////////////////////////////////////////////////////////////// + // Inspiratory / Expiratory Time setup + ///////////////////////////////////////////////////////////////////////////////// + double lastte2=0,lastti2=0,lastte=0, lastti=0, te, ti, ti1, te1,c; + EventList * Te=NULL, * Ti=NULL; + if (calcTi) { + Ti=m_session->AddEventList(CPAP_Ti,EVL_Event); + Ti->setGain(0.1); + } + if (calcTe) { + Te=m_session->AddEventList(CPAP_Te,EVL_Event); + Te->setGain(0.1); + } + + + ///////////////////////////////////////////////////////////////////////////////// + // Tidal Volume setup + ///////////////////////////////////////////////////////////////////////////////// + EventList * TV; + EventDataType mintv, maxtv, tv; + double val1, val2; + quint32 * tv_tptr; + EventStoreType * tv_dptr; + int tv_count=0; + if (calcTv) { + TV=m_session->AddEventList(CPAP_TidalVolume,EVL_Event); + mintv=TV->Min(), maxtv=TV->Max(); + TV->setFirst(start); + TV->getData().resize(nm); + TV->getTime().resize(nm); + tv_tptr=TV->rawTime(); + tv_dptr=TV->rawData(); + } + ///////////////////////////////////////////////////////////////////////////////// + // Minute Ventilation setup + ///////////////////////////////////////////////////////////////////////////////// + EventList * MV=NULL; + EventDataType mv; + if (calcMv) { + MV=m_session->AddEventList(CPAP_MinuteVent,EVL_Event); + MV->setGain(0.1); + } + + EventDataType sps=(1000.0/m_rate); // Samples Per Second + + + qint32 timeval=0; // Time relative to start + + for (idx=0;idxAddEvent(mt,ti1); + lastti2=lastti; + lastti=ti; + } + ///////////////////////////////////////////////////////////////////// + // Calculate Expiratory Time (Te) for this breath + ///////////////////////////////////////////////////////////////////// + if (calcTe) { + te=(et-mt)/100.0; // (/1000 * 10) + // Average last three values.. + te1=(lastte2+lastte+te)/3.0; + Te->AddEvent(mt,te1); + + lastte2=lastte; + lastte=te; + } + ///////////////////////////////////////////////////////////////////// + // Calculate TidalVolume + ///////////////////////////////////////////////////////////////////// + if (calcTv) { + val1=0, val2=0; + // Scan the upper breath + for (int j=bs;j maxtv) maxtv=tv; + *tv_tptr++ = timeval; + *tv_dptr++ = tv * 10.0; + tv_count++; + + } + + ///////////////////////////////////////////////////////////////////// + // Respiratory Rate Calculations + ///////////////////////////////////////////////////////////////////// + if (calcResp) { + stmin=et-minute; + if (stmin < start) + stmin=start; + len=et-stmin; + if (len < minute) + continue; + + rr=0; + //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; + + 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; + *rr_dptr++ = rr * 50.0; + rr_count++; + + //rr->AddEvent(et,br * 50.0); + } + if (calcMv && calcResp && calcTv) { + mv=(tv/1000.0) * rr; + MV->AddEvent(et,mv * 10.0); + } + } + + ///////////////////////////////////////////////////////////////////// + // Respiratory Rate post filtering + ///////////////////////////////////////////////////////////////////// + + RR->setGain(0.02); + RR->setMin(minrr); + RR->setMax(maxrr); + RR->setFirst(start); + RR->setLast(et); + RR->setCount(rr_count); + + ///////////////////////////////////////////////////////////////////// + // Tidal Volume post filtering + ///////////////////////////////////////////////////////////////////// + + TV->setGain(0.1); + TV->setMin(mintv); + TV->setMax(maxtv); + TV->setFirst(start); + TV->setLast(et); + TV->setCount(tv_count); + +} + + +/* // Support function for calcRespRate() int filterFlow(Session *session, EventList *in, EventList *out, EventList *tv, EventList *mv, double rate) { - int size=in->count(); - EventDataType *stage1=new EventDataType [size]; - EventDataType *stage2=new EventDataType [size]; + int samples=size; + + // Create two buffers for filter stages. + EventDataType *stage1=new EventDataType [samples]; + EventDataType *stage2=new EventDataType [samples]; QVector med; med.reserve(8); @@ -39,65 +561,11 @@ int filterFlow(Session *session, EventList *in, EventList *out, EventList *tv, E int cnt; EventDataType c; - //double avg; int i; - // Extra median filter stage - /*i=3; - stage1[0]=in->data(0); - stage1[1]=in->data(1); - stage1[2]=in->data(2); - for (;idata(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); - */ + percentileFilter(in->rawData(), stage1, samples, 11, 0.5); + percentileFilter(stage1, stage2, samples, 7, 0.5); - //i++; - //stage1[i]=in->data(i); - - // Anti-Alias the flow waveform to get rid of jagged edges. - stage2[0]=in->data(0); - stage2[1]=in->data(1); - stage2[2]=in->data(2); - - i=3; - for (;idata(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;idata(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; @@ -111,11 +579,6 @@ int filterFlow(Session *session, EventList *in, EventList *out, EventList *tv, E QVector breaths_max_peak; QVector breaths_max; -// const int ringsize=256; -// EventDataType ringmax[ringsize]={0}; -// EventDataType ringmin[ringsize]={0}; -// qint64 ringtime[ringsize]={0}; -// int rpos=0; EventDataType min=0,max=0; qint64 peakmin=0, peakmax=0; double avgmax=0; @@ -145,9 +608,11 @@ int filterFlow(Session *session, EventList *in, EventList *out, EventList *tv, E breaths.push_back(len); // keep previously calculated negative peak - breaths_min_peak.push_back(peakmin); - breaths_min.push_back(min); - avgmin+=min; + if (peakmin) { + breaths_min_peak.push_back(peakmin); + breaths_min.push_back(min); + avgmin+=min; + } max=0; } } else { @@ -178,9 +643,11 @@ int filterFlow(Session *session, EventList *in, EventList *out, EventList *tv, E } } // keep previously calculated positive peak - breaths_max_peak.push_back(peakmax); - breaths_max.push_back(max); - avgmax+=max; + if (peakmax>0) { + breaths_max_peak.push_back(peakmax); + breaths_max.push_back(max); + avgmax+=max; + } min=0; } @@ -423,6 +890,57 @@ int calcRespRate(Session *session) return cnt; } + */ + +void calcRespRate(Session *session, FlowParser * flowparser) +{ + if (session->machine()->GetType()!=MT_CPAP) return; + +// if (session->machine()->GetClass()!=STR_MACH_PRS1) return; + + if (!session->eventlist.contains(CPAP_FlowRate)) { + qDebug() << "calcRespRate called without FlowRate waveform available"; + return; //need flow waveform + } + + bool trashfp; + if (!flowparser) { + flowparser=new FlowParser(); + trashfp=true; + qDebug() << "calcRespRate called without valid FlowParser object.. using a slow throw-away!"; + //return; + } else { + trashfp=false; + } + + bool calcResp=!session->eventlist.contains(CPAP_RespRate); + bool calcTv=!session->eventlist.contains(CPAP_TidalVolume); + bool calcTi=!session->eventlist.contains(CPAP_Ti); + bool calcTe=!session->eventlist.contains(CPAP_Te); + bool calcMv=!session->eventlist.contains(CPAP_MinuteVent); + + flowparser->clearFilters(); + + // No filters works rather well with the new peak detection algorithm.. + // Although the output could use filtering. + + //flowparser->addFilter(FilterPercentile,7,0.5); + //flowparser->addFilter(FilterPercentile,5,0.5); + //flowparser->addFilter(FilterXPass,0.5); + EventList *flow; + int cnt=0; + for (int ws=0; ws < session->eventlist[CPAP_FlowRate].size(); ws++) { + flow=session->eventlist[CPAP_FlowRate][ws]; + if (flow->count() > 5) { + flowparser->openFlow(session, flow); + flowparser->calc(calcResp, calcTv, calcTi ,calcTe, calcMv); + } + } + if (trashfp) { + delete flowparser; + } +} + EventDataType calcAHI(Session *session,qint64 start, qint64 end) { diff --git a/SleepLib/calcs.h b/SleepLib/calcs.h index 5eee64eb..39fc40b9 100644 --- a/SleepLib/calcs.h +++ b/SleepLib/calcs.h @@ -8,8 +8,134 @@ #include "day.h" +//! param samples Number of samples +//! width number of surrounding samples to consider +//! percentile fractional percentage, between 0 and 1 +void percentileFilter(EventDataType * input, EventDataType * output, int samples, int width, EventDataType percentile); +void xpassFilter(EventDataType * input, EventDataType * output, int samples, EventDataType weight); + +enum FilterType { FilterNone=0, FilterPercentile, FilterXPass }; + +struct Filter { + Filter(FilterType t,EventDataType p1,EventDataType p2,EventDataType p3) { + type=t; + param1=p1; + param2=p2; + param3=p3; + } + Filter() { + type=FilterNone; + param1=0; + param2=0; + param3=0; + } + Filter(const Filter & copy) { + type=copy.type; + param1=copy.param1; + param2=copy.param2; + param3=copy.param3; + } + + FilterType type; + EventDataType param1; + EventDataType param2; + EventDataType param3; +}; + +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) { + min=_min; + max=_max; + start=_start; + middle=_middle; + end=_end; + 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; + } + int samplelength() { return end-start; } + int upperLength() { return middle-start; } + int lowerLength() { return end-middle; } + + 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 +}; + +bool operator<(const BreathPeak & p1, const BreathPeak & p2); + +const int num_filter_buffers=2; + +const int max_filter_buf_size=1048576*4; + +//! \brief Class to process Flow Rate waveform data +class FlowParser { +public: + FlowParser(); + ~FlowParser(); + + //! \brief Clears the (input) filter chain + void clearFilters(); + + //! \brief Applies the filter chain to input, with supplied number of samples + EventDataType * applyFilters(EventDataType * input, int samples); + + //! \brief Add the filter + void addFilter(FilterType ft, EventDataType p1=0, EventDataType p2=0, EventDataType p3=0) { + m_filters.push_back(Filter(ft,p1,p2,p3)); + } + //! \brief Opens the flow rate EventList, applies the input filter chain, and calculates peaks + void openFlow(Session * session, EventList * flow); + + //! \brief Calculates the upper and lower breath peaks + void calcPeaks(EventDataType * input, int samples); + + // Minute vent needs Resp & TV calcs made here.. + void calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool calcMv); + + /*void calcTidalVolume(); + void calcRespRate(); + void calcMinuteVent(); */ + + + QList m_filters; +protected: + QVector breaths_lower; + QVector breaths_upper; + QVector breaths; + QList breatsh_good; + + int m_samples; + EventList * m_flow; + Session * m_session; + EventDataType m_gain; + EventDataType m_rate; + EventDataType m_minutes; + //! \brief The filtered waveform + EventDataType * m_filtered; + //! \brief BreathPeak's start on positive cycle? + bool m_startsUpper; +private: + EventDataType * m_buffers[num_filter_buffers]; +}; + +bool SearchApnea(Session *session, qint64 time, qint64 dist=15000); + //! \brief Calculate Respiratory Rate, Tidal Volume & Minute Ventilation for PRS1 data -int calcRespRate(Session *session); +void calcRespRate(Session *session, FlowParser * flowparser=NULL); //! \brief Calculates the sliding window AHI graph int calcAHIGraph(Session *session); diff --git a/SleepLib/machine.cpp b/SleepLib/machine.cpp index d81c7096..fea91b1d 100644 --- a/SleepLib/machine.cpp +++ b/SleepLib/machine.cpp @@ -445,7 +445,7 @@ CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_FlowRate, CPAP_MaskPressure, CPAP_RespEvent, CPAP_Snore, CPAP_MinuteVent, CPAP_RespRate, CPAP_TidalVolume, CPAP_PTB, CPAP_Leak, CPAP_LeakMedian, CPAP_LeakTotal, CPAP_MaxLeak, CPAP_FLG, CPAP_IE, CPAP_Te, CPAP_Ti, CPAP_TgMV, CPAP_UserFlag1, CPAP_UserFlag2, CPAP_BrokenSummary, CPAP_BrokenWaveform, CPAP_RDI, -CPAP_PresReliefSet, CPAP_PresReliefMode, CPAP_PresReliefType, CPAP_PSMin, CPAP_PSMax; +CPAP_PresReliefSet, CPAP_PresReliefMode, CPAP_PresReliefType, CPAP_PSMin, CPAP_PSMax, CPAP_Test1, CPAP_Test2; ChannelID RMS9_E01, RMS9_E02, RMS9_EPR, RMS9_EPRSet, RMS9_SetPressure; diff --git a/SleepLib/machine_common.h b/SleepLib/machine_common.h index dfd3ccf9..2a08c9fd 100644 --- a/SleepLib/machine_common.h +++ b/SleepLib/machine_common.h @@ -91,7 +91,7 @@ CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_FlowRate, CPAP_MaskPressure, CPAP_RespEvent, CPAP_Snore, CPAP_MinuteVent, CPAP_RespRate, CPAP_TidalVolume, CPAP_PTB, CPAP_Leak, CPAP_LeakMedian, CPAP_LeakTotal, CPAP_MaxLeak, CPAP_FLG, CPAP_IE, CPAP_Te, CPAP_Ti, CPAP_TgMV, CPAP_UserFlag1, CPAP_UserFlag2, CPAP_BrokenSummary, CPAP_BrokenWaveform, CPAP_RDI, -CPAP_PresReliefSet, CPAP_PresReliefMode, CPAP_PresReliefType, CPAP_PSMin, CPAP_PSMax; +CPAP_PresReliefSet, CPAP_PresReliefMode, CPAP_PresReliefType, CPAP_PSMin, CPAP_PSMax, CPAP_Test1, CPAP_Test2; extern ChannelID RMS9_E01, RMS9_E02, RMS9_EPR, RMS9_EPRSet, RMS9_SetPressure; extern ChannelID INTP_SmartFlex; diff --git a/SleepLib/profiles.cpp b/SleepLib/profiles.cpp index 639918c9..36634458 100644 --- a/SleepLib/profiles.cpp +++ b/SleepLib/profiles.cpp @@ -299,7 +299,7 @@ Day * Profile::GetDay(QDate date,MachineType type) day=tmp; break; } - if ((*di)->machine_type()==type) { + if (tmp->machine_type()==type) { day=tmp; break; } diff --git a/SleepLib/schema.cpp b/SleepLib/schema.cpp index ed67bd2c..26266baa 100644 --- a/SleepLib/schema.cpp +++ b/SleepLib/schema.cpp @@ -109,6 +109,8 @@ void init() CPAP_Te=schema::channel["Te"].id(); CPAP_Ti=schema::channel["Ti"].id(); CPAP_TgMV=schema::channel["TgMV"].id(); + CPAP_Test1=schema::channel["TestChan1"].id(); + CPAP_Test2=schema::channel["TestChan2"].id(); CPAP_PresReliefSet=schema::channel["PresRelSet"].id(); CPAP_PresReliefMode=schema::channel["PresRelMode"].id(); diff --git a/daily.cpp b/daily.cpp index 435b1d93..b03bcf98 100644 --- a/daily.cpp +++ b/daily.cpp @@ -246,15 +246,24 @@ Daily::Daily(QWidget *parent,gGraphView * shared) 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(CPAP_RespRate,Qt::darkMagenta,square))); + RR->AddLayer(AddCPAP(lc=new gLineChart(CPAP_RespRate,Qt::darkBlue,square))); + + // Delete me!! +// lc->addPlot(CPAP_Test1,Qt::darkRed,square); + MV->AddLayer(AddCPAP(new gLineChart(CPAP_MinuteVent,Qt::darkCyan,square))); - TV->AddLayer(AddCPAP(new gLineChart(CPAP_TidalVolume,Qt::magenta,square))); + TV->AddLayer(AddCPAP(lc=new gLineChart(CPAP_TidalVolume,Qt::magenta,square))); + //lc->addPlot(CPAP_Test2,Qt::darkYellow,square); + + + //TV->AddLayer(AddCPAP(new gLineChart("TidalVolume2",Qt::magenta,square))); 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,square))); - TE->AddLayer(AddCPAP(new gLineChart(CPAP_Te,Qt::darkGreen,square))); - TI->AddLayer(AddCPAP(new gLineChart(CPAP_Ti,Qt::darkBlue,square))); + IE->AddLayer(AddCPAP(lc=new gLineChart(CPAP_IE,Qt::darkRed,square))); + TE->AddLayer(AddCPAP(lc=new gLineChart(CPAP_Te,Qt::darkGreen,square))); + TI->AddLayer(AddCPAP(lc=new gLineChart(CPAP_Ti,Qt::darkBlue,square))); + //lc->addPlot(CPAP_Test2,Qt::darkYellow,square); TgMV->AddLayer(AddCPAP(new gLineChart(CPAP_TgMV,Qt::darkCyan,square))); //INTPULSE->AddLayer(AddCPAP(new gLineChart(OXI_Pulse,Qt::red,square))); //INTSPO2->AddLayer(AddCPAP(new gLineChart(OXI_SPO2,Qt::blue,square))); diff --git a/docs/channels.xml b/docs/channels.xml index 1b0c058a..cfdc6f80 100644 --- a/docs/channels.xml +++ b/docs/channels.xml @@ -58,6 +58,9 @@ Important: One id code per item, DO NOT CHANGE ID NUMBERS!!! + + + diff --git a/mainwindow.cpp b/mainwindow.cpp index ee73bc07..2ff9ad74 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -2333,7 +2333,8 @@ void MainWindow::on_actionAll_Data_for_current_CPAP_machine_triggered() if (QMessageBox::question(this,tr("Are you sure?"),tr("Are you sure you want to purge all CPAP data for the following machine:\n")+m->properties[STR_PROP_Brand]+" "+m->properties[STR_PROP_Model]+" "+m->properties[STR_PROP_ModelNumber]+" ("+m->properties[STR_PROP_Serial]+")",QMessageBox::Yes,QMessageBox::No)==QMessageBox::Yes) { m->Purge(3478216); PROFILE.machlist.erase(PROFILE.machlist.find(m->id())); - delete m; + // delete or not to delete.. this needs to delete later.. :/ + //delete m; RestartApplication(); } } diff --git a/profileselect.cpp b/profileselect.cpp index 529ce0be..7d0b02b1 100644 --- a/profileselect.cpp +++ b/profileselect.cpp @@ -184,8 +184,9 @@ void ProfileSelect::on_newProfileButton_clicked() { NewProfile newprof(this); newprof.skipWelcomeScreen(); - newprof.exec(); - accept(); + if (newprof.exec()==NewProfile::Rejected) { +// reject(); + } else accept(); }