diff --git a/Graphs/gGraphView.cpp b/Graphs/gGraphView.cpp index bf4ded2d..1ea0e9dd 100644 --- a/Graphs/gGraphView.cpp +++ b/Graphs/gGraphView.cpp @@ -1917,6 +1917,9 @@ void gGraphView::ResetBounds(bool refresh) //short group) } if (!g) g=m_graphs[0]; + m_minx=g->min_x; + m_maxx=g->max_x; + qint64 xx=g->max_x - g->min_x; double d=xx/86400000L; int h=xx/3600000L; @@ -1934,6 +1937,11 @@ void gGraphView::ResetBounds(bool refresh) //short group) } updateScale(); } +void gGraphView::GetXBounds(qint64 & st,qint64 & et) +{ + st=m_minx; + et=m_maxx; +} void gGraphView::SetXBounds(qint64 minx, qint64 maxx,short group,bool refresh) { @@ -1942,6 +1950,9 @@ void gGraphView::SetXBounds(qint64 minx, qint64 maxx,short group,bool refresh) m_graphs[i]->SetXBounds(minx,maxx); } } + m_minx=minx; + m_maxx=maxx; + qint64 xx=maxx-minx; double d=xx/86400000L; int h=xx/3600000L; diff --git a/Graphs/gGraphView.h b/Graphs/gGraphView.h index 346807a2..a554cd97 100644 --- a/Graphs/gGraphView.h +++ b/Graphs/gGraphView.h @@ -402,6 +402,7 @@ public: float scaleY() { return m_scaleY; } + void GetXBounds(qint64 & st,qint64 & et); void ResetBounds(bool refresh=true); //short group=0); void SetXBounds(qint64 minx, qint64 maxx, short group=0,bool refresh=true); void SaveSettings(QString title); @@ -504,6 +505,8 @@ protected: QString m_emptytext; bool m_showsplitter; + + qint64 m_minx,m_maxx; signals: diff --git a/SleepLib/calcs.cpp b/SleepLib/calcs.cpp index d98b0697..76d3bc4e 100644 --- a/SleepLib/calcs.cpp +++ b/SleepLib/calcs.cpp @@ -7,51 +7,9 @@ #include "calcs.h" #include "profiles.h" -Calculation::Calculation(ChannelID id,QString name) - :m_id(id),m_name(name) -{ -} -Calculation::~Calculation() -{ -} - -CalcRespRate::CalcRespRate(ChannelID id) - :Calculation(id,"Resp. Rate") -{ -} - -// Generate RespiratoryRate graph -int CalcRespRate::calculate(Session *session) -{ - if (session->machine()->GetType()!=MT_CPAP) return 0; - if (session->eventlist.contains(CPAP_RespRate)) return 0; // already exists? - - if (!session->eventlist.contains(CPAP_FlowRate)) return 0; //need flow waveform - - EventList *flow, *rr, *tv=NULL, *mv=NULL; - if (!session->eventlist.contains(CPAP_TidalVolume)) { - tv=new EventList(EVL_Event); - session->eventlist[CPAP_TidalVolume].push_back(tv); - } - if (!session->eventlist.contains(CPAP_MinuteVent)) { - mv=new EventList(EVL_Event); - session->eventlist[CPAP_MinuteVent].push_back(mv); - } - int cnt=0; - for (int ws=0; ws < session->eventlist[CPAP_FlowRate].size(); ws++) { - flow=session->eventlist[CPAP_FlowRate][ws]; - if (flow->count() > 5) { - rr=new EventList(EVL_Event); - session->eventlist[CPAP_RespRate].push_back(rr); - - cnt+=filterFlow(flow,rr,tv,mv,flow->rate()); - } - } - return cnt; -} - -int CalcRespRate::filterFlow(EventList *in, EventList *out, EventList *tv, EventList *mv, double rate) +// Support function for calcRespRate() +int filterFlow(EventList *in, EventList *out, EventList *tv, EventList *mv, double rate) { int size=in->count(); @@ -273,6 +231,37 @@ int CalcRespRate::filterFlow(EventList *in, EventList *out, EventList *tv, Event return out->count(); } +// Generate RespiratoryRate graph +int calcRespRate(Session *session) +{ + if (session->machine()->GetType()!=MT_CPAP) return 0; + if (session->eventlist.contains(CPAP_RespRate)) return 0; // already exists? + + if (!session->eventlist.contains(CPAP_FlowRate)) return 0; //need flow waveform + + EventList *flow, *rr, *tv=NULL, *mv=NULL; + if (!session->eventlist.contains(CPAP_TidalVolume)) { + tv=new EventList(EVL_Event); + session->eventlist[CPAP_TidalVolume].push_back(tv); + } + if (!session->eventlist.contains(CPAP_MinuteVent)) { + mv=new EventList(EVL_Event); + session->eventlist[CPAP_MinuteVent].push_back(mv); + } + int cnt=0; + for (int ws=0; ws < session->eventlist[CPAP_FlowRate].size(); ws++) { + flow=session->eventlist[CPAP_FlowRate][ws]; + if (flow->count() > 5) { + rr=new EventList(EVL_Event); + session->eventlist[CPAP_RespRate].push_back(rr); + + cnt+=filterFlow(flow,rr,tv,mv,flow->rate()); + } + } + return cnt; +} + + EventDataType calcAHI(Session *session,qint64 start, qint64 end) { double hours,ahi,cnt; @@ -297,12 +286,7 @@ EventDataType calcAHI(Session *session,qint64 start, qint64 end) return ahi; } -CalcAHIGraph::CalcAHIGraph(ChannelID id): - Calculation(id,"AHI/hour") -{ -} - -int CalcAHIGraph::calculate(Session *session) +int calcAHIGraph(Session *session) { if (session->machine()->GetType()!=MT_CPAP) return 0; if (session->eventlist.contains(CPAP_AHI)) return 0; // abort if already there diff --git a/SleepLib/calcs.h b/SleepLib/calcs.h index 9838e9e0..96d471b3 100644 --- a/SleepLib/calcs.h +++ b/SleepLib/calcs.h @@ -8,39 +8,14 @@ #include "day.h" -class Calculation -{ -public: - Calculation(ChannelID id,QString name); - virtual ~Calculation(); - virtual int calculate(Session *session)=0; -protected: - ChannelID m_id; - QString m_name; -}; - -class CalcRespRate:public Calculation -{ -public: - CalcRespRate(ChannelID id=CPAP_RespRate); - virtual int calculate(Session *session); -protected: - int filterFlow(EventList *in, EventList *out,EventList *tv, EventList *mv, double rate); -}; - -class CalcAHIGraph:public Calculation -{ -public: - CalcAHIGraph(ChannelID id=CPAP_AHI); - virtual int calculate(Session *session); -protected: -}; +int calcRespRate(Session *session); +int calcAHIGraph(Session *session); +EventDataType calcAHI(Session *session,qint64 start=0, qint64 end=0); int calcLeaks(Session *session); int calcPulseChange(Session *session); int calcSPO2Drop(Session *session); -EventDataType calcAHI(Session *session,qint64 start=0, qint64 end=0); #endif // CALCS_H diff --git a/SleepLib/session.cpp b/SleepLib/session.cpp index 8c03b5a2..a5a8571d 100644 --- a/SleepLib/session.cpp +++ b/SleepLib/session.cpp @@ -452,12 +452,8 @@ bool Session::LoadEvents(QString filename) void Session::UpdateSummaries() { - CalcAHIGraph ahi; - CalcRespRate calc; - - ahi.calculate(this); - calc.calculate(this); - + calcAHIGraph(this); + calcRespRate(this); calcLeaks(this); calcSPO2Drop(this); diff --git a/daily.cpp b/daily.cpp index 536120a7..d7b39252 100644 --- a/daily.cpp +++ b/daily.cpp @@ -43,7 +43,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared, MainWindow *mw) ui->setupUi(this); // Remove Incomplete Extras Tab - ui->tabWidget->removeTab(3); + //ui->tabWidget->removeTab(3); QList a; a.push_back(300); @@ -73,6 +73,11 @@ Daily::Daily(QWidget *parent,gGraphView * shared, MainWindow *mw) scrollbar->setSizePolicy(QSizePolicy::Maximum,QSizePolicy::Expanding); scrollbar->setMaximumWidth(20); + ui->bookmarkTable->setColumnCount(2); + ui->bookmarkTable->setColumnWidth(0,70); + //ui->bookmarkTable->setEditTriggers(QAbstractItemView::SelectedClicked); + //ui->bookmarkTable->setColumnHidden(2,true); + //ui->bookmarkTable->setColumnHidden(3,true); GraphView->setScrollBar(scrollbar); layout->addWidget(GraphView,1); layout->addWidget(scrollbar,0); @@ -461,9 +466,11 @@ void Daily::LoadDate(QDate date) void Daily::on_calendar_selectionChanged() { + if (previous_date.isValid()) Unload(previous_date); + ZombieMeterMoved=false; Load(ui->calendar->selectedDate()); ui->calButton->setText(ui->calendar->selectedDate().toString(Qt::TextDate)); ui->calendar->setFocus(Qt::ActiveWindowFocusReason); @@ -787,30 +794,121 @@ void Daily::Load(QDate date) ui->webView->setHtml(html); ui->JournalNotes->clear(); + + ui->bookmarkTable->clear(); + ui->bookmarkTable->setRowCount(0); + QStringList sl; + sl.append("Starts"); + sl.append("Notes"); + ui->bookmarkTable->setHorizontalHeaderLabels(sl); + ui->weightSpinBox->setValue(0); + ui->ZombieMeter->setValue(50); Session *journal=GetJournalSession(date); if (journal) { - ui->JournalNotes->setHtml(journal->settings[Journal_Notes].toString()); + bool ok; + if (journal->settings.contains(Journal_Notes)) + ui->JournalNotes->setHtml(journal->settings[Journal_Notes].toString()); + + if (journal->settings.contains("Weight")) + ui->weightSpinBox->setValue(journal->settings["Weight"].toDouble(&ok)); + + if (journal->settings.contains("ZombieMeter")) + ui->ZombieMeter->setValue(journal->settings["ZombieMeter"].toDouble(&ok)); + + if (journal->settings.contains("BookmarkStart")) { + QVariantList start=journal->settings["BookmarkStart"].toList(); + QVariantList end=journal->settings["BookmarkEnd"].toList(); + QStringList notes=journal->settings["BookmarkNotes"].toStringList(); + + bool ok; + for (int i=0;ibookmarkTable->rowCount(); + ui->bookmarkTable->insertRow(i); + QTableWidgetItem *tw=new QTableWidgetItem(notes.at(i)); + QTableWidgetItem *dw=new QTableWidgetItem(d.time().toString("HH:mm:ss")); + dw->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled); + ui->bookmarkTable->setItem(i,0,dw); + ui->bookmarkTable->setItem(i,1,tw); + tw->setData(Qt::UserRole,st); + tw->setData(Qt::UserRole+1,et); + } + } } } void Daily::Unload(QDate date) { Session *journal=GetJournalSession(date); - if (!ui->JournalNotes->toPlainText().isEmpty()) { - QString jhtml=ui->JournalNotes->toHtml(); - if (journal) { - if (journal->settings[Journal_Notes]!=jhtml) { - journal->settings[Journal_Notes]=jhtml; - journal->SetChanged(true); - } - } else { - journal=CreateJournalSession(date); + bool nonotes=ui->JournalNotes->toPlainText().isEmpty(); + bool ok; + if (journal) { + QString jhtml=ui->JournalNotes->toHtml(); + if ((!journal->settings.contains(Journal_Notes) && !nonotes) || (!nonotes && (journal->settings[Journal_Notes]!=jhtml))) { journal->settings[Journal_Notes]=jhtml; journal->SetChanged(true); } - + if ((!journal->settings.contains("Weight") && (ui->weightSpinBox->value()>0)) || (journal->settings["Weight"].toDouble(&ok)!=ui->weightSpinBox->value())) { + journal->settings["Weight"]=ui->weightSpinBox->value(); + journal->SetChanged(true); + } + if ((!journal->settings.contains("ZombieMeter") && (ui->ZombieMeter->value()!=50)) || (journal->settings["ZombieMeter"].toDouble(&ok)!=ui->ZombieMeter->value())) { + journal->settings["ZombieMeter"]=ui->ZombieMeter->value(); + journal->SetChanged(true); + } + bool ok; + if (ui->bookmarkTable->rowCount()>0) { + QVariantList start; + QVariantList end; + QStringList notes; + QTableWidgetItem *item; + for (int row=0;rowbookmarkTable->rowCount();row++) { + item=ui->bookmarkTable->item(row,1); + start.push_back(item->data(Qt::UserRole)); + end.push_back(item->data(Qt::UserRole+1)); + notes.push_back(item->text()); + } + journal->settings["BookmarkStart"]=start; + journal->settings["BookmarkEnd"]=end; + journal->settings["BookmarkNotes"]=notes; + } + } else { + if (!nonotes || ZombieMeterMoved || (ui->weightSpinBox->value() > 0) || (ui->bookmarkTable->rowCount()>0)) { + journal=CreateJournalSession(date); + if (!nonotes) { + journal->settings[Journal_Notes]=ui->JournalNotes->toHtml(); + journal->SetChanged(true); + } + if (ZombieMeterMoved) { + journal->settings["ZombieMeter"]=ui->ZombieMeter->value(); + journal->SetChanged(true); + } + if (ui->weightSpinBox->value() > 0) { + journal->settings["Weight"]=ui->weightSpinBox->value(); + journal->SetChanged(true); + } + if (ui->bookmarkTable->rowCount()>0) { + QVariantList start; + QVariantList end; + QStringList notes; + QTableWidgetItem *item; + for (int row=0;rowbookmarkTable->rowCount();row++) { + item=ui->bookmarkTable->item(row,1); + start.push_back(item->data(Qt::UserRole)); + end.push_back(item->data(Qt::UserRole+1)); + notes.push_back(item->text()); + } + journal->settings["BookmarkStart"]=start; + journal->settings["BookmarkEnd"]=end; + journal->settings["BookmarkNotes"]=notes; + } + } } + if (journal) { Machine *jm=PROFILE.GetMachine(MT_JOURNAL); if (jm) jm->SaveSession(journal); @@ -925,16 +1023,6 @@ Session * Daily::GetJournalSession(QDate date) // Get the first journal session return *s; return NULL; } -void Daily::on_EnergySlider_sliderMoved(int position) -{ - position=position; - //Session *s=GetJournalSession(previous_date); - //if (!s) - // s=CreateJournalSession(previous_date); - - //s->summary[JOURNAL_Energy]=position; - //s->SetChanged(true); -} void Daily::UpdateCPAPGraphs(Day *day) { @@ -1113,3 +1201,63 @@ void Daily::on_evViewSlider_valueChanged(int value) } } +void Daily::on_bookmarkTable_itemClicked(QTableWidgetItem *item) +{ + int row=item->row(); + qint64 st,et; + + QTableWidgetItem *it=ui->bookmarkTable->item(row,1); + bool ok; + st=it->data(Qt::UserRole).toLongLong(&ok); + et=it->data(Qt::UserRole+1).toLongLong(&ok); + GraphView->SetXBounds(st,et); + GraphView->updateGL(); +} + +void Daily::on_bookmarkTable_itemActivated(QTableWidgetItem *item) +{ +} + +void Daily::on_addBookmarkButton_clicked() +{ + qint64 st,et; + GraphView->GetXBounds(st,et); + QDateTime d=QDateTime::fromTime_t(st/1000L); + int row=ui->bookmarkTable->rowCount(); + ui->bookmarkTable->insertRow(row); + QTableWidgetItem *tw=new QTableWidgetItem("Bookmark at "+d.time().toString("HH:mm:ss")); + QTableWidgetItem *dw=new QTableWidgetItem(d.time().toString("HH:mm:ss")); + dw->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled); + ui->bookmarkTable->setItem(row,0,dw); + ui->bookmarkTable->setItem(row,1,tw); + tw->setData(Qt::UserRole,st); + tw->setData(Qt::UserRole+1,et); + + //ui->bookmarkTable->setItem(row,2,new QTableWidgetItem(QString::number(st))); + //ui->bookmarkTable->setItem(row,3,new QTableWidgetItem(QString::number(et))); +} + +void Daily::on_removeBookmarkButton_clicked() +{ + int row=ui->bookmarkTable->currentRow(); + if (row>=0) { + ui->bookmarkTable->removeRow(row); + } +} + +void Daily::on_ZombieMeter_actionTriggered(int action) +{ + ZombieMeterMoved=true; + qDebug() << "ZombieMeter" << action; +} + +//void Daily::on_EnergySlider_sliderMoved(int position) +//{ + // position=position; + //Session *s=GetJournalSession(previous_date); + //if (!s) + // s=CreateJournalSession(previous_date); + + //s->summary[JOURNAL_Energy]=position; + //s->SetChanged(true); +//} diff --git a/daily.h b/daily.h index 9ac61679..2d176415 100644 --- a/daily.h +++ b/daily.h @@ -17,6 +17,7 @@ #include #include #include +#include #include "Graphs/gSummaryChart.h" #include @@ -55,7 +56,6 @@ private slots: void on_JournalNotesBold_clicked(); void on_JournalNotesFontsize_activated(int index); void on_JournalNotesColour_clicked(); - void on_EnergySlider_sliderMoved(int position); void on_treeWidget_itemSelectionChanged(); @@ -75,6 +75,16 @@ private slots: void on_treeWidget_itemClicked(QTreeWidgetItem *item, int column); void on_graphtogglebutton_toggled(bool); + void on_ZombieMeter_actionTriggered(int action); + + void on_addBookmarkButton_clicked(); + + void on_removeBookmarkButton_clicked(); + + void on_bookmarkTable_itemActivated(QTableWidgetItem *item); + + void on_bookmarkTable_itemClicked(QTableWidgetItem *item); + protected: private: @@ -110,6 +120,10 @@ private: gGraphView *GraphView,*snapGV; MyScrollBar *scrollbar; QHBoxLayout *layout; + bool ZombieMeterMoved; }; #endif // DAILY_H + + + diff --git a/daily.ui b/daily.ui index 43fb69fc..4a63c028 100644 --- a/daily.ui +++ b/daily.ui @@ -6,8 +6,8 @@ 0 0 - 671 - 433 + 687 + 484 @@ -248,7 +248,7 @@ - 1 + 3 true @@ -489,182 +489,176 @@ + + + + + 4 + + + 4 + + + 2 + + + 0 + + + 2 + + + 2 + + + + + + + + 0 + 0 + + + + + 75 + true + + + + Zombie + + + + + + + 50 + + + Qt::Horizontal + + + QSlider::TicksAbove + + + 5 + + + + + + + + 0 + 0 + + + + + 75 + true + + + + Awesome + + + Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter + + + + + + + + 50 + false + + + + I'm feeling... + + + Qt::AlignCenter + + + + + + + + + + 0 + 0 + + + + Weight + + + + + + + 399.990000000000009 + + + + + + - + - Extras + Bookmarks - + - - - - 0 - 0 - - - - QFrame::StyledPanel - - - QFrame::Sunken - + - This tab is fake.. What to put here? - - - false - - - Qt::AlignCenter + Bookmark Selection - - - 2 + + + true - - - - - - - 0 - 0 - - - - Energy Level - - - - - - - Qt::Horizontal - - - QSlider::TicksAbove - - - - - - - - - - - - 0 - 0 - - - - Mood - - - - - - - 99 - - - Qt::Horizontal - - - QSlider::TicksAbove - - - - - - + + QAbstractItemView::SingleSelection + + + QAbstractItemView::SelectRows + + + true + + + true + + + + Starts + + + + + Notes + + + - - - 2 + + + Remove Bookmark - - - - - - QLayout::SetMinimumSize - - - - - - 0 - 0 - - - - Weight - - - - - - - - - - 999.990000000000009 - - - - - - - - - - - - - QLayout::SetMinimumSize - - - - - - 0 - 0 - - - - Glucose - - - - - - - - - - - - - - - - Qt::Vertical - - - - 20 - 77 - - - +