diff --git a/SleepLib/day.cpp b/SleepLib/day.cpp index 04cb66a3..9dbc9085 100644 --- a/SleepLib/day.cpp +++ b/SleepLib/day.cpp @@ -141,15 +141,12 @@ EventDataType Day::percentile(ChannelID code,EventDataType percentile) QVector ar; for (s=sessions.begin();s!=sessions.end();s++) { Session & sess=*(*s); - QHash >::iterator ei=sess.eventlist.find(code); - - if (ei==sess.eventlist.end()) - continue; - for (int e=0;ecount();j++) { - ar.push_back(ev->data(j)); + QHash > ::iterator ei=sess.m_valuesummary.find(code); + if (ei==sess.m_valuesummary.end()) continue; + EventDataType gain=sess.m_gain[code]; + for (QHash::iterator i=ei.value().begin();i!=ei.value().end();i++) { + for (int j=0;jp90(code); - - QHash::iterator i=m_p90.find(code); - if (i!=m_p90.end()) - return i.value(); - - - QVector::iterator s; - - // Don't assume sessions are in order. - - unsigned cnt=0,c; - EventDataType p; - - QMap pmap; - QMap tmap; - QMap::iterator pi; - for (s=sessions.begin();s!=sessions.end();s++) { - Session & sess=*(*s); - c=sess.count(code); - if (c>0) { - cnt+=c; - p=sess.p90(code); //percentile(code,0.9); - if (!pmap.contains(p)) { - pmap[p]=c; - tmap[p]=sess.hours(); - } else { - pmap[p]+=c; - tmap[p]+=sess.hours(); - } - } - } - EventDataType val; - size=pmap.size(); - if (!size) { - m_p90[code]=val=0; - return val; - } else if (size==1) { - m_p90[code]=val=pmap.begin().key(); - return val; - } - - OpenEvents(); - EventDataType realp90=percentile(code,0.9); - - val=realp90; - /*double s0=0,s1=0,s2=0,s3=0; - - for (pi=pmap.begin();pi!=pmap.end();pi++) { - s2=tmap[pi.key()]; - s3=pi.value(); - s0+=pi.key() * s3 * s2; - s1+=s3*s2; - } - if (s1==0) - return 0; - - val=s0/s1; - qDebug() << first() << code << realp90 << val; */ - - m_p90[code]=val; - return val; + return percentile(code,0.90); } EventDataType Day::avg(ChannelID code) diff --git a/SleepLib/day.h b/SleepLib/day.h index 628c5204..34321119 100644 --- a/SleepLib/day.h +++ b/SleepLib/day.h @@ -71,13 +71,6 @@ public: //! \brief Returns if the cache contains SummaryType information about the requested code bool hasData(ChannelID code, SummaryType type); - //! \brief Cache for holding slow calculated 90th Percentile for this day - QHash m_p90; - - //EventDataType percentile(ChannelID mc,double percent); - - // Note, the following convert to doubles without considering the consequences fully.? - //! \brief Returns the Average of all Sessions setting 'code' for this day EventDataType settings_avg(ChannelID code); diff --git a/SleepLib/machine.cpp b/SleepLib/machine.cpp index 3c0ecf09..e47c90fc 100644 --- a/SleepLib/machine.cpp +++ b/SleepLib/machine.cpp @@ -266,7 +266,7 @@ bool Machine::Load() if (sess->LoadSummary(s.value()[0])) { sess->SetEventFile(s.value()[1]); - sess->OpenEvents(); + //sess->OpenEvents(); AddSession(sess,profile); } else { qWarning() << "Error unpacking summary data"; diff --git a/SleepLib/profiles.cpp b/SleepLib/profiles.cpp index d9a20189..2180317a 100644 --- a/SleepLib/profiles.cpp +++ b/SleepLib/profiles.cpp @@ -63,46 +63,6 @@ Profile::Profile(QString path) } bool Profile::Save(QString filename) { - if (p_profile==this) { - QString cachefile=Get("{DataFolder}")+QDir::separator()+"cache.day"; - QFile f(cachefile); - if (ExistsAndTrue("TrashDayCache")) { - if (f.exists()) { - f.remove(); - } - } else { - QMap >::iterator di; - QHash > > cache; - - QHash > >::iterator ci; - for (di=daylist.begin();di!=daylist.end();di++) { - QDate date=di.key(); - for (QList::iterator d=di.value().begin();d!=di.value().end();d++) { - Day *day=*d; - MachineID mach=day->machine->id(); - QHash::iterator i; - - for (i=day->m_p90.begin();i!=day->m_p90.end();i++) { - cache[mach][date][i.key()]=day->m_p90[i.key()]; - } - } - } - if (f.open(QFile::WriteOnly)) { - QDataStream out(&f); - out.setVersion(QDataStream::Qt_4_6); - out.setByteOrder(QDataStream::LittleEndian); - quint16 size=cache.size(); - out << size; - for (ci=cache.begin();ci!=cache.end();ci++) { - quint32 mid=ci.key(); - out << mid; - out << ci.value(); - } - f.close(); - } - } - (*this)["TrashDayCache"]=false; - } return Preferences::Save(filename); } @@ -145,29 +105,6 @@ void Profile::LoadMachineData() { QHash > > cache; - QString filename=Get("{DataFolder}")+QDir::separator()+"cache.day"; - QFile f(filename); - if (f.exists(filename) && (f.open(QFile::ReadOnly))) { - QDataStream in(&f); - in.setVersion(QDataStream::Qt_4_6); - in.setByteOrder(QDataStream::LittleEndian); - - quint16 size; - quint32 mid; - in >> size; - for (int i=0;i> mid; - in >> cache[mid]; - } - PROFILE.general->setRebuildCache(false); - } else { - if (mainwin) { - mainwin->Notify(QObject::tr("Caching session data, this may take a little while.")); - PROFILE.general->setRebuildCache(true); - - QApplication::processEvents(); - } - } for (QHash::iterator i=machlist.begin(); i!=machlist.end(); i++) { Machine *m=i.value(); @@ -197,18 +134,6 @@ void Profile::LoadMachineData() m->Load(); } } - for (QMap >::iterator di=daylist.begin();di!=daylist.end();di++) { - for (QList::iterator d=di.value().begin();d!=di.value().end();d++) { - Day *day=*d; - MachineID mid=day->machine->id(); - if (cache.contains(mid)) { - if (cache[mid].contains(di.key())) { - day->m_p90=cache[mid][di.key()]; - } - } - } - } - // Load Day Cache here.. } /** @@ -798,6 +723,14 @@ EventDataType Profile::calcSettingsMax(ChannelID code, MachineType mt, QDate sta if (max<=-99999999) max=0; return max; } + +struct CountSummary { + CountSummary(EventStoreType v) :val(v), count(0), time(0) {} + EventStoreType val; + EventStoreType count; + quint32 time; +}; + EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, MachineType mt, QDate start, QDate end) { if (!start.isValid()) start=LastDay(mt); @@ -807,32 +740,58 @@ EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, Mac // This is one messy function.. It requires all data to be loaded.. :( + QHash summary; + QHash::iterator sumi; QVector array; + + QHash >::iterator vsi; + float gain; + bool setgain=false; //double val=0,tmp,hours=0; do { Day * day=GetDay(date,mt); if (day) { - bool open=day->eventsLoaded(); - if (!open) - day->OpenEvents(); for (int i=0;isize();i++) { for (QVector::iterator s=day->begin();s!=day->end();s++) { - QHash >::iterator el=(*s)->eventlist.find(code); + Session *sess=*s; + gain=sess->m_gain[code]; + if (!gain) gain=1; + setgain=true; + vsi=sess->m_valuesummary.find(code); + if (vsi==sess->m_valuesummary.end()) continue; + + QHash & vsum=vsi.value(); + + for (QHash::iterator k=vsum.begin();k!=vsum.end();k++) { + for (int z=0;z >::iterator el=(*s)->eventlist.find(code); if (el==(*s)->eventlist.end()) continue; for (int j=0;jCloseEvents(); } date=date.addDays(1); } while (date::iterator k=summary.begin();k!=summary.end();k++) { +// for (int i=0;iarray.size()-1) idx=array.size()-1; diff --git a/SleepLib/session.cpp b/SleepLib/session.cpp index 1790e8c2..65e947ea 100644 --- a/SleepLib/session.cpp +++ b/SleepLib/session.cpp @@ -22,7 +22,7 @@ const quint16 filetype_data=1; // This is the uber important database version for SleepyHeads internal storage // Increment this after stuffing with Session's save & load code. -const quint16 summary_version=7; +const quint16 summary_version=9; const quint16 events_version=8; Session::Session(Machine * m,SessionID session) @@ -145,7 +145,9 @@ bool Session::StoreSummary(QString filename) out << m_firstchan; out << m_lastchan; - + out << m_valuesummary; + out << m_timesummary; + out << m_gain; // First output the Machine Code and type for each summary record /* for (QHash::iterator i=settings.begin(); i!=settings.end(); i++) { @@ -302,9 +304,20 @@ bool Session::LoadSummary(QString filename) in >> m_sph; in >> m_firstchan; in >> m_lastchan; + + if (version >= 8) { + in >> m_valuesummary; + in >> m_timesummary; + if (version >= 9) { + in >> m_gain; + } + } + + } if (version >::iterator ev=eventlist.find(code); + if (ev==eventlist.end()) return; + + QHash valsum; + QHash timesum; + QHash::iterator vsi; + QHash::iterator tsi; + + EventDataType raw,lastraw=0; + qint64 time,lasttime=0; + qint32 len; + for (int i=0;i0) { + len=(time-lasttime) / 1000; + tsi=timesum.find(lastraw); + if (tsi==timesum.end()) timesum[raw]=len; + else tsi.value()+=len; + } + lastraw=raw; + lasttime=time; + } + } + if (valsum.size()==0) return; + + m_valuesummary[code]=valsum; + m_timesummary[code]=timesum; +} + void Session::UpdateSummaries() { calcAHIGraph(this); @@ -547,6 +600,12 @@ void Session::UpdateSummaries() id=c.key(); if (schema::channel[id].type()==schema::DATA) { //sum(id); // avg calculates this and cnt. + if (c.value().size()>0) { + EventList * el=c.value()[0]; + EventDataType gain=el->gain(); + m_gain[id]=gain; + } else m_gain[id]=0; + updateCountSummary(id); Min(id); Max(id); count(id); diff --git a/SleepLib/session.h b/SleepLib/session.h index 27e70365..02aeedb2 100644 --- a/SleepLib/session.h +++ b/SleepLib/session.h @@ -142,6 +142,11 @@ public: QHash m_firstchan; QHash m_lastchan; + QHash > m_valuesummary; + QHash > m_timesummary; + QHash m_gain; + + void updateCountSummary(ChannelID code); // UpdateSummaries may recalculate all these, but it may be faster setting upfront void setCount(ChannelID id,int val) { m_cnt[id]=val; } diff --git a/mainwindow.cpp b/mainwindow.cpp index 6ed6f3ee..b602532e 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -512,14 +512,13 @@ void MainWindow::on_summaryButton_clicked() mach.append(cpap_machines); mach.append(oximeters); + int cpapdays=PROFILE.countDays(MT_CPAP,firstcpap,lastcpap); CPAPMode cpapmode=(CPAPMode)p_profile->calcSettingsMax(CPAP_Mode,MT_CPAP,firstcpap,lastcpap); - if (mach.size()==0) { + if (cpapdays==0) { html+="

No Machine Data Imported

"; } else { - - html+="
"; html+=QString("

Summary Information as of %1

").arg(lastcpap.toString(Qt::SystemLocaleLongDate)); html+=QString("");