From 2e49a51b176b6861aac607329f954909a0d3d963 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Mon, 14 Oct 2013 02:35:53 +1000 Subject: [PATCH] Cleaned up Daily pages html formatting, schema needed to initialize before Machine loader registrations --- sleepyhead/SleepLib/common.h | 1 + .../SleepLib/loader_plugins/icon_loader.cpp | 18 +- .../loader_plugins/intellipap_loader.cpp | 15 +- .../SleepLib/loader_plugins/prs1_loader.cpp | 19 +- .../SleepLib/loader_plugins/resmed_loader.cpp | 25 +- sleepyhead/daily.cpp | 738 +++++++++--------- sleepyhead/daily.h | 8 + sleepyhead/main.cpp | 2 +- 8 files changed, 451 insertions(+), 375 deletions(-) diff --git a/sleepyhead/SleepLib/common.h b/sleepyhead/SleepLib/common.h index 48ed2636..3e68015a 100644 --- a/sleepyhead/SleepLib/common.h +++ b/sleepyhead/SleepLib/common.h @@ -57,6 +57,7 @@ const QString STR_PREF_AllowEarlyUpdates="AllowEarlyUpdates"; const QString STR_PROP_Brand="Brand"; const QString STR_PROP_Model="Model"; +const QString STR_PROP_Series="Series"; const QString STR_PROP_ModelNumber="ModelNumber"; const QString STR_PROP_SubModel="SubModel"; const QString STR_PROP_Serial="Serial"; diff --git a/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp b/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp index c6c80a88..1ece98c9 100644 --- a/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp @@ -23,8 +23,6 @@ FPIcon::FPIcon(Profile *p,MachineID id) :CPAP(p,id) { m_class=fpicon_class_name; - properties[STR_PROP_Brand]="Fisher & Paykel"; - properties[STR_PROP_Model]=STR_MACH_FPIcon; } FPIcon::~FPIcon() @@ -290,6 +288,10 @@ bool FPIconLoader::OpenFLW(Machine * mach,QString filename, Profile * profile) htxt >> model; htxt >> type; + if (mach->properties[STR_PROP_Model].isEmpty()) { + mach->properties[STR_PROP_Model]=model+" "+type; + } + fname.chop(4); QString num=fname.right(4); int filenum=num.toInt(); @@ -793,16 +795,24 @@ Machine *FPIconLoader::CreateMachine(QString serial,Profile *profile) QList ml=profile->GetMachines(MT_CPAP); bool found=false; QList::iterator i; + Machine *m; for (i=ml.begin(); i!=ml.end(); i++) { if (((*i)->GetClass()==fpicon_class_name) && ((*i)->properties[STR_PROP_Serial]==serial)) { MachList[serial]=*i; found=true; + m=*i; break; } } - if (found) return *i; + if (!found) { + m=new FPIcon(profile,0); + } + m->properties[STR_PROP_Brand]="Fisher & Paykel"; + m->properties[STR_PROP_Series]=STR_MACH_FPIcon; + m->properties[STR_PROP_Model]=STR_MACH_FPIcon; + if (found) + return m; - Machine *m=new FPIcon(profile,0); MachList[serial]=m; profile->AddMachine(m); diff --git a/sleepyhead/SleepLib/loader_plugins/intellipap_loader.cpp b/sleepyhead/SleepLib/loader_plugins/intellipap_loader.cpp index 6dac753e..9a4f5349 100644 --- a/sleepyhead/SleepLib/loader_plugins/intellipap_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/intellipap_loader.cpp @@ -21,8 +21,6 @@ Intellipap::Intellipap(Profile *p,MachineID id) :CPAP(p,id) { m_class=intellipap_class_name; - properties[STR_PROP_Brand]="DeVilbiss"; - properties[STR_PROP_Model]=STR_MACH_Intellipap; } Intellipap::~Intellipap() @@ -136,6 +134,7 @@ int IntellipapLoader::Open(QString & path,Profile *profile) for (QHash::iterator i=set1.begin();i!=set1.end();i++) { mach->properties[i.key()]=i.value(); } + mach->properties[STR_PROP_Model]=STR_MACH_Intellipap+" "+mach->properties[STR_PROP_ModelNumber]; f.close(); @@ -383,16 +382,24 @@ Machine *IntellipapLoader::CreateMachine(QString serial,Profile *profile) QList ml=profile->GetMachines(MT_CPAP); bool found=false; QList::iterator i; + Machine *m=NULL; for (i=ml.begin(); i!=ml.end(); i++) { if (((*i)->GetClass()==intellipap_class_name) && ((*i)->properties[STR_PROP_Serial]==serial)) { MachList[serial]=*i; //static_cast(*i); found=true; + m=*i; break; } } - if (found) return *i; + if (!found) { + m=new Intellipap(profile,0); + } + m->properties[STR_PROP_Brand]="DeVilbiss"; + m->properties[STR_PROP_Series]=STR_MACH_Intellipap; + + if (found) + return m; - Machine *m=new Intellipap(profile,0); MachList[serial]=m; profile->AddMachine(m); diff --git a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp index ff7149b1..3568181c 100644 --- a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp @@ -99,9 +99,6 @@ crc_t CRC16(const unsigned char *data, size_t data_len) PRS1::PRS1(Profile *p,MachineID id):CPAP(p,id) { m_class=prs1_class_name; - properties[STR_PROP_Brand]="Philips Respironics"; - properties[STR_PROP_Model]="System One"; - } PRS1::~PRS1() { @@ -136,17 +133,25 @@ Machine *PRS1Loader::CreateMachine(QString serial,Profile *profile) QList ml=profile->GetMachines(MT_CPAP); bool found=false; QList::iterator i; + Machine *m=NULL; + for (i=ml.begin(); i!=ml.end(); i++) { if (((*i)->GetClass()==STR_MACH_PRS1) && ((*i)->properties[STR_PROP_Serial]==serial)) { PRS1List[serial]=*i; //static_cast(*i); found=true; + m=*i; break; } } - if (found) return *i; + if (!found) { + m=new PRS1(profile,0); + } + m->properties[STR_PROP_Brand]="Philips Respironics"; + m->properties[STR_PROP_Series]="System One"; - //assert(PRS1List.find(serial)==PRS1List.end()) - Machine *m=new PRS1(profile,0); + if (found) { + return m; + } PRS1List[serial]=m; profile->AddMachine(m); @@ -262,7 +267,7 @@ bool PRS1Loader::ParseProperties(Machine *m,QString filename) int i=pt.toInt(&ok,16); if (ok) { if (ModelMap.find(i)!=ModelMap.end()) { - m->properties[STR_PROP_SubModel]=ModelMap[i]; + m->properties[STR_PROP_Model]=ModelMap[i]; } } if (prop["SerialNumber"]!=m->properties[STR_PROP_Serial]) { diff --git a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp index cd22d75b..96e6d534 100644 --- a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp @@ -271,24 +271,31 @@ Machine *ResmedLoader::CreateMachine(QString serial,Profile *profile) QList ml=profile->GetMachines(MT_CPAP); bool found=false; QList::iterator i; + Machine *m=NULL; for (i=ml.begin(); i!=ml.end(); i++) { if (((*i)->GetClass()==resmed_class_name) && ((*i)->properties[STR_PROP_Serial]==serial)) { ResmedList[serial]=*i; //static_cast(*i); found=true; + m=*i; break; } } - if (found) return *i; + if (!found) { + m=new CPAP(profile,0); + } + + m->properties[STR_PROP_Brand]=STR_MACH_ResMed; + m->properties[STR_PROP_Series]="S9"; + if (found) + return m; qDebug() << "Create ResMed Machine" << serial; - Machine *m=new CPAP(profile,0); m->SetClass(resmed_class_name); ResmedList[serial]=m; profile->AddMachine(m); m->properties[STR_PROP_Serial]=serial; - m->properties[STR_PROP_Brand]=STR_MACH_ResMed; m->properties[STR_PROP_DataVersion]=QString::number(resmed_data_version); QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/"; @@ -405,7 +412,8 @@ int ResmedLoader::Open(QString & path,Profile *profile) if (i.key()=="PCD") { // Lookup Product Code for real model string bool ok; int j=i.value().toInt(&ok); - if (ok) m->properties[STR_PROP_Model]=RMS9ModelMap[j]; + if (ok) + m->properties[STR_PROP_Model]=RMS9ModelMap[j]; } } @@ -1151,8 +1159,9 @@ int ResmedLoader::Open(QString & path,Profile *profile) sess->SetChanged(true); qint64 dif=sess->first()-stredf.startdate; int dn=dif/86400000L; + int mode; + EventDataType prset, prmode; if (dndata[dn]; @@ -1163,12 +1172,14 @@ int ResmedLoader::Open(QString & path,Profile *profile) // AutoSV machines don't have both fields sig=stredf.lookupSignal(RMS9_EPR); if (sig) { - sess->settings[CPAP_PresReliefMode]=EventDataType(sig->data[dn])*sig->gain; + prmode=EventDataType(sig->data[dn])*sig->gain; + sess->settings[CPAP_PresReliefMode]=prmode; } sig=stredf.lookupSignal(RMS9_EPRSet); if (sig) { - sess->settings[CPAP_PresReliefSet]=EventDataType(sig->data[dn])*sig->gain; + prset=EventDataType(sig->data[dn])*sig->gain; + sess->settings[CPAP_PresReliefSet]=prset; } diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index e34553d0..b9bb0e93 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -721,12 +721,359 @@ MyWebView::MyWebView(QWidget *parent): } +QString Daily::getSessionInformation(Day * cpap, Day * oxi, Day * stage) +{ + QString html; + QList list; + if (cpap) list.push_back(cpap); + if (oxi) list.push_back(oxi); + if (stage) list.push_back(stage); + + if (list.isEmpty()) + return html; + + html=""; + html+=QString(""); + html+=""; + QFontMetrics FM(*defaultfont); + QRect r=FM.boundingRect('@'); + if (cpap) { + html+=QString("").arg(r.height()*3,0,10); + html+=""; + } + + + QDateTime fd,ld; + bool corrupted_waveform=false; + QString tooltip; + + html+=QString(""); + QList::iterator di; + QString type; + + for (di=list.begin();di!=list.end();di++) { + Day * day=*di; + html+="\n"; + for (QVector::iterator s=day->begin();s!=day->end();s++) { + + if ((day->machine_type()==MT_CPAP) && + ((*s)->settings.find(CPAP_BrokenWaveform)!=(*s)->settings.end())) + corrupted_waveform=true; + + fd=QDateTime::fromTime_t((*s)->first()/1000L); + ld=QDateTime::fromTime_t((*s)->last()/1000L); + int len=(*s)->length()/1000L; + int h=len/3600; + int m=(len/60) % 60; + int s1=len % 60; + tooltip=day->machine->GetClass()+QString(":#%1").arg((*s)->session(),8,10,QChar('0')); + // tooltip needs to lookup language.. :-/ + + Session *sess=*s; + if (!sess->settings.contains(SESSION_ENABLED)) { + sess->settings[SESSION_ENABLED]=true; + } + bool b=sess->settings[SESSION_ENABLED].toBool(); + html+=QString("" + "") + .arg((*s)->session()) + .arg(tooltip) + .arg(QString("%1h %2m %3s").arg(h,2,10,QChar('0')).arg(m,2,10,QChar('0')).arg(s1,2,10,QChar('0'))) + .arg((b ? "on" : "off")) + .arg(fd.date().toString(Qt::SystemLocaleShortDate)) + .arg(fd.toString("HH:mm")) + .arg(ld.toString("HH:mm")); + } + } + + if (corrupted_waveform) { + html+=QString("").arg(tr("One or more waveform record for this session had faulty source data. Some waveform overlay points may not match up correctly.")); + } + html+="
"+tr("Session Information")+"
 
" + "" +// "" + "
 
"+STR_TR_On+""+STR_TR_Date+""+STR_TR_Start+""+STR_TR_End+""+tr("Duration")+"
"; + switch (day->machine_type()) { + case MT_CPAP: type="cpap"; + html+=tr("CPAP Sessions"); + break; + case MT_OXIMETER: type="oxi"; + html+=tr("Oximetery Sessions"); + break; + case MT_SLEEPSTAGE: type="stage"; + html+=tr("Sleep Stage Sessions"); + break; + default: + type="unknown"; + html+=tr("Unknown Session"); + break; + } + html+="
" + "%5%6%7%3%2
%1
"; + return html; +} + +QString Daily::getMachineSettings(Day * cpap) { + QString html; + if (cpap && cpap->hasEnabledSessions()) { + html=""; + html+=QString("").arg(tr("Machine Settings")); + html+=""; + int i=cpap->settings_max(CPAP_PresReliefType); + int j=cpap->settings_max(CPAP_PresReliefSet); + QString flexstr=(i>1) ? schema::channel[CPAP_PresReliefType].option(i)+" x"+QString::number(j) : STR_TR_None; + html+=QString("") + .arg(STR_TR_PrRelief) + .arg(schema::channel[CPAP_PresReliefType].description()) + .arg(flexstr); + QString mclass=cpap->machine->GetClass(); + if (mclass==STR_MACH_PRS1 || mclass==STR_MACH_FPIcon) { + int humid=round(cpap->settings_wavg(CPAP_HumidSetting)); + html+=QString("") + .arg(schema::channel[CPAP_HumidSetting].description()) + .arg(humid==0 ? STR_GEN_Off : "x"+QString::number(humid)); + } + html+="
%1
 
%1%2%3
"+STR_TR_Humidifier+"%1%2
"; + html+="
\n"; + } + return html; +} + +QString Daily::getOximeterInformation(Day * oxi) +{ + QString html; + if (oxi && oxi->hasEnabledSessions()) { + html=""; + html+=QString("\n").arg(tr("Oximeter Information")); + html+=""; + html+="\n"; + html+=""; + html+=QString("").arg(tr("SpO2 Desaturations")).arg(oxi->count(OXI_SPO2Drop)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_SPO2Drop)/3600.0),0,'f',2); + html+=QString("").arg(tr("Pulse Change events")).arg(oxi->count(OXI_PulseChange)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_PulseChange)/3600.0),0,'f',2); + html+=QString("").arg(tr("SpO2 Baseline Used")).arg(oxi->settings_wavg(OXI_SPO2Drop),0,'f',2); // CHECKME: Should this value be wavg OXI_SPO2 isntead? + html+="
%1
 
"+oxi->machine->properties[STR_PROP_Brand]+" "+oxi->machine->properties[STR_PROP_Model]+"
 
%1: %2 (%3)\%
%1: %2 (%3)\%
%1: %2\%
\n"; + html+="
\n"; + } + return html; +} + +QString Daily::getCPAPInformation(Day * cpap) +{ + QString html; + if (!cpap) + return html; + + QString brand=cpap->machine->properties[STR_PROP_Brand]; + QString series=cpap->machine->properties[STR_PROP_Series]; + QString model=cpap->machine->properties[STR_PROP_Model]; + QString number=cpap->machine->properties[STR_PROP_ModelNumber]; + + html="\n"; + + html+="\n"; + CPAPMode mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode); + html+="\n"; + html+="
"+model+""; + QString tooltip=(brand+"\n"+series+" "+number+"\n"+cpap->machine->properties[STR_PROP_Serial]); + tooltip=tooltip.replace(" "," "); + + + html+=tooltip; + html+="
"+tr("PAP Setting")+": "; + + if (mode==MODE_CPAP) { + EventDataType min=round(cpap->settings_wavg(CPAP_Pressure)*2)/2.0; + html+=STR_TR_CPAP+" "+QString::number(min)+STR_UNIT_CMH2O; + } else if (mode==MODE_APAP) { + EventDataType min=cpap->settings_min(CPAP_PressureMin); + EventDataType max=cpap->settings_max(CPAP_PressureMax); + html+=STR_TR_APAP+" "+QString::number(min)+"-"+QString::number(max)+STR_UNIT_CMH2O; + } else if (mode==MODE_BIPAP) { + EventDataType epap=cpap->settings_min(CPAP_EPAP); + EventDataType ipap=cpap->settings_max(CPAP_IPAP); + EventDataType ps=cpap->settings_max(CPAP_PS); + html+=STR_TR_BiLevel+QString("
"+STR_TR_EPAP+": %1 "+STR_TR_IPAP+": %2 %3
"+STR_TR_PS+": %4") + .arg(epap,0,'f',1).arg(ipap,0,'f',1).arg(STR_UNIT_CMH2O).arg(ps,0,'f',1); + } + else if (mode==MODE_ASV) { + EventDataType epap=cpap->settings_min(CPAP_EPAP); + EventDataType low=cpap->settings_min(CPAP_IPAPLo); + EventDataType high=cpap->settings_max(CPAP_IPAPHi); + EventDataType psl=cpap->settings_min(CPAP_PSMin); + EventDataType psh=cpap->settings_max(CPAP_PSMax); + html+=tr("ASV")+QString("
"+STR_TR_EPAP+": %1 "+STR_TR_IPAP+": %2 - %3 %4
"+STR_TR_PS+": %5 / %6") + .arg(epap,0,'f',1) + .arg(low,0,'f',1) + .arg(high,0,'f',1) + .arg(STR_UNIT_CMH2O) + .arg(psl,0,'f',1) + .arg(psh,0,'f',1); + } + else html+=STR_TR_Unknown; + html+="
\n"; + // html+="
\n"; + return html; +} + + +QString Daily::getStatisticsInfo(Day * cpap,Day * oxi) +{ + + QList list; + + list.push_back(cpap); + list.push_back(oxi); + + + int mididx=PROFILE.general->prefCalcMiddle(); + SummaryType ST_mid; + if (mididx==0) ST_mid=ST_PERC; + if (mididx==1) ST_mid=ST_WAVG; + if (mididx==2) ST_mid=ST_AVG; + + float percentile=PROFILE.general->prefCalcPercentile()/100.0; + + SummaryType ST_max=PROFILE.general->prefCalcMax() ? ST_MAX : ST_PERC; + const EventDataType maxperc=0.995F; + + QString midname; + if (ST_mid==ST_WAVG) midname=tr("Avg"); + else if (ST_mid==ST_AVG) midname=tr("Avg"); + else if (ST_mid==ST_PERC) midname=tr("Med"); + + QString html; + + html+="\n"; + html+=QString("\n").arg(tr("Statistics")); + html+=QString("") + .arg(STR_TR_Channel) + .arg(STR_TR_Min) + .arg(midname) + .arg(tr("%1%").arg(percentile*100,0,'f',0)) + .arg(STR_TR_Max); + + ChannelID chans[]={ + CPAP_Pressure,CPAP_EPAP,CPAP_IPAP,CPAP_PS,CPAP_PTB, + CPAP_MinuteVent, CPAP_RespRate, CPAP_RespEvent,CPAP_FLG, + CPAP_Leak, CPAP_LeakTotal, CPAP_Snore,CPAP_IE,CPAP_Ti,CPAP_Te, CPAP_TgMV, + CPAP_TidalVolume, OXI_Pulse, OXI_SPO2 + }; + int numchans=sizeof(chans)/sizeof(ChannelID); + int ccnt=0; + EventDataType tmp,med,perc,mx,mn; + + QList::iterator di; + + for (di=list.begin();di!=list.end();di++) { + Day * day=*di; + + if (!day) + continue; + + for (int i=0;ichannelHasData(code)) + continue; + + QString tooltip=schema::channel[code].description(); + + if (!schema::channel[code].units().isEmpty()) tooltip+=" ("+schema::channel[code].units()+")"; + + if (ST_max==ST_MAX) { + mx=day->Max(code); + } else { + mx=day->percentile(code,maxperc); + } + + mn=day->Min(code); + perc=day->percentile(code,percentile); + + if (ST_mid==ST_PERC) { + med=day->percentile(code,0.5); + tmp=day->wavg(code); + if (tmp>0 || mx==0) { + tooltip+=QString("
"+STR_TR_WAvg+": %1").arg(tmp,0,'f',2); + } + } else if (ST_mid==ST_WAVG) { + med=day->wavg(code); + tmp=day->percentile(code,0.5); + if (tmp>0 || mx==0) { + tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); + } + } else if (ST_mid==ST_AVG) { + med=day->avg(code); + tmp=day->percentile(code,0.5); + if (tmp>0 || mx==0) { + tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); + } + } + + html+=QString("
") + .arg(schema::channel[code].label()) + .arg(mn,0,'f',2) + .arg(med,0,'f',2) + .arg(perc,0,'f',2) + .arg(mx,0,'f',2) + .arg(tooltip); + ccnt++; + } + } + if (GraphView->isEmpty() && (ccnt>0)) { + html+="\n"; + html+=QString("").arg(tr("Please Note: This day just contains summary data, only limited information is available .")); + } + html+="
%1
%1%2%3%4%5
%1%6%2%3%4%5
 
%1
\n"; + html+="
\n"; + return html; +} + +QString Daily::getEventBreakdown(Day * cpap) +{ + QString html; + html+="\n"; + + html+="
"; + return html; +} + +QString Daily::getSleepTime(Day * cpap, Day * oxi) +{ + QString html; + + Day * day=NULL; + if (cpap && cpap->hours()>0) + day=cpap; + else if (oxi && oxi->hours()>0) + day=oxi; + else + return html; + html+="\n"; + html+=""; + int tt=qint64(day->total_time())/1000L; + QDateTime date=QDateTime::fromTime_t(day->first()/1000L); + QDateTime date2=QDateTime::fromTime_t(day->last()/1000L); + + int h=tt/3600; + int m=(tt/60)%60; + int s=tt % 60; + html+=QString("\n") + .arg(date.date().toString(Qt::SystemLocaleShortDate)) + .arg(date.toString("HH:mm")) + .arg(date2.toString("HH:mm")) + .arg(QString().sprintf("%02i:%02i:%02i",h,m,s)); + html+="
"+STR_TR_Date+""+tr("Sleep")+""+tr("Wake")+""+STR_UNIT_Hours+"
%1%2%3%4
\n"; +// html+="
"; + + return html; +} + + void Daily::Load(QDate date) { - static int calls=0; - - calls++; - qDebug() << "in Load()" << date << calls; dateDisplay->setText(""+date.toString(Qt::SystemLocaleLongDate)+""); static Day * lastcpapday=NULL; previous_date=date; @@ -768,8 +1115,7 @@ void Daily::Load(QDate date) "}" "-->" "" - "" - "\n"; + ""; QString tmp; UpdateOXIGraphs(oxi); @@ -795,11 +1141,7 @@ void Daily::Load(QDate date) bool isBrick=false; updateGraphCombo(); - int mididx=PROFILE.general->prefCalcMiddle(); - SummaryType ST_mid; - if (mididx==0) ST_mid=ST_PERC; - if (mididx==1) ST_mid=ST_WAVG; - if (mididx==2) ST_mid=ST_AVG; + if (cpap) { float hours=cpap->hours(); @@ -833,77 +1175,27 @@ void Daily::Load(QDate date) //float p90=cpap->p90(CPAP_Pressure); //eap90=cpap->p90(CPAP_EPAP); //iap90=cpap->p90(CPAP_IPAP); - QString submodel; //=tr("Unknown Model"); - - //html+="\n"; - if (cpap->machine->properties.find(STR_PROP_SubModel)!=cpap->machine->properties.end()) - submodel="
"+cpap->machine->properties[STR_PROP_SubModel]; - html+="\n"; - if (PROFILE.general->showSerialNumbers()) { - html+="\n"; - } - CPAPMode mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode); - html+="\n"; - - - if (hours>0) { - html+=""; - int tt=qint64(cpap->total_time())/1000L; - QDateTime date=QDateTime::fromTime_t(cpap->first()/1000L); - QDateTime date2=QDateTime::fromTime_t(cpap->last()/1000L); - - int h=tt/3600; - int m=(tt/60)%60; - int s=tt % 60; - html+=QString("\n" - "\n") - .arg(date.date().toString(Qt::SystemLocaleShortDate)) - .arg(date.toString("HH:mm")) - .arg(date2.toString("HH:mm")) - .arg(QString().sprintf("%02i:%02i:%02i",h,m,s)); - } QString cs; if (!isBrick && hours>0) { + html+="
"+tr("Machine Information")+"
"+cpap->machine->properties[STR_PROP_Brand]+"
"+cpap->machine->properties[STR_PROP_Model]+" "+cpap->machine->properties[STR_PROP_ModelNumber]+submodel+"
"+cpap->machine->properties[STR_PROP_Serial]+"
Mode: "; - - if (mode==MODE_CPAP) { - EventDataType min=round(cpap->settings_wavg(CPAP_Pressure)*2)/2.0; - html+=STR_TR_CPAP+" "+QString::number(min)+STR_UNIT_CMH2O; - } else if (mode==MODE_APAP) { - EventDataType min=cpap->settings_min(CPAP_PressureMin); - EventDataType max=cpap->settings_max(CPAP_PressureMax); - html+=STR_TR_APAP+" "+QString::number(min)+"-"+QString::number(max)+STR_UNIT_CMH2O; - } else if (mode==MODE_BIPAP) { - EventDataType epap=cpap->settings_min(CPAP_EPAP); - EventDataType ipap=cpap->settings_max(CPAP_IPAP); - EventDataType ps=cpap->settings_max(CPAP_PS); - html+=STR_TR_BiLevel+QString("
"+STR_TR_EPAP+": %1 "+STR_TR_IPAP+": %2 %3
"+STR_TR_PS+": %4") - .arg(epap,0,'f',1).arg(ipap,0,'f',1).arg(STR_UNIT_CMH2O).arg(ps,0,'f',1); - } - else if (mode==MODE_ASV) { - EventDataType epap=cpap->settings_min(CPAP_EPAP); - EventDataType low=cpap->settings_min(CPAP_IPAPLo); - EventDataType high=cpap->settings_max(CPAP_IPAPHi); - EventDataType psl=cpap->settings_min(CPAP_PSMin); - EventDataType psh=cpap->settings_max(CPAP_PSMax); - html+=tr("ASV")+QString("
"+STR_TR_EPAP+": %1 "+STR_TR_IPAP+": %2 - %3 %4
"+STR_TR_PS+": %5 / %6") - .arg(epap,0,'f',1) - .arg(low,0,'f',1) - .arg(high,0,'f',1) - .arg(STR_UNIT_CMH2O) - .arg(psl,0,'f',1) - .arg(psh,0,'f',1); - } - else html+=STR_TR_Unknown; - html+="
"+STR_TR_Date+""+tr("Sleep")+""+tr("Wake")+""+STR_UNIT_Hours+"
%1%2%3%4

\n"; + ChannelID ahichan=CPAP_AHI; + QString ahiname=STR_TR_AHI; if (PROFILE.general->calculateRDI()) { - html+=QString("\n") - .arg("#F88017").arg(COLOR_Text.name()).arg(STR_TR_RDI).arg(schema::channel[CPAP_RDI].description()).arg(ahi,0,'f',2); - } else { - html+=QString("\n") - .arg("#F88017").arg(COLOR_Text.name()).arg(STR_TR_AHI).arg(schema::channel[CPAP_AHI].description()).arg(ahi,0,'f',2); + ahichan=CPAP_RDI; + ahiname=STR_TR_RDI; } + html+=""; + html+=QString("\n") + .arg("#F88017").arg(COLOR_Text.name()).arg(ahiname).arg(schema::channel[ahichan].description()).arg(ahi,0,'f',2); + html+="\n"; + html+="
%3%4 %5
%3%4 %5
%3%4   %5
\n"; + html+=getCPAPInformation(cpap); + html+=getSleepTime(cpap,oxi); + + + html+="\n"; if (cpap->machine->GetClass()==STR_MACH_ResMed || cpap->machine->GetClass()==STR_MACH_FPIcon) { cs="4 width='70%' align=center>"; @@ -985,11 +1277,11 @@ void Daily::Load(QDate date) if ((hours > 0) && PROFILE.appearance->graphSnapshots()) { // AHI Pie Chart if ((oai+hi+cai+uai+rei+fli)>0) { html+=""; - html+=""; + //html+=""; html+=QString("").arg(tr("Event Breakdown")); GAHI->setShowTitle(false); - QPixmap pixmap=GAHI->renderPixmap(150,150,false); + QPixmap pixmap=GAHI->renderPixmap(160,160,false); if (!pixmap.isNull()) { QByteArray byteArray; QBuffer buffer(&byteArray); // use buffer to store pixmap into byteArray @@ -1004,8 +1296,10 @@ void Daily::Load(QDate date) } } + html+="
 


%1
\n"; - } else { // machine is a brick + } else { + html+="\n"; if (!isBrick) { html+="\n"; if (cpap->size()>0) { @@ -1015,298 +1309,38 @@ void Daily::Load(QDate date) html+=""; html+="\n"; } - } else { + } else { // machine is a brick html+=""; - html+="\n"; + html+="\n"; html+="\n"; } html+="\n"; + html+="
 

"+tr("Impossibly short session")+"

"+tr("Zero hours??")+"

"+tr("BRICK :(")+"

"+tr("Sorry, your machine does not record data.")+"
"+tr("Sorry, your machine only provides compliance data.")+"
"+tr("Complain to your Equipment Provider!")+"
 
\n"; } - html+=""; + html+="
\n"; } // if (!CPAP) - html+="\n"; - - float percentile=PROFILE.general->prefCalcPercentile()/100.0; - - SummaryType ST_max=PROFILE.general->prefCalcMax() ? ST_MAX : ST_PERC; - const EventDataType maxperc=0.995F; - - QString midname; - if (ST_mid==ST_WAVG) midname=tr("Avg"); - else if (ST_mid==ST_AVG) midname=tr("Avg"); - else if (ST_mid==ST_PERC) midname=tr("Med"); + else html+=getSleepTime(cpap,oxi); if ((cpap && !isBrick && (cpap->hours()>0)) || oxi) { - // html+="\n"; - html+="\n"; - html+=QString("\n").arg(tr("Statistics")); - html+=QString("") - .arg(STR_TR_Channel) - .arg(STR_TR_Min) - .arg(midname) - .arg(tr("%1%").arg(percentile*100,0,'f',0)) - .arg(STR_TR_Max); - ChannelID chans[]={ - CPAP_Pressure,CPAP_EPAP,CPAP_IPAP,CPAP_PS,CPAP_PTB, - CPAP_MinuteVent, CPAP_RespRate, CPAP_RespEvent,CPAP_FLG, - CPAP_Leak, CPAP_LeakTotal, CPAP_Snore,CPAP_IE,CPAP_Ti,CPAP_Te, CPAP_TgMV, - CPAP_TidalVolume, OXI_Pulse, OXI_SPO2 - }; - int numchans=sizeof(chans)/sizeof(ChannelID); - //int suboffset=0; - int ccnt=0; - EventDataType tmp,med,perc,mx,mn; - for (int i=0;ichannelHasData(code)) { - //if (code==CPAP_LeakTotal) suboffset=PROFILEIntentionalLeak"].toDouble(); else suboffset=0; - QString tooltip=schema::channel[code].description(); - if (!schema::channel[code].units().isEmpty()) tooltip+=" ("+schema::channel[code].units()+")"; - if (ST_max==ST_MAX) { - mx=cpap->Max(code); - } else { - mx=cpap->percentile(code,maxperc); - } - mn=cpap->Min(code); - perc=cpap->percentile(code,percentile); - - if (ST_mid==ST_PERC) { - med=cpap->percentile(code,0.5); - tmp=cpap->wavg(code); - if (tmp>0 || mx==0) { - tooltip+=QString("
"+STR_TR_WAvg+": %1").arg(tmp,0,'f',2); - } - } else if (ST_mid==ST_WAVG) { - med=cpap->wavg(code); - tmp=cpap->percentile(code,0.5); - if (tmp>0 || mx==0) { - tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); - } - } else if (ST_mid==ST_AVG) { - med=cpap->avg(code); - tmp=cpap->percentile(code,0.5); - if (tmp>0 || mx==0) { - tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); - } - } - - html+=QString("
") - //.arg(QString("%3%2") //"+STR_TR_RDI+""+ - //.arg(QString::number(code)).arg(tooltip).arg(schema::channel[code].label())) - .arg(schema::channel[code].label()) - .arg(mn,0,'f',2) - .arg(med,0,'f',2) - .arg(perc,0,'f',2) - .arg(mx,0,'f',2) - .arg(tooltip); - ccnt++; - } - if (oxi && oxi->channelHasData(code)) { - QString tooltip=schema::channel[code].description(); - if (!schema::channel[code].units().isEmpty()) tooltip+=" ("+schema::channel[code].units()+")"; - //wavg=oxi->wavg(code); - mx=oxi->Max(code); - mn=oxi->Min(code); - perc=oxi->percentile(code,percentile); - //med=oxi->percentile(code,0.5); - - if (ST_mid==ST_PERC) { - med=oxi->percentile(code,0.5); - tmp=oxi->wavg(code); - if (tmp>0 || mx==0) { - tooltip+=QString("
"+STR_TR_WAvg+": %1").arg(tmp,0,'f',2); - } - } else if (ST_mid==ST_WAVG) { - med=oxi->wavg(code); - tmp=oxi->percentile(code,0.5); - if (tmp>0 || mx==0) { - tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); - } - } else if (ST_mid==ST_AVG) { - med=oxi->avg(code); - tmp=oxi->percentile(code,0.5); - if (tmp>0 || mx==0) { - tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); - } - } - - -// if ((med>0 && wavg>0) || (med==0)) { -// tooltip+=QString("
Avg: %1").arg(wavg,0,'f',2); -// } - - html+=QString("
") - .arg(QString("%2%3") - .arg(QString::number(code)).arg(schema::channel[code].label()).arg(tooltip)) - .arg(mn,0,'f',2) - .arg(med,0,'f',2) - .arg(perc,0,'f',2) - .arg(mx,0,'f',2); - ccnt++; - } - } - if (GraphView->isEmpty() && (ccnt>0)) { - html+="\n"; - html+=QString("").arg(tr("Please Note: This day just contains summary data, only limited information is available .")); - } + html+=getStatisticsInfo(cpap,oxi); } else { if (cpap && cpap->hours()==0) { } else { - html+=""; + html+="
 

%1
%1%2%3%4%5
%1%6%2%3%4%5
%1%2%3%4%5
 
%1
"+tr("No data available")+"
\n"; + html+="\n"; html+="\n"; + html+="
"+tr("No data available")+"
 
\n"; + html+="
\n"; } } - if (oxi && oxi->hasEnabledSessions()) { - html+="
"; - html+=QString("%1\n").arg(tr("Oximeter Information")); - html+=" "; - html+=""+oxi->machine->properties[STR_PROP_Brand]+" "+oxi->machine->properties[STR_PROP_Model]+"\n"; - html+=" "; - html+=QString("%1: %2 (%3)\%").arg(tr("SpO2 Desaturations")).arg(oxi->count(OXI_SPO2Drop)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_SPO2Drop)/3600.0),0,'f',2); - html+=QString("%1: %2 (%3)\%").arg(tr("Pulse Change events")).arg(oxi->count(OXI_PulseChange)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_PulseChange)/3600.0),0,'f',2); - html+=QString("%1: %2\%").arg(tr("SpO2 Baseline Used")).arg(oxi->settings_wavg(OXI_SPO2Drop),0,'f',2); // CHECKME: Should this value be wavg OXI_SPO2 isntead? - } - if (cpap && cpap->hasEnabledSessions()) { - html+="
"; - html+=QString("%1").arg(tr("Machine Settings")); - html+=" "; - int i=cpap->settings_max(CPAP_PresReliefType); - int j=cpap->settings_max(CPAP_PresReliefSet); - QString flexstr=(i>1) ? schema::channel[CPAP_PresReliefType].option(i)+" x"+QString::number(j) : STR_TR_None; - html+=QString("%1%2%3") - .arg(STR_TR_PrRelief) - .arg(schema::channel[CPAP_PresReliefType].description()) - .arg(flexstr); - QString mclass=cpap->machine->GetClass(); - if (mclass==STR_MACH_PRS1 || mclass==STR_MACH_FPIcon) { - int humid=round(cpap->settings_wavg(CPAP_HumidSetting)); - html+=QString(""+STR_TR_Humidifier+"%1%2") - .arg(schema::channel[CPAP_HumidSetting].description()) - .arg(humid==0 ? STR_GEN_Off : "x"+QString::number(humid)); - } + html+=getOximeterInformation(oxi); + html+=getMachineSettings(cpap); + html+=getSessionInformation(cpap,oxi,stage); - } - html+=""; - - if (cpap || oxi) { - html+=""; - html+=""; - html+=QString(""); - html+=""; - QFontMetrics FM(webView->font()); - QRect r=FM.boundingRect('@'); - html+=QString("").arg(r.height()*3,0,10); - html+="" -; - - QDateTime fd,ld; - bool corrupted_waveform=false; - QString tooltip; - html+=QString(""); - if (cpap) { - html+=QString(""); - for (QVector::iterator s=cpap->begin();s!=cpap->end();s++) { - fd=QDateTime::fromTime_t((*s)->first()/1000L); - ld=QDateTime::fromTime_t((*s)->last()/1000L); - int len=(*s)->length()/1000L; - int h=len/3600; - int m=(len/60) % 60; - int s1=len % 60; - tooltip=cpap->machine->GetClass()+QString(":#%1").arg((*s)->session(),8,10,QChar('0')); - // tooltip needs to lookup language.. :-/ - - QHash::iterator i=(*s)->settings.find(CPAP_BrokenWaveform); - corrupted_waveform=(i!=(*s)->settings.end()) && i.value().toBool(); - Session *sess=*s; - if (!sess->settings.contains(SESSION_ENABLED)) { - sess->settings[SESSION_ENABLED]=true; - } - bool b=sess->settings[SESSION_ENABLED].toBool(); - html+=QString("") - .arg((*s)->session()) - .arg(tooltip) - .arg(QString("%1h %2m %3s").arg(h,2,10,QChar('0')).arg(m,2,10,QChar('0')).arg(s1,2,10,QChar('0'))) - .arg((b ? "on" : "off")) - .arg(fd.date().toString(Qt::SystemLocaleShortDate)) - .arg(fd.toString("HH:mm")) - .arg(ld.toString("HH:mm")); - } - } - - if (oxi) { - html+=QString(""); - for (QVector::iterator s=oxi->begin();s!=oxi->end();s++) { - fd=QDateTime::fromTime_t((*s)->first()/1000L); - ld=QDateTime::fromTime_t((*s)->last()/1000L); - int len=(*s)->length()/1000L; - int h=len/3600; - int m=(len/60) % 60; - int s1=len % 60; - tooltip=oxi->machine->GetClass()+QString(":#%1").arg((*s)->session(),8,10,QChar('0')); - - Session *sess=*s; - if (!sess->settings.contains(SESSION_ENABLED)) { - sess->settings[SESSION_ENABLED]=true; - } - bool b=sess->settings[SESSION_ENABLED].toBool(); - - QHash::iterator i=(*s)->settings.find(CPAP_BrokenWaveform); - corrupted_waveform=(i!=(*s)->settings.end()) && i.value().toBool(); - html+=QString("") - .arg((*s)->session()) - .arg(tooltip) - .arg(QString("%1h %2m %3s").arg(h,2,10,QChar('0')).arg(m,2,10,QChar('0')).arg(s1,2,10,QChar('0'))) - .arg((b ? "on" : "off")) - .arg(fd.date().toString(Qt::SystemLocaleShortDate)) - .arg(fd.toString("HH:mm")) - .arg(ld.toString("HH:mm")); - } - } - if (stage) { - html+=QString("").arg(tr("Sleep Stage Sessions")); - for (QVector::iterator s=stage->begin();s!=stage->end();s++) { - fd=QDateTime::fromTime_t((*s)->first()/1000L); - ld=QDateTime::fromTime_t((*s)->last()/1000L); - int len=(*s)->length()/1000L; - int h=len/3600; - int m=(len/60) % 60; - int s1=len % 60; - tooltip=stage->machine->GetClass()+QString(":#%1").arg((*s)->session(),8,10,QChar('0')); - - Session *sess=*s; - if (!sess->settings.contains(SESSION_ENABLED)) { - sess->settings[SESSION_ENABLED]=true; - } - bool b=sess->settings[SESSION_ENABLED].toBool(); - - QHash::iterator i=(*s)->settings.find(CPAP_BrokenWaveform); - corrupted_waveform=(i!=(*s)->settings.end()) && i.value().toBool(); - html+=QString("") - .arg((*s)->session()) - .arg(tooltip) - .arg(QString("%1h %2m %3s").arg(h,2,10,QChar('0')).arg(m,2,10,QChar('0')).arg(s1,2,10,QChar('0'))) - .arg((b ? "on" : "off")) - .arg(fd.date().toString(Qt::SystemLocaleShortDate)) - .arg(fd.toString("HH:mm")) - .arg(ld.toString("HH:mm")); - } - } - if (corrupted_waveform) { - html+=QString("").arg(tr("One or more waveform record for this session had faulty source data. Some waveform overlay points may not match up correctly.")); - } - html+="

"+tr("Session Information")+"
 
" - "" - "" - "
 
"+STR_TR_On+""+STR_TR_Date+""+STR_TR_Start+""+STR_TR_End+""+tr("Duration")+"
"+tr("CPAP Sessions")+"
%5%6%7%3%2
"+tr("Oximetry Sessions")+"
%5%6%7%3%2
%1
%5%6%7%3%2
%1

"; - } html+=""; QColor cols[]={ diff --git a/sleepyhead/daily.h b/sleepyhead/daily.h index e4977b99..96027cb2 100644 --- a/sleepyhead/daily.h +++ b/sleepyhead/daily.h @@ -287,6 +287,14 @@ private: void updateCube(); void updateGraphCombo(); + QString getSessionInformation(Day * cpap, Day * oxi, Day * stage); + QString getMachineSettings(Day * cpap); + QString getStatisticsInfo(Day * cpap, Day * oxi); + QString getCPAPInformation(Day * cpap); + QString getOximeterInformation(Day * oxi); + QString getEventBreakdown(Day * cpap); + QString getSleepTime(Day * cpap, Day * oxi); + gGraph *PRD,*FRW,*GAHI,*TAP,*LEAK,*SF,*TAP_EAP,*TAP_IAP,*PULSE,*SPO2, *SNORE,*RR,*MP,*MV,*TV,*FLG,*PTB,*OF, *THPR, *PLETHY,*TI,*TE, *RE, *IE, *AHI, *RDI, *STAGE; //*TgMV, diff --git a/sleepyhead/main.cpp b/sleepyhead/main.cpp index 902f74a9..32e6885e 100644 --- a/sleepyhead/main.cpp +++ b/sleepyhead/main.cpp @@ -164,6 +164,7 @@ int main(int argc, char *argv[]) //////////////////////////////////////////////////////////////////////////////////////////// // Register Importer Modules //////////////////////////////////////////////////////////////////////////////////////////// + initialize(); PRS1Loader::Register(); CMS50Loader::Register(); //ZEOLoader::Register(); // Use outside of directory importer.. @@ -258,7 +259,6 @@ int main(int argc, char *argv[]) a.installTranslator(&translator); a.setApplicationName(STR_TR_SleepyHead); - initialize();