/* Daily Panel Copyright (c)2011 Mark Watkins License: GPL */ #include #include #include #include #include #include #include #include #include #include #include #include #include //#include //#include #include "daily.h" #include "ui_daily.h" #include "common_gui.h" #include "SleepLib/profiles.h" #include "SleepLib/session.h" #include "Graphs/graphdata_custom.h" #include "Graphs/gLineOverlay.h" #include "Graphs/gFlagsLine.h" #include "Graphs/gFooBar.h" #include "Graphs/gXAxis.h" #include "Graphs/gYAxis.h" #include "Graphs/gSegmentChart.h" #include "Graphs/gStatsLine.h" //extern QProgressBar *qprogress; extern MainWindow * mainwin; const int min_height=150; Daily::Daily(QWidget *parent,gGraphView * shared) :QWidget(parent), ui(new Ui::Daily) { ui->setupUi(this); // Remove Incomplete Extras Tab //ui->tabWidget->removeTab(3); ZombieMeterMoved=false; BookmarksChanged=false; QList a; a.push_back(300); a.push_back(this->width()-300); ui->splitter_2->setStretchFactor(1,1); ui->splitter_2->setSizes(a); ui->splitter_2->setStretchFactor(1,1); layout=new QHBoxLayout(ui->graphMainArea); layout->setSpacing(0); layout->setMargin(0); layout->setContentsMargins(0,0,0,0); ui->graphMainArea->setLayout(layout); //ui->graphMainArea->setLayout(layout); ui->graphMainArea->setAutoFillBackground(false); GraphView=new gGraphView(ui->graphMainArea,shared); GraphView->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding); snapGV=new gGraphView(ui->graphMainArea); snapGV->setMinimumSize(172,172); snapGV->hideSplitter(); snapGV->hide(); scrollbar=new MyScrollBar(ui->graphMainArea); scrollbar->setOrientation(Qt::Vertical); 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); int default_height=PROFILE.appearance->graphHeight(); SF=new gGraph(GraphView,STR_TR_EventFlags,STR_TR_EventFlags,default_height); FRW=new gGraph(GraphView,tr("Flow Rate"),schema::channel[CPAP_FlowRate].description()+"\n("+schema::channel[CPAP_FlowRate].units()+")",default_height); if (PROFILE.general->calculateRDI()) { AHI=new gGraph(GraphView,tr("RDI"),schema::channel[CPAP_RDI].description()+"\n("+schema::channel[CPAP_RDI].units()+")",default_height); } else AHI=new gGraph(GraphView,tr("AHI"),schema::channel[CPAP_AHI].description()+"\n("+schema::channel[CPAP_AHI].units()+")",default_height); MP=new gGraph(GraphView,tr("Mask Pressure"),schema::channel[CPAP_MaskPressure].description()+"\n("+schema::channel[CPAP_MaskPressure].units()+")",default_height); PRD=new gGraph(GraphView,tr("Pressure"),schema::channel[CPAP_Pressure].description()+"\n("+schema::channel[CPAP_Pressure].units()+")",default_height); LEAK=new gGraph(GraphView,tr("Leak"),schema::channel[CPAP_Leak].description()+"\n("+schema::channel[CPAP_Leak].units()+")",default_height); SNORE=new gGraph(GraphView,tr("Snore"),schema::channel[CPAP_Snore].description()+"\n("+schema::channel[CPAP_Snore].units()+")",default_height); 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); 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); 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); SPO2=new gGraph(GraphView,STR_TR_SpO2,schema::channel[OXI_SPO2].description()+"\n("+schema::channel[OXI_SPO2].units()+")",default_height,oxigrp); PLETHY=new gGraph(GraphView,STR_TR_Plethy,schema::channel[OXI_Plethy].description()+"\n("+schema::channel[OXI_Plethy].units()+")",default_height,oxigrp); // Event Pie Chart (for snapshot purposes) // TODO: Convert snapGV to generic for snapshotting multiple graphs (like reports does) // TAP=new gGraph(GraphView,"Time@Pressure",STR_UNIT_CMH2O,100); // TAP->showTitle(false); // gTAPGraph * tap=new gTAPGraph(CPAP_Pressure,GST_Line); // TAP->AddLayer(AddCPAP(tap)); //TAP->setMargins(0,0,0,0); GAHI=new gGraph(snapGV,tr("Breakdown"),tr("events"),172); gSegmentChart * evseg=new gSegmentChart(GST_Pie); evseg->AddSlice(CPAP_Hypopnea,QColor(0x40,0x40,0xff,0xff),tr("H")); evseg->AddSlice(CPAP_Apnea,QColor(0x20,0x80,0x20,0xff),tr("A")); evseg->AddSlice(CPAP_Obstructive,QColor(0x40,0xaf,0xbf,0xff),tr("OA")); evseg->AddSlice(CPAP_ClearAirway,QColor(0xb2,0x54,0xcd,0xff),tr("CA")); evseg->AddSlice(CPAP_RERA,QColor(0xff,0xff,0x80,0xff),tr("RE")); evseg->AddSlice(CPAP_FlowLimit,QColor(0x40,0x40,0x40,0xff),tr("FL")); GAHI->AddLayer(AddCPAP(evseg)); GAHI->setMargins(0,0,0,0); //SF->AddLayer(AddCPAP(evseg),LayerRight,100); gFlagsGroup *fg=new gFlagsGroup(); SF->AddLayer(AddCPAP(fg)); fg->AddLayer((new gFlagsLine(CPAP_CSR,QColor("light green"),tr("PB"),false,FT_Span))); fg->AddLayer((new gFlagsLine(CPAP_ClearAirway,QColor("purple"),tr("CA"),false))); fg->AddLayer((new gFlagsLine(CPAP_Obstructive,QColor("#40c0ff"),tr("OA"),true))); fg->AddLayer((new gFlagsLine(CPAP_Apnea,QColor("dark green"),tr("A")))); fg->AddLayer((new gFlagsLine(CPAP_Hypopnea,QColor("blue"),tr("H"),true))); fg->AddLayer((new gFlagsLine(CPAP_ExP,QColor("dark cyan"),tr("E"),false))); fg->AddLayer((new gFlagsLine(CPAP_LeakFlag,QColor("dark blue"),tr("L"),false))); fg->AddLayer((new gFlagsLine(CPAP_NRI,QColor("dark magenta"),tr("NRI"),false))); fg->AddLayer((new gFlagsLine(CPAP_FlowLimit,QColor("black"),tr("FL")))); fg->AddLayer((new gFlagsLine(CPAP_RERA,QColor("gold"),tr("RE")))); fg->AddLayer((new gFlagsLine(CPAP_VSnore,QColor("red"),tr("VS")))); fg->AddLayer((new gFlagsLine(CPAP_UserFlag1,QColor("yellow"),tr("UF1")))); fg->AddLayer((new gFlagsLine(CPAP_UserFlag2,QColor("green"),tr("UF2")))); //fg->AddLayer((new gFlagsLine(PRS1_0B,QColor("dark green"),tr("U0B")))); fg->AddLayer((new gFlagsLine(CPAP_VSnore2,QColor("red"),tr("VS2")))); SF->setBlockZoom(true); SF->AddLayer(new gShadowArea()); SF->AddLayer(new gYSpacer(),LayerLeft,gYAxis::Margin); //SF->AddLayer(new gFooBar(),LayerBottom,0,1); SF->AddLayer(new gXAxis(Qt::black,false),LayerBottom,0,20); //gXAxis::Margin); gLineChart *l; l=new gLineChart(CPAP_FlowRate,Qt::black,false,false); gLineOverlaySummary *los=new gLineOverlaySummary(tr("Selection AHI"),5,-4); AddCPAP(l); FRW->AddLayer(new gXGrid()); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_CSR,QColor("light green"),tr("CSR"),FT_Span))); FRW->AddLayer(l); FRW->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); FRW->AddLayer(new gXAxis(),LayerBottom,0,20); FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Hypopnea,QColor("blue"),tr("H"))))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_PressurePulse,QColor("red"),tr("PR"),FT_Dot))); //FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Pressure,QColor("white"),tr("P"),FT_Dot))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0B,QColor("blue"),"0B",FT_Dot))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_10,QColor("orange"),"10",FT_Dot))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0E,QColor("dark red"),"0E",FT_Dot))); if (PROFILE.general->calculateRDI()) FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_RERA,QColor("gold"),tr("RE"))))); else FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_RERA,QColor("gold"),tr("RE")))); FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Apnea,QColor("dark green"),tr("A"))))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_VSnore,QColor("red"),tr("VS")))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_FlowLimit,QColor("black"),tr("FL")))); FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Obstructive,QColor("#40c0ff"),tr("OA"))))); FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_ClearAirway,QColor("purple"),tr("CA"))))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_UserFlag1,QColor("yellow"),tr("U1"),FT_Bar))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_UserFlag2,QColor("orange"),tr("U2"),FT_Bar))); FRW->AddLayer(AddOXI(new gLineOverlayBar(OXI_SPO2Drop,QColor("red"),tr("O2")))); FRW->AddLayer(AddOXI(new gLineOverlayBar(OXI_PulseChange,QColor("blue"),tr("PC"),FT_Dot))); FRW->AddLayer(AddCPAP(los)); gGraph *graphs[]={ PRD, LEAK, AHI, SNORE, PTB, MP, RR, MV, TV, FLG, IE, TI, TE, TgMV, SPO2, PLETHY, PULSE }; int ng=sizeof(graphs)/sizeof(gGraph*); for (int i=0;iAddLayer(new gXGrid()); } /*PRD->AddLayer(AddCPAP(new gStatsLine(CPAP_Pressure,"Pressure")),LayerBottom,0,20,1); PRD->AddLayer(AddCPAP(new gStatsLine(CPAP_EPAP,"EPAP")),LayerBottom,0,20,1); PRD->AddLayer(AddCPAP(new gStatsLine(CPAP_IPAP,"IPAP")),LayerBottom,0,20,1); LEAK->AddLayer(AddCPAP(new gStatsLine(CPAP_Leak)),LayerBottom,0,20,1); SNORE->AddLayer(AddCPAP(new gStatsLine(CPAP_Snore)),LayerBottom,0,20,1); PTB->AddLayer(AddCPAP(new gStatsLine(CPAP_PatientTriggeredBreaths)),LayerBottom,0,20,1); RR->AddLayer(AddCPAP(new gStatsLine(CPAP_RespiratoryRate)),LayerBottom,0,20,1); MV->AddLayer(AddCPAP(new gStatsLine(CPAP_MinuteVentilation)),LayerBottom,0,20,1); TV->AddLayer(AddCPAP(new gStatsLine(CPAP_TidalVolume)),LayerBottom,0,20,1); FLG->AddLayer(AddCPAP(new gStatsLine(CPAP_FlowLimitGraph)),LayerBottom,0,20,1); IE->AddLayer(AddCPAP(new gStatsLine(CPAP_IE)),LayerBottom,0,20,1); TE->AddLayer(AddCPAP(new gStatsLine(CPAP_Te)),LayerBottom,0,20,1); TI->AddLayer(AddCPAP(new gStatsLine(CPAP_Ti)),LayerBottom,0,20,1); */ bool square=PROFILE.appearance->squareWavePlots(); gLineChart *pc=new gLineChart(CPAP_Pressure,QColor("dark green"),square); PRD->AddLayer(AddCPAP(pc)); pc->addPlot(CPAP_EPAP,Qt::blue,square); pc->addPlot(CPAP_IPAPLo,Qt::darkRed,square); pc->addPlot(CPAP_IPAP,Qt::red,square); pc->addPlot(CPAP_IPAPHi,Qt::darkRed,square); if (PROFILE.general->calculateRDI()) { AHI->AddLayer(AddCPAP(new AHIChart(QColor("#37a24b")))); } else { AHI->AddLayer(AddCPAP(new gLineChart(CPAP_AHI,QColor("light green"),square))); } gLineChart *lc=new gLineChart(CPAP_LeakTotal,Qt::darkYellow,square); lc->addPlot(CPAP_Leak,Qt::darkMagenta,square); lc->addPlot(CPAP_MaxLeak,Qt::darkRed,square); LEAK->AddLayer(AddCPAP(lc)); //LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_Leak,Qt::darkMagenta,square))); //LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_MaxLeak,Qt::darkRed,square))); SNORE->AddLayer(AddCPAP(new gLineChart(CPAP_Snore,Qt::darkGray,true))); 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))); MV->AddLayer(AddCPAP(new gLineChart(CPAP_MinuteVent,Qt::darkCyan,square))); TV->AddLayer(AddCPAP(new gLineChart(CPAP_TidalVolume,Qt::magenta,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))); 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))); gLineOverlaySummary *los1=new gLineOverlaySummary(tr("Events/hour"),5,-4); gLineOverlaySummary *los2=new gLineOverlaySummary(tr("Events/hour"),5,-4); PULSE->AddLayer(AddOXI(los1->add(new gLineOverlayBar(OXI_PulseChange,QColor("light gray"),tr("PD"),FT_Span)))); PULSE->AddLayer(AddOXI(los1)); SPO2->AddLayer(AddOXI(los2->add(new gLineOverlayBar(OXI_SPO2Drop,QColor("light blue"),tr("O2"),FT_Span)))); SPO2->AddLayer(AddOXI(los2)); PULSE->AddLayer(AddOXI(new gLineChart(OXI_Pulse,Qt::red,square))); SPO2->AddLayer(AddOXI(new gLineChart(OXI_SPO2,Qt::blue,true))); PLETHY->AddLayer(AddOXI(new gLineChart(OXI_Plethy,Qt::darkBlue,false))); PTB->setForceMaxY(100); SPO2->setForceMaxY(100); for (int i=0;iAddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); graphs[i]->AddLayer(new gXAxis(),LayerBottom,0,20); } layout->layout(); QTextCharFormat format = ui->calendar->weekdayTextFormat(Qt::Saturday); format.setForeground(QBrush(Qt::black, Qt::SolidPattern)); ui->calendar->setWeekdayTextFormat(Qt::Saturday, format); ui->calendar->setWeekdayTextFormat(Qt::Sunday, format); Qt::DayOfWeek dow=firstDayOfWeekFromLocale(); ui->calendar->setFirstDayOfWeek(dow); ui->tabWidget->setCurrentWidget(ui->details); ui->webView->settings()->setFontSize(QWebSettings::DefaultFontSize,QApplication::font().pointSize()); ui->webView->page()->setLinkDelegationPolicy(QWebPage::DelegateAllLinks); connect(ui->webView,SIGNAL(linkClicked(QUrl)),this,SLOT(Link_clicked(QUrl))); int ews=PROFILE.general->eventWindowSize(); ui->evViewSlider->setValue(ews); ui->evViewLCD->display(ews); GraphView->LoadSettings("Daily"); emptyToggleArea=new QLabel(this); emptyToggleArea->setText("This may take a while..."); ui->graphToggleArea->addWidget(emptyToggleArea,1,Qt::AlignCenter); emptyToggleArea->setVisible(false); for (int i=0;isize();i++) { QString title=(*GraphView)[i]->title(); QPushButton *btn=new QPushButton(title,this); btn->setCheckable(true); btn->setChecked((*GraphView)[i]->visible()); btn->setToolTip(tr("Show/Hide %1").arg(title)); btn->setVisible(false); GraphToggles[title]=btn; btn->setSizePolicy(QSizePolicy::Fixed,QSizePolicy::Minimum); ui->graphToggleArea->addWidget(btn); connect(btn,SIGNAL(toggled(bool)),this,SLOT(graphtogglebutton_toggled(bool))); } ui->graphToggleArea->addSpacerItem(new QSpacerItem(0,0,QSizePolicy::Expanding)); ui->graphVisibilityToggleArea->setVisible(false); ui->splitter->setVisible(false); // TODO: Add preference to hide do this for Widget Haters.. //ui->calNavWidget->hide(); if (PROFILE.general->unitSystem()==US_Archiac) { ui->weightSpinBox->setSuffix(STR_UNIT_POUND); ui->weightSpinBox->setDecimals(0); ui->ouncesSpinBox->setVisible(true); ui->ouncesSpinBox->setSuffix(STR_UNIT_OUNCE); } else { ui->ouncesSpinBox->setVisible(false); ui->weightSpinBox->setDecimals(3); ui->weightSpinBox->setSuffix(STR_UNIT_KG); } GraphView->setCubeImage(images["sheep"]); GraphView->setEmptyText(tr("No Data")); } Daily::~Daily() { GraphView->SaveSettings("Daily"); disconnect(ui->webView,SIGNAL(linkClicked(QUrl)),this,SLOT(Link_clicked(QUrl))); // Save any last minute changes.. if (previous_date.isValid()) Unload(previous_date); // delete splitter; delete ui; } void Daily::Link_clicked(const QUrl &url) { QString code=url.toString().section("=",0,0).toLower(); QString data=url.toString().section("=",1); int sid=data.toInt(); Day *day=NULL; if (code=="togglecpapsession") { // Enable/Disable CPAP session day=PROFILE.GetDay(previous_date,MT_CPAP); Session *sess=day->find(sid); if (!sess) return; int i=ui->webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-ui->webView->page()->mainFrame()->scrollBarValue(Qt::Vertical); sess->setEnabled(!sess->enabled()); // Messy, this rewrites both summary & events.. TODO: Write just the session summary file day->machine->Save(); // Reload day this->LoadDate(previous_date); ui->webView->page()->mainFrame()->setScrollBarValue(Qt::Vertical, ui->webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-i); return; } else if (code=="toggleoxisession") { // Enable/Disable Oximetry session day=PROFILE.GetDay(previous_date,MT_OXIMETER); Session *sess=day->find(sid); if (!sess) return; int i=ui->webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-ui->webView->page()->mainFrame()->scrollBarValue(Qt::Vertical); sess->setEnabled(!sess->enabled()); // Messy, this rewrites both summary & events.. TODO: Write just the session summary file day->machine->Save(); // Reload day this->LoadDate(previous_date); ui->webView->page()->mainFrame()->setScrollBarValue(Qt::Vertical, ui->webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-i); return; } else if (code=="cpap") { day=PROFILE.GetDay(previous_date,MT_CPAP); } else if (code=="oxi") { day=PROFILE.GetDay(previous_date,MT_OXIMETER); Session *sess=day->machine->sessionlist[sid]; if (mainwin->getOximetry()) { mainwin->getOximetry()->openSession(sess); mainwin->selectOximetryTab(); } return; } else if (code=="event") { QList list=ui->treeWidget->findItems(schema::channel[sid].description(),Qt::MatchContains); if (list.size()>0) { ui->treeWidget->collapseAll(); ui->treeWidget->expandItem(list.at(0)); QTreeWidgetItem *wi=list.at(0)->child(0); ui->treeWidget->setCurrentItem(wi); ui->tabWidget->setCurrentIndex(1); } else { mainwin->Notify(tr("No %1 events are recorded this day").arg(schema::channel[sid].description()),"",1500); } } else if (code=="graph") { qDebug() << "Select graph " << data; } else { qDebug() << "Clicked on" << code << data; } if (day) { Session *sess=day->machine->sessionlist[sid]; if (sess && sess->enabled()) { GraphView->SetXBounds(sess->first(),sess->last()); } } } void Daily::ReloadGraphs() { ui->splitter->setVisible(true); QDate d; if (previous_date.isValid()) { d=previous_date; Unload(d); } //else d=PROFILE.LastDay(); if (!d.isValid()) { d=ui->calendar->selectedDate(); } on_calendar_currentPageChanged(d.year(),d.month()); ui->calendar->setSelectedDate(d); //Load(d); } void Daily::on_calendar_currentPageChanged(int year, int month) { QDate d(year,month,1); int dom=d.daysInMonth(); for (int i=1;i<=dom;i++) { d=QDate(year,month,i); this->UpdateCalendarDay(d); } } void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day) { tree->clear(); if (!day) return; tree->setColumnCount(1); // 1 visible common.. (1 hidden) QTreeWidgetItem *root=NULL; QHash mcroot; QHash mccnt; int total_events=0; for (QVector::iterator s=day->begin();s!=day->end();s++) { if (!(*s)->enabled()) continue; QHash >::iterator m; for (m=(*s)->eventlist.begin();m!=(*s)->eventlist.end();m++) { ChannelID code=m.key(); if ((code!=CPAP_Obstructive) && (code!=CPAP_Hypopnea) && (code!=CPAP_Apnea) && (code!=PRS1_0B) && (code!=CPAP_ClearAirway) && (code!=CPAP_CSR) && (code!=CPAP_RERA) && (code!=CPAP_UserFlag1) && (code!=CPAP_UserFlag2) && (code!=CPAP_NRI) && (code!=CPAP_LeakFlag) && (code!=CPAP_ExP) && (code!=CPAP_FlowLimit) && (code!=CPAP_PressurePulse) && (code!=CPAP_VSnore2) && (code!=CPAP_VSnore)) continue; QTreeWidgetItem *mcr; if (mcroot.find(code)==mcroot.end()) { int cnt=day->count(code); total_events+=cnt; QString st=schema::channel[code].description(); if (st.isEmpty()) { st="Fixme %1"+code; } st+=" "; if (cnt==1) st+=tr("%1 event").arg(cnt); else st+=tr("%1 events").arg(cnt); QStringList l(st); l.append(""); mcroot[code]=mcr=new QTreeWidgetItem(root,l); mccnt[code]=0; } else { mcr=mcroot[code]; } for (int z=0;zcount();o++) { qint64 t=m.value()[z]->time(o); if (code==CPAP_CSR) { // center it in the middle of span t-=float(m.value()[z]->raw(o)/2.0)*1000.0; } QStringList a; QDateTime d=QDateTime::fromTime_t(t/1000); QString s=QString("#%1: %2 (%3)").arg((int)(++mccnt[code]),(int)3,(int)10,QChar('0')).arg(d.toString("HH:mm:ss")).arg(m.value()[z]->raw(o)); a.append(s); QTreeWidgetItem *item=new QTreeWidgetItem(a); item->setData(0,Qt::UserRole,t); //a.append(d.toString("yyyy-MM-dd HH:mm:ss")); mcr->addChild(item); } } } } int cnt=0; for (QHash::iterator m=mcroot.begin();m!=mcroot.end();m++) { tree->insertTopLevelItem(cnt++,m.value()); } //tree->insertTopLevelItem(cnt++,new QTreeWidgetItem(QStringList("[Total Events ("+QString::number(total_events)+")]"))); tree->sortByColumn(0,Qt::AscendingOrder); //tree->expandAll(); } void Daily::UpdateCalendarDay(QDate date) { QTextCharFormat nodata; QTextCharFormat cpaponly; QTextCharFormat cpapjour; QTextCharFormat oxiday; QTextCharFormat oxicpap; QTextCharFormat jourday; cpaponly.setForeground(QBrush(Qt::blue, Qt::SolidPattern)); cpaponly.setFontWeight(QFont::Normal); cpapjour.setForeground(QBrush(Qt::blue, Qt::SolidPattern)); cpapjour.setFontWeight(QFont::Bold); oxiday.setForeground(QBrush(Qt::red, Qt::SolidPattern)); oxiday.setFontWeight(QFont::Normal); oxicpap.setForeground(QBrush(Qt::red, Qt::SolidPattern)); oxicpap.setFontWeight(QFont::Bold); jourday.setForeground(QBrush(QColor("black"), Qt::SolidPattern)); jourday.setFontWeight(QFont::Bold); nodata.setForeground(QBrush(QColor("black"), Qt::SolidPattern)); nodata.setFontWeight(QFont::Normal); bool hascpap=PROFILE.GetDay(date,MT_CPAP)!=NULL; bool hasoxi=PROFILE.GetDay(date,MT_OXIMETER)!=NULL; bool hasjournal=PROFILE.GetDay(date,MT_JOURNAL)!=NULL; if (hascpap) { if (hasoxi) { ui->calendar->setDateTextFormat(date,oxicpap); } else if (hasjournal) { ui->calendar->setDateTextFormat(date,cpapjour); } else { ui->calendar->setDateTextFormat(date,cpaponly); } } else if (hasoxi) { ui->calendar->setDateTextFormat(date,oxiday); } else if (hasjournal) { ui->calendar->setDateTextFormat(date,jourday); } else { ui->calendar->setDateTextFormat(date,nodata); } ui->calendar->setHorizontalHeaderFormat(QCalendarWidget::ShortDayNames); } void Daily::LoadDate(QDate date) { ui->calendar->blockSignals(true); if (date.month()!=previous_date.month()) { on_calendar_currentPageChanged(date.year(),date.month()); } ui->calendar->setSelectedDate(date); ui->calendar->blockSignals(false); on_calendar_selectionChanged(); } void Daily::on_calendar_selectionChanged() { this->setCursor(Qt::BusyCursor); if (previous_date.isValid()) { // GraphView->fadeOut(); Unload(previous_date); } bool fadedir=previous_date < ui->calendar->selectedDate(); ZombieMeterMoved=false; Load(ui->calendar->selectedDate()); //GraphView->fadeIn(fadedir); GraphView->redraw(); ui->calButton->setText(ui->calendar->selectedDate().toString(Qt::TextDate)); ui->calendar->setFocus(Qt::ActiveWindowFocusReason); if (PROFILE.general->unitSystem()==US_Archiac) { ui->weightSpinBox->setSuffix(STR_UNIT_POUND); ui->weightSpinBox->setDecimals(0); ui->ouncesSpinBox->setVisible(true); ui->ouncesSpinBox->setSuffix(STR_UNIT_OUNCE); } else { ui->ouncesSpinBox->setVisible(false); ui->weightSpinBox->setDecimals(3); ui->weightSpinBox->setSuffix(STR_UNIT_KG); } this->setCursor(Qt::ArrowCursor); } void Daily::ResetGraphLayout() { GraphView->resetLayout(); } void Daily::graphtogglebutton_toggled(bool b) { Q_UNUSED(b) for (int i=0;isize();i++) { QString title=(*GraphView)[i]->title(); (*GraphView)[i]->setVisible(GraphToggles[title]->isChecked()); } GraphView->updateScale(); GraphView->redraw(); } void Daily::Load(QDate date) { static Day * lastcpapday=NULL; previous_date=date; Day *cpap=PROFILE.GetDay(date,MT_CPAP); Day *oxi=PROFILE.GetDay(date,MT_OXIMETER); // Day *sleepstage=profile->GetDay(date,MT_SLEEPSTAGE); if (!PROFILE.session->cacheSessions()) { if (lastcpapday && (lastcpapday!=cpap)) { for (QVector::iterator s=lastcpapday->begin();s!=lastcpapday->end();s++) { (*s)->TrashEvents(); } } } if ((cpap && oxi) && oxi->hasEnabledSessions()) { int gr; if (qAbs(cpap->first() - oxi->first())>30000) { mainwin->Notify(tr("Oximetry data exists for this day, however it's timestamps are too different, so the Graphs will not be linked."),"",3000); gr=1; } else gr=0; GraphView->findGraph(STR_TR_PulseRate)->setGroup(gr); GraphView->findGraph(STR_TR_SpO2)->setGroup(gr); GraphView->findGraph(STR_TR_Plethy)->setGroup(gr); } lastcpapday=cpap; QString html="" "" "" "" "" "\n"; QString tmp; //const int gwwidth=240; //const int gwheight=100; UpdateOXIGraphs(oxi); UpdateCPAPGraphs(cpap); UpdateEventsTree(ui->treeWidget,cpap); mainwin->refreshStatistics(); snapGV->setDay(cpap); GraphView->ResetBounds(false); //snapGV->ResetBounds(); //GraphView->ResetBounds(1); //GraphView->setEmptyText(tr("No Data")); //tr("No data for ")+date.toString(Qt::SystemLocaleLongDate)); if (!cpap && !oxi) { //splitter->setMinimumHeight(0); scrollbar->hide(); // GraphView->hide(); } else { //NoData->hide(); // GraphView->show(); scrollbar->show(); } //GraphView->redraw(); //snapGV->redraw(); int graphsAvailable=0; for (int i=0;isize();i++) { QString title=(*GraphView)[i]->title(); bool empty=(*GraphView)[i]->isEmpty(); if (!empty) graphsAvailable++; GraphToggles[title]->setVisible(!empty); } emptyToggleArea->setVisible(graphsAvailable==0); //ui->graphVisibilityToggleArea->setVisible(graphsAvailable>0); //RedrawGraphs(); QString epr,modestr; //float iap90,eap90; CPAPMode mode=MODE_UNKNOWN; QString a; bool isBrick=false; ui->graphVisibilityToggleArea->setVisible(true); if (graphsAvailable>0) { GraphView->setCubeImage(images["sheep"]); GraphView->setEmptyText(tr("Graphs Switched Off")); } else { GraphView->setCubeImage(images["nodata"]); GraphView->setEmptyText(tr("No Data")); emptyToggleArea->setText("No graph data available for this day"); } if (cpap) { if (GraphView->isEmpty()) { if (cpap->machine->GetClass()!=STR_MACH_ResMed) { GraphView->setCubeImage(images["brick"]); GraphView->setEmptyText(tr("No Graphs :(")); isBrick=true; } } mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode); modestr=schema::channel[CPAP_Mode].m_options[mode]; float ahi=(cpap->count(CPAP_Obstructive)+cpap->count(CPAP_Hypopnea)+cpap->count(CPAP_ClearAirway)+cpap->count(CPAP_Apnea)); if (PROFILE.general->calculateRDI()) ahi+=cpap->count(CPAP_RERA); ahi/=cpap->hours(); float csr=(100.0/cpap->hours())*(cpap->sum(CPAP_CSR)/3600.0); float uai=cpap->count(CPAP_Apnea)/cpap->hours(); float oai=cpap->count(CPAP_Obstructive)/cpap->hours(); float hi=(cpap->count(CPAP_ExP)+cpap->count(CPAP_Hypopnea))/cpap->hours(); float cai=cpap->count(CPAP_ClearAirway)/cpap->hours(); float rei=cpap->count(CPAP_RERA)/cpap->hours(); float fli=cpap->count(CPAP_FlowLimit)/cpap->hours(); float nri=cpap->count(CPAP_NRI)/cpap->hours(); float lki=cpap->count(CPAP_LeakFlag)/cpap->hours(); float exp=cpap->count(CPAP_ExP)/cpap->hours(); //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.session->showSerialNumbers()) { html+="\n"; } CPAPMode mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode); html+="\n"; 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) { if (cpap->machine->GetClass()==STR_MACH_ResMed) { cs="4 width='100%' align=center>"; } else cs="2 width='50%'>"; html+="\n") .arg("#F88017").arg("black").arg(tr("RDI")).arg(schema::channel[CPAP_RDI].description()).arg(ahi,0,'f',2); } else { html+=QString("\n") .arg("#F88017").arg("black").arg(tr("AHI")).arg(schema::channel[CPAP_AHI].description()).arg(ahi,0,'f',2); //html+="\n"; } html+=QString("\n") .arg("#4040ff").arg("white").arg(tr("Hypopnea")).arg(schema::channel[CPAP_Hypopnea].description()).arg(hi,0,'f',2).arg(CPAP_Hypopnea); if (cpap->machine->GetClass()==STR_MACH_ResMed) { html+=QString("\n") .arg("#208020").arg("black") .arg(tr("Apnea")).arg(schema::channel[CPAP_Apnea].description()).arg(uai,0,'f',2).arg(CPAP_Apnea); } html+=QString("\n") .arg("#40afbf").arg("black").arg(tr("Obstructive")).arg(schema::channel[CPAP_Obstructive].description()).arg(oai,0,'f',2).arg(CPAP_Obstructive); html+=QString("\n") .arg("#b254cd").arg("black").arg(tr("Clear Airway")).arg(schema::channel[CPAP_ClearAirway].description()).arg(cai,0,'f',2).arg(CPAP_ClearAirway); 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=cpap->settings_min(CPAP_PressureMin); html+=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+=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+=tr("Bi-Level")+QString("
EPAP: %1 IPAP: %2 %3
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("
EPAP: %1 IPAP: %2 - %3 %4
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+=tr("Unknown"); html+="
"+tr("Date")+""+tr("Sleep")+""+tr("Wake")+""+STR_UNIT_Hours+"
%1%2%3%4

"; if (PROFILE.general->calculateRDI()) { html+=QString("
%3%4%5
%3%4%5
"+tr("AHI")+""+schema::channel[CPAP_AHI].description()+""+QString().sprintf("%.2f",ahi)+"
%3%4%5
%3%4%5
%3%4%5
%3%4%5
"; if (cpap->machine->GetClass()==STR_MACH_PRS1) { html+=""; html+=QString("\n") .arg("#ffff80").arg("black").arg(tr("RERA")).arg(schema::channel[CPAP_RERA].description()).arg(rei,0,'f',2).arg(CPAP_RERA); if (mode>MODE_CPAP) { html+=QString("\n") .arg("#404040").arg("white").arg(tr("Flow Limit")).arg(schema::channel[CPAP_FlowLimit].description()).arg(fli,0,'f',2).arg(CPAP_FlowLimit); html+=QString("\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+=QString("\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("\n") .arg("#80ff80").arg("black").arg(tr("PB/CSR")).arg(schema::channel[CPAP_CSR].description()).arg(csr,0,'f',2).arg(CPAP_CSR); html+="
%3%4%5
%3%4%5
%3%4%5
 
%3%4%5
%3%4%5%
"; } else if (cpap->machine->GetClass()==STR_MACH_Intellipap) { html+=""; html+=QString("\n") .arg("#ffff80").arg("black").arg(tr("NRI")).arg(schema::channel[CPAP_NRI].description()).arg(nri,0,'f',2).arg(CPAP_NRI); html+=QString("\n") .arg("#404040").arg("black").arg(tr("Leak")).arg(schema::channel[CPAP_LeakFlag].description()).arg(lki,0,'f',2).arg(CPAP_LeakFlag); html+=QString("\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); html+=QString("\n") .arg("#80ff80").arg("black").arg(tr("Exh Puff")).arg(schema::channel[CPAP_ExP].description()).arg(exp,0,'f',2).arg(CPAP_ExP); html+="
%3%4%5%
%3%4%5%
%3%4%5
%3%4%5%
"; } html+=""; // Note, this may not be a problem since Qt bug 13622 was discovered // as it only relates to text drawing, which the Pie chart does not do // ^^ Scratch that.. pie now includes text.. if (PROFILE.appearance->graphSnapshots()) { // AHI Pie Chart if (oai+hi+cai+uai+rei+fli>0) { html+=" "; html+=QString("%1").arg(tr("Event Breakdown")); html+="
"; //G_AHI->setFixedSize(gwwidth,120); //mainwin->snapshotGraph()->setPrintScaleX(1); //mainwin->snapshotGraph()->setPrintScaleY(1); GAHI->setShowTitle(false); //snapGV->setFixedSize(150,150); QPixmap pixmap=GAHI->renderPixmap(150,150,false); QByteArray byteArray; QBuffer buffer(&byteArray); // use buffer to store pixmap into byteArray buffer.open(QIODevice::WriteOnly); pixmap.save(&buffer, "PNG"); html += "\n"; } else { html += "\n"; } } } else { // machine is a brick html+="

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

"; html+=""+tr("Sorry, your machine does not record data.")+"\n"; html+=""+tr("Complain to your Equipment Provider!")+"\n"; html+=" \n"; } html+=""; } // if (!CPAP) html+="\n"; float percentile=0.95; if ((cpap && !isBrick) || oxi) { html+="\n"; html+=QString("\n").arg(tr("Statistics")); html+="\n"; html+=QString("") .arg(tr("Channel")) .arg(tr("Min")) .arg(tr("Avg")) .arg(tr("%1%").arg(percentile*100,0,'f',0)) .arg(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; 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()+")"; html+=QString("") .arg(QString("%3%2") //"+tr("RDI")+""+ .arg(QString::number(code)).arg(tooltip).arg(schema::channel[code].label())) .arg(cpap->Min(code),0,'f',2) .arg(cpap->wavg(code),0,'f',2) .arg(cpap->percentile(code,percentile),0,'f',2) .arg(cpap->Max(code),0,'f',2); } if (oxi && oxi->channelHasData(code)) { QString tooltip=schema::channel[code].description(); if (!schema::channel[code].units().isEmpty()) tooltip+=" ("+schema::channel[code].units()+")"; html+=QString("") .arg(QString("%2%3") .arg(QString::number(code)).arg(schema::channel[code].label()).arg(tooltip)) .arg(oxi->Min(code),0,'f',2) .arg(oxi->wavg(code),0,'f',2) .arg(oxi->percentile(code,percentile),0,'f',2) .arg(oxi->Max(code),0,'f',2); } } } else { html+=""; html+="\n"; } 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); } if (cpap && cpap->hasEnabledSessions()) { html+=""; // html+="
 
%1

%1%2%3%4%5
%1%2%3%4%5
%1%2%3%4%5
"+tr("No data available")+"
 
 
%1

"+oxi->machine->properties[STR_PROP_Brand]+" "+oxi->machine->properties[STR_PROP_Model]+"
 
%1: %2 (%3)\%
%1: %2 (%3)\%
%1: %2\%
 
"; html+=QString("").arg(tr("Machine Settings")); html+=""; if (cpap->machine->GetClass()==STR_MACH_PRS1) { int i=cpap->settings_max(PRS1_FlexMode); int j=cpap->settings_max(PRS1_FlexSet); QString flexstr=(i>1) ? schema::channel[PRS1_FlexMode].option(i)+" "+schema::channel[PRS1_FlexSet].option(j) : "None"; html+=QString("").arg(tr("Flex")) .arg(flexstr); int humid=round(cpap->settings_wavg(PRS1_HumidSetting)); html+=QString("").arg(tr("Humidifier")) .arg(humid==0 ? STR_GEN_Off : "x"+QString::number(humid)); } else if (cpap->machine->GetClass()==STR_MACH_ResMed) { int epr=cpap->settings_max(RMS9_EPR); int epr2=cpap->settings_max(RMS9_EPRSet); html+=QString("") .arg(tr("EPR")).arg(epr).arg(epr2); } } html+="
%1

%1%2
%1%2
%1%2 / %3
"; if (cpap || oxi) { html+=""; html+=""; html+=QString("").arg(tr("Session Information")); html+=""; QDateTime fd,ld; bool corrupted_waveform=false; QString tooltip; html+=QString("") .arg(tr("SessionID")) .arg(tr("On")) .arg(tr("Date")) .arg(tr("Start")) .arg(tr("End")); if (cpap) { html+=QString("").arg(tr("CPAP Sessions")); 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; QHash::iterator i=(*s)->settings.find(CPAP_BrokenWaveform); tooltip=cpap->machine->GetClass()+" "+tr("CPAP")+" "+QString().sprintf("%2ih, %2im, %2is",h,m,s1); // tooltip needs to lookup language.. :-/ if ((i!=(*s)->settings.end()) && i.value().toBool()) corrupted_waveform=true; 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((*s)->session(),8,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("").arg(tr("Oximetry Sessions")); 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; QHash::iterator i=(*s)->settings.find(CPAP_BrokenWaveform); tooltip=oxi->machine->GetClass()+" "+tr("Oximeter")+" "+QString().sprintf("%2ih, %2im, %2is",h,m,s1); Session *sess=*s; if (!sess->settings.contains(SESSION_ENABLED)) { sess->settings[SESSION_ENABLED]=true; } bool b=sess->settings[SESSION_ENABLED].toBool(); if ((i!=(*s)->settings.end()) && i.value().toBool()) corrupted_waveform=true; html+=QString("") .arg((*s)->session()) .arg(tooltip) .arg((*s)->session(),8,10,QChar('0')) .arg((b ? "on" : "off")) .arg(fd.date().toString(Qt::SystemLocaleShortDate)) .arg(fd.toString("HH:mm")) .arg(ld.toString("HH:mm")); //tmp.sprintf(("").toLatin1(),(*s)->session(),(*s)->session()); //html+=tmp; } } 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+="
 
%1

%1%2%3%4%5
%1
%3%2%5%6%7
%1
%3%2%5%6%7
%08i"+fd.date().toString(Qt::SystemLocaleShortDate)+""+fd.toString("HH:mm ")+""+ld.toString("HH:mm")+"
%1

"; } html+=""; ui->webView->setHtml(html); ui->JournalNotes->clear(); ui->bookmarkTable->clearContents(); ui->bookmarkTable->setRowCount(0); QStringList sl; //sl.append(tr("Starts")); //sl.append(tr("Notes")); ui->bookmarkTable->setHorizontalHeaderLabels(sl); ui->ZombieMeter->blockSignals(true); ui->weightSpinBox->blockSignals(true); ui->ouncesSpinBox->blockSignals(true); ui->weightSpinBox->setValue(0); ui->ouncesSpinBox->setValue(0); ui->ZombieMeter->setValue(5); ui->ouncesSpinBox->blockSignals(false); ui->weightSpinBox->blockSignals(false); ui->ZombieMeter->blockSignals(false); ui->BMI->display(0); ui->BMI->setVisible(false); ui->BMIlabel->setVisible(false); BookmarksChanged=false; Session *journal=GetJournalSession(date); if (journal) { bool ok; if (journal->settings.contains(Journal_Notes)) ui->JournalNotes->setHtml(journal->settings[Journal_Notes].toString()); if (journal->settings.contains(Journal_Weight)) { double kg=journal->settings[Journal_Weight].toDouble(&ok); if (PROFILE.general->unitSystem()==US_Metric) { ui->weightSpinBox->setDecimals(3); ui->weightSpinBox->blockSignals(true); ui->weightSpinBox->setValue(kg); ui->weightSpinBox->blockSignals(false); ui->ouncesSpinBox->setVisible(false); ui->weightSpinBox->setSuffix(STR_UNIT_KG); } else { float ounces=(kg*1000.0)/ounce_convert; int pounds=ounces/16.0; double oz; double frac=modf(ounces,&oz); ounces=(int(ounces) % 16)+frac; ui->weightSpinBox->blockSignals(true); ui->ouncesSpinBox->blockSignals(true); ui->weightSpinBox->setValue(pounds); ui->ouncesSpinBox->setValue(ounces); ui->ouncesSpinBox->blockSignals(false); ui->weightSpinBox->blockSignals(false); ui->weightSpinBox->setSuffix(STR_UNIT_POUND); ui->weightSpinBox->setDecimals(0); ui->ouncesSpinBox->setVisible(true); ui->ouncesSpinBox->setSuffix(STR_UNIT_OUNCE); } double height=PROFILE.user->height()/100.0; if (height>0 && kg>0) { double bmi=kg/(height*height); ui->BMI->setVisible(true); ui->BMIlabel->setVisible(true); ui->BMI->display(bmi); } } if (journal->settings.contains(Journal_ZombieMeter)) { ui->ZombieMeter->blockSignals(true); ui->ZombieMeter->setValue(journal->settings[Journal_ZombieMeter].toDouble(&ok)); ui->ZombieMeter->blockSignals(false); } if (journal->settings.contains(Bookmark_Start)) { QVariantList start=journal->settings[Bookmark_Start].toList(); QVariantList end=journal->settings[Bookmark_End].toList(); QStringList notes=journal->settings[Bookmark_Notes].toStringList(); ui->bookmarkTable->blockSignals(true); 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); } // for (int i ui->bookmarkTable->blockSignals(false); } // if (journal->settings.contains(Bookmark_Start)) } // if (journal) } void Daily::UnitsChanged() { double kg; if (PROFILE.general->unitSystem()==US_Metric) { kg=ui->weightSpinBox->value(); float ounces=(kg*1000.0)/ounce_convert; int pounds=ounces/16; float oz=fmodf(ounces,16); ui->weightSpinBox->setValue(pounds); ui->ouncesSpinBox->setValue(oz); ui->weightSpinBox->setDecimals(0); ui->weightSpinBox->setSuffix(STR_UNIT_POUND); ui->ouncesSpinBox->setVisible(true); ui->ouncesSpinBox->setSuffix(STR_UNIT_OUNCE); } else { kg=(ui->weightSpinBox->value()*(ounce_convert*16.0))+(ui->ouncesSpinBox->value()*ounce_convert); kg/=1000.0; ui->weightSpinBox->setDecimals(3); ui->weightSpinBox->setValue(kg); ui->ouncesSpinBox->setVisible(false); ui->weightSpinBox->setSuffix(STR_UNIT_KG); } } void Daily::Unload(QDate date) { Session *journal=GetJournalSession(date); bool nonotes=ui->JournalNotes->toPlainText().isEmpty(); if (journal) { QString jhtml=ui->JournalNotes->toHtml(); if ((!journal->settings.contains(Journal_Notes) && !nonotes) || (journal->settings[Journal_Notes]!=jhtml)) { journal->settings[Journal_Notes]=jhtml; journal->SetChanged(true); } } else { if (!nonotes) { journal=CreateJournalSession(date); if (!nonotes) { journal->settings[Journal_Notes]=ui->JournalNotes->toHtml(); journal->SetChanged(true); } } } if (journal) { if (nonotes) { QHash::iterator it=journal->settings.find(Journal_Notes); if (it!=journal->settings.end()) { journal->settings.erase(it); } } if (journal->IsChanged()) { mainwin->getOverview()->ResetGraphs(); } Machine *jm=PROFILE.GetMachine(MT_JOURNAL); if (jm) jm->SaveSession(journal); } UpdateCalendarDay(date); } void Daily::on_JournalNotesItalic_clicked() { QTextCursor cursor = ui->JournalNotes->textCursor(); if (!cursor.hasSelection()) cursor.select(QTextCursor::WordUnderCursor); QTextCharFormat format=cursor.charFormat(); format.setFontItalic(!format.fontItalic()); cursor.mergeCharFormat(format); //ui->JournalNotes->mergeCurrentCharFormat(format); } void Daily::on_JournalNotesBold_clicked() { QTextCursor cursor = ui->JournalNotes->textCursor(); if (!cursor.hasSelection()) cursor.select(QTextCursor::WordUnderCursor); QTextCharFormat format=cursor.charFormat(); int fw=format.fontWeight(); if (fw!=QFont::Bold) format.setFontWeight(QFont::Bold); else format.setFontWeight(QFont::Normal); cursor.mergeCharFormat(format); //ui->JournalNotes->mergeCurrentCharFormat(format); } void Daily::on_JournalNotesFontsize_activated(int index) { QTextCursor cursor = ui->JournalNotes->textCursor(); if (!cursor.hasSelection()) cursor.select(QTextCursor::WordUnderCursor); QTextCharFormat format=cursor.charFormat(); QFont font=format.font(); int fontsize=10; if (index==1) fontsize=15; else if (index==2) fontsize=25; font.setPointSize(fontsize); format.setFont(font); cursor.mergeCharFormat(format); } void Daily::on_JournalNotesColour_clicked() { QColor col=QColorDialog::getColor(Qt::black,this,tr("Pick a Colour")); //,QColorDialog::NoButtons); if (!col.isValid()) return; QTextCursor cursor = ui->JournalNotes->textCursor(); if (!cursor.hasSelection()) cursor.select(QTextCursor::WordUnderCursor); QBrush b(col); QPalette newPalette = palette(); newPalette.setColor(QPalette::ButtonText, col); ui->JournalNotesColour->setPalette(newPalette); QTextCharFormat format=cursor.charFormat(); format.setForeground(b); cursor.setCharFormat(format); } Session * Daily::CreateJournalSession(QDate date) { Machine *m=PROFILE.GetMachine(MT_JOURNAL); if (!m) { m=new Machine(p_profile,0); m->SetClass("Journal"); m->properties[STR_PROP_Brand]="Virtual"; m->SetType(MT_JOURNAL); PROFILE.AddMachine(m); } Session *sess=new Session(m,0); qint64 st,et; Day *cday=PROFILE.GetDay(date,MT_CPAP); if (cday) { st=cday->first(); et=cday->last(); } else { QDateTime dt(date,QTime(20,0)); st=qint64(dt.toTime_t())*1000L; et=st+3600000; } sess->set_first(st); sess->set_last(et); sess->SetChanged(true); m->AddSession(sess,p_profile); return sess; } Session * Daily::GetJournalSession(QDate date) // Get the first journal session { Day *journal=PROFILE.GetDay(date,MT_JOURNAL); if (!journal) return NULL; //CreateJournalSession(date); QVector::iterator s; s=journal->begin(); if (s!=journal->end()) return *s; return NULL; } void Daily::UpdateCPAPGraphs(Day *day) { //if (!day) return; if (day) { day->OpenEvents(); } for (QList::iterator g=CPAPData.begin();g!=CPAPData.end();g++) { (*g)->SetDay(day); } } void Daily::UpdateOXIGraphs(Day *day) { //if (!day) return; if (day) { day->OpenEvents(); } for (QList::iterator g=OXIData.begin();g!=OXIData.end();g++) { (*g)->SetDay(day); } } void Daily::RedrawGraphs() { GraphView->redraw(); } void Daily::on_treeWidget_itemClicked(QTreeWidgetItem *item, int column) { Q_UNUSED(column); QDateTime d; if (!item->data(0,Qt::UserRole).isNull()) { qint64 winsize=qint64(PROFILE.general->eventWindowSize())*60000L; qint64 t=item->data(0,Qt::UserRole).toLongLong(); double st=t-(winsize/2); double et=t+(winsize/2); gGraph *g=GraphView->findGraph(STR_TR_EventFlags); if (!g) return; if (strmin_x) { st=g->rmin_x; et=st+winsize; } if (et>g->rmax_x) { et=g->rmax_x; st=et-winsize; } GraphView->SetXBounds(st,et); } } void Daily::on_treeWidget_itemSelectionChanged() { if (ui->treeWidget->selectedItems().size()==0) return; QTreeWidgetItem *item=ui->treeWidget->selectedItems().at(0); if (!item) return; on_treeWidget_itemClicked(item, 0); } void Daily::on_JournalNotesUnderline_clicked() { QTextCursor cursor = ui->JournalNotes->textCursor(); if (!cursor.hasSelection()) cursor.select(QTextCursor::WordUnderCursor); QTextCharFormat format=cursor.charFormat(); format.setFontUnderline(!format.fontUnderline()); cursor.mergeCharFormat(format); //ui->JournalNotes->mergeCurrentCharFormat(format); } void Daily::on_prevDayButton_clicked() { if (!PROFILE.ExistsAndTrue("SkipEmptyDays")) { LoadDate(previous_date.addDays(-1)); } else { QDate d=previous_date; for (int i=0;i<90;i++) { d=d.addDays(-1); if (PROFILE.GetDay(d)) { LoadDate(d); break; } } } } void Daily::on_nextDayButton_clicked() { if (!PROFILE.ExistsAndTrue("SkipEmptyDays")) { LoadDate(previous_date.addDays(1)); } else { QDate d=previous_date; for (int i=0;i<90;i++) { d=d.addDays(1); if (PROFILE.GetDay(d)) { LoadDate(d); break; } } } } void Daily::on_calButton_toggled(bool checked) { //bool b=!ui->calendar->isVisible(); bool b=checked; ui->calendar->setVisible(b); if (!b) ui->calButton->setArrowType(Qt::DownArrow); else ui->calButton->setArrowType(Qt::UpArrow); } void Daily::on_todayButton_clicked() { QDate d=QDate::currentDate(); if (d > PROFILE.LastDay()) d=PROFILE.LastDay(); LoadDate(d); } void Daily::on_evViewSlider_valueChanged(int value) { ui->evViewLCD->display(value); PROFILE.general->setEventWindowSize(value); int winsize=value*60; gGraph *g=GraphView->findGraph(STR_TR_EventFlags); if (!g) return; qint64 st=g->min_x; qint64 et=g->max_x; qint64 len=et-st; qint64 d=st+len/2.0; st=d-(winsize/2)*1000; et=d+(winsize/2)*1000; if (strmin_x) { st=g->rmin_x; et=st+winsize*1000; } if (et>g->rmax_x) { et=g->rmax_x; st=et-winsize*1000; } GraphView->SetXBounds(st,et); } 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); qint64 st2,et2,st3,et3; Day * day=PROFILE.GetGoodDay(previous_date,MT_CPAP); if (day) { st2=day->first(); et2=day->last(); } Day * oxi=PROFILE.GetGoodDay(previous_date,MT_OXIMETER); if (oxi) { st3=oxi->first(); et3=oxi->last(); } if (oxi && day) { st2=qMin(st2,st3); et2=qMax(et2,et3); } else if (oxi) { st2=st3; et2=et3; } else if (!day) return; if ((etet2)) { mainwin->Notify("This bookmarked is in a currently disabled area.."); return; } if (stet2) et=et2; GraphView->SetXBounds(st,et); GraphView->redraw(); } void Daily::on_addBookmarkButton_clicked() { qint64 st,et; ui->bookmarkTable->blockSignals(true); 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->blockSignals(false); update_Bookmarks(); mainwin->updateFavourites(); //ui->bookmarkTable->setItem(row,2,new QTableWidgetItem(QString::number(st))); //ui->bookmarkTable->setItem(row,3,new QTableWidgetItem(QString::number(et))); } void Daily::update_Bookmarks() { 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()); } Session *journal=GetJournalSession(previous_date); if (!journal) { journal=CreateJournalSession(previous_date); } journal->settings[Bookmark_Start]=start; journal->settings[Bookmark_End]=end; journal->settings[Bookmark_Notes]=notes; journal->SetChanged(true); BookmarksChanged=true; mainwin->updateFavourites(); } void Daily::on_removeBookmarkButton_clicked() { int row=ui->bookmarkTable->currentRow(); if (row>=0) { ui->bookmarkTable->blockSignals(true); ui->bookmarkTable->removeRow(row); ui->bookmarkTable->blockSignals(false); update_Bookmarks(); } mainwin->updateFavourites(); } void Daily::on_ZombieMeter_valueChanged(int action) { Q_UNUSED(action); ZombieMeterMoved=true; Session *journal=GetJournalSession(previous_date); if (!journal) { journal=CreateJournalSession(previous_date); } journal->settings[Journal_ZombieMeter]=ui->ZombieMeter->value(); journal->SetChanged(true); gGraph *g; if (mainwin->getOverview()) { g=mainwin->getOverview()->graphView()->findGraph("Zombie"); if (g) g->setDay(NULL); //mainwin->getOverview()->RedrawGraphs(); } } void Daily::on_bookmarkTable_itemChanged(QTableWidgetItem *item) { Q_UNUSED(item); update_Bookmarks(); } void Daily::on_weightSpinBox_valueChanged(double arg1) { double height=PROFILE.user->height()/100.0; Session *journal=GetJournalSession(previous_date); if (!journal) { journal=CreateJournalSession(previous_date); } double kg; if (PROFILE.general->unitSystem()==US_Archiac) { kg=(arg1*pound_convert) + (ui->ouncesSpinBox->value()*ounce_convert); } else { kg=arg1; } journal->settings[Journal_Weight]=kg; gGraphView *gv=mainwin->getOverview()->graphView(); gGraph *g; if (gv) { g=gv->findGraph(STR_TR_Weight); if (g) g->setDay(NULL); } if ((height>0) && (kg>0)) { double bmi=kg/(height * height); ui->BMI->display(bmi); ui->BMI->setVisible(true); journal->settings[Journal_BMI]=bmi; if (gv) { g=gv->findGraph(STR_TR_BMI); if (g) g->setDay(NULL); } } journal->SetChanged(true); } void Daily::on_ouncesSpinBox_valueChanged(int arg1) { Session *journal=GetJournalSession(previous_date); if (!journal) { journal=CreateJournalSession(previous_date); } double height=PROFILE.user->height()/100.0; double kg=(ui->weightSpinBox->value()*pound_convert) + (arg1*ounce_convert); journal->settings[Journal_Weight]=kg; gGraph *g; if (mainwin->getOverview()) { g=mainwin->getOverview()->graphView()->findGraph(STR_TR_Weight); if (g) g->setDay(NULL); } if ((height>0) && (kg>0)) { double bmi=kg/(height * height); ui->BMI->display(bmi); ui->BMI->setVisible(true); journal->settings[Journal_BMI]=bmi; if (mainwin->getOverview()) { g=mainwin->getOverview()->graphView()->findGraph(STR_TR_BMI); if (g) g->setDay(NULL); } } journal->SetChanged(true); } QString Daily::GetDetailsText() { ui->webView->triggerPageAction(QWebPage::SelectAll); QString text=ui->webView->page()->selectedText(); ui->webView->triggerPageAction(QWebPage::MoveToEndOfDocument); ui->webView->triggerPageAction(QWebPage::SelectEndOfDocument); return text; }