/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- * vim: set ts=8 sts=4 et sw=4 tw=99: * * Daily Panel * * Copyright (c) 2011-2014 Mark Watkins * * This file is subject to the terms and conditions of the GNU General Public * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ #include #include #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; // This was Sean Stangl's idea.. but I couldn't apply that patch. inline QString channelInfo(ChannelID code) { return schema::channel[code].fullname()+"\n"+schema::channel[code].description()+"\n("+schema::channel[code].units()+")"; } void Daily::setCalendarVisible(bool visible) { on_calButton_toggled(visible); } void Daily::setSidebarVisible(bool visible) { QList a; int panel_width = visible ? 350 : 0; a.push_back(panel_width); a.push_back(this->width() - panel_width); ui->splitter_2->setStretchFactor(1,1); ui->splitter_2->setSizes(a); ui->splitter_2->setStretchFactor(1,1); } 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; lastcpapday=nullptr; setSidebarVisible(true); layout=new QHBoxLayout(); layout->setSpacing(0); layout->setMargin(0); layout->setContentsMargins(0,0,0,0); dateDisplay=new QLabel("",this); dateDisplay->setAlignment(Qt::AlignCenter); dateDisplay->setTextFormat(Qt::RichText); ui->sessionBarLayout->addWidget(dateDisplay,1); // const bool sessbar_under_graphs=false; // if (sessbar_under_graphs) { // ui->sessionBarLayout->addWidget(sessbar,1); // } else { // ui->splitter->insertWidget(2,sessbar); // sessbar->setMaximumHeight(sessbar->height()); // sessbar->setMinimumHeight(sessbar->height()); // } ui->calNavWidget->setMaximumHeight(ui->calNavWidget->height()); ui->calNavWidget->setMinimumHeight(ui->calNavWidget->height()); sessbar=nullptr; webView=new MyWebView(this); QWebSettings::globalSettings()->setAttribute(QWebSettings::PluginsEnabled, true); ui->tabWidget->insertTab(0,webView,QIcon(),"Details"); ui->graphFrame->setLayout(layout); //ui->graphMainArea->setLayout(layout); ui->graphMainArea->setAutoFillBackground(false); GraphView=new gGraphView(ui->graphFrame,shared); GraphView->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding); snapGV=new gGraphView(GraphView); //ui->graphMainArea); snapGV->setMinimumSize(172,172); snapGV->hideSplitter(); snapGV->hide(); scrollbar=new MyScrollBar(ui->graphFrame); 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=p_profile->appearance->graphHeight(); gGraph *GAHI = nullptr, // *TAP = nullptr, *SF = nullptr, *AHI = nullptr; graphlist[STR_GRAPH_SleepFlags] = SF = new gGraph(STR_GRAPH_SleepFlags, GraphView, STR_TR_EventFlags, STR_TR_EventFlags, default_height); SF->setPinned(true); ChannelID cpapcodes[] = { CPAP_FlowRate, CPAP_MaskPressure, CPAP_Pressure, CPAP_Leak, CPAP_Snore, CPAP_RespRate, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_FLG, CPAP_PTB, CPAP_RespEvent, CPAP_Ti, CPAP_Te, CPAP_IE, ZEO_SleepStage, POS_Inclination, POS_Orientation, CPAP_Test1 }; int cpapsize = sizeof(cpapcodes) / sizeof(ChannelID); ChannelID oxicodes[] = { OXI_Pulse, OXI_SPO2, OXI_Plethy }; int oxisize = sizeof(oxicodes) / sizeof(ChannelID); for (int i=0; i < cpapsize; ++i) { ChannelID code = cpapcodes[i]; graphlist[schema::channel[code].code()] = new gGraph(schema::channel[code].code(), GraphView, schema::channel[code].label(), channelInfo(code), default_height); } int oxigrp=p_profile->ExistsAndTrue("SyncOximetry") ? 0 : 1; // Contemplating killing this setting... for (int i=0; i < oxisize; ++i) { ChannelID code = oxicodes[i]; graphlist[schema::channel[code].code()] = new gGraph(schema::channel[code].code(), GraphView, schema::channel[code].label(), channelInfo(code), default_height, oxigrp); } if (p_profile->general->calculateRDI()) { AHI=new gGraph("AHI", GraphView,STR_TR_RDI, channelInfo(CPAP_RDI), default_height); } else { AHI=new gGraph("AHI", GraphView,STR_TR_AHI, channelInfo(CPAP_AHI), default_height); } graphlist["AHI"] = AHI; graphlist["INTPULSE"] = new gGraph("INTPULSE", GraphView,tr("Int. Pulse"), channelInfo(OXI_Pulse), default_height, oxigrp); graphlist["INTSPO2"] = new gGraph("INTSPO2", GraphView,tr("Int. SpO2"), channelInfo(OXI_SPO2), 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); graphlist[STR_GRAPH_EventBreakdown] = GAHI = new gGraph(STR_GRAPH_EventBreakdown, snapGV,tr("Breakdown"),tr("events"),172); gSegmentChart * evseg=new gSegmentChart(GST_Pie); evseg->AddSlice(CPAP_Hypopnea,QColor(0x40,0x40,0xff,0xff),STR_TR_H); evseg->AddSlice(CPAP_Apnea,QColor(0x20,0x80,0x20,0xff),STR_TR_UA); evseg->AddSlice(CPAP_Obstructive,QColor(0x40,0xaf,0xbf,0xff),STR_TR_OA); evseg->AddSlice(CPAP_ClearAirway,QColor(0xb2,0x54,0xcd,0xff),STR_TR_CA); evseg->AddSlice(CPAP_RERA,QColor(0xff,0xff,0x80,0xff),STR_TR_RE); evseg->AddSlice(CPAP_NRI,QColor(0x00,0x80,0x40,0xff),STR_TR_NR); evseg->AddSlice(CPAP_FlowLimit,QColor(0x40,0x40,0x40,0xff),STR_TR_FL); evseg->AddSlice(CPAP_SensAwake,QColor(0x40,0xC0,0x40,0xff),STR_TR_SA); if (p_profile->cpap->userEventPieChart()) { evseg->AddSlice(CPAP_UserFlag1,QColor(0xe0,0xe0,0xe0,0xff),tr("UF1")); evseg->AddSlice(CPAP_UserFlag2,QColor(0xc0,0xc0,0xe0,0xff),tr("UF2")); } GAHI->AddLayer(AddCPAP(evseg)); GAHI->setMargins(0,0,0,0); gFlagsGroup *fg=new gFlagsGroup(); SF->AddLayer(AddCPAP(fg)); // Spans // fg->AddLayer((new gFlagsLine(CPAP_CSR, COLOR_CSR, STR_TR_PB, false, FT_Span))); // fg->AddLayer((new gFlagsLine(CPAP_LargeLeak, COLOR_LargeLeak, STR_TR_LL, false, FT_Span))); // fg->AddLayer((new gFlagsLine(CPAP_Ramp, COLOR_Ramp, schema::channel[CPAP_Ramp].label(), false, FT_Span))); // // Flags // fg->AddLayer((new gFlagsLine(CPAP_ClearAirway, COLOR_ClearAirway, STR_TR_CA,false))); // fg->AddLayer((new gFlagsLine(CPAP_Obstructive, COLOR_Obstructive, STR_TR_OA,true))); // fg->AddLayer((new gFlagsLine(CPAP_Apnea, COLOR_Apnea, STR_TR_UA))); // fg->AddLayer((new gFlagsLine(CPAP_Hypopnea, COLOR_Hypopnea, STR_TR_H,true))); // fg->AddLayer((new gFlagsLine(CPAP_ExP, COLOR_ExP, STR_TR_EP,false))); // fg->AddLayer((new gFlagsLine(CPAP_LeakFlag, COLOR_LeakFlag, STR_TR_LE,false))); // fg->AddLayer((new gFlagsLine(CPAP_NRI, COLOR_NRI, STR_TR_NRI,false))); // fg->AddLayer((new gFlagsLine(CPAP_FlowLimit, COLOR_FlowLimit, STR_TR_FL))); // fg->AddLayer((new gFlagsLine(CPAP_SensAwake, COLOR_SensAwake, STR_TR_SA))); // fg->AddLayer((new gFlagsLine(CPAP_RERA, COLOR_RERA, STR_TR_RE))); // fg->AddLayer((new gFlagsLine(CPAP_VSnore, COLOR_VibratorySnore, STR_TR_VS))); // fg->AddLayer((new gFlagsLine(CPAP_VSnore2, COLOR_VibratorySnore, STR_TR_VS2))); // if (p_profile->cpap->userEventFlagging()) { // fg->AddLayer((new gFlagsLine(CPAP_UserFlag1, COLOR_Yellow, STR_TR_UF1))); // fg->AddLayer((new gFlagsLine(CPAP_UserFlag2, COLOR_DarkGreen, STR_TR_UF2))); // fg->AddLayer((new gFlagsLine(CPAP_UserFlag3, COLOR_Brown, STR_TR_UF3))); // } //fg->AddLayer((new gFlagsLine(PRS1_0B,COLOR_DarkGreen,tr("U0B")))); SF->setBlockZoom(true); SF->AddLayer(new gShadowArea()); SF->AddLayer(new gFlagsLabelArea(fg),LayerLeft,gYAxis::Margin); //SF->AddLayer(new gFooBar(),LayerBottom,0,1); SF->AddLayer(new gXAxis(COLOR_Text,false),LayerBottom,0,20); //gXAxis::Margin); // The following list contains graphs that don't have standard xgrid/yaxis labels QStringList skipgraph; skipgraph.push_back("EventBreakdown"); skipgraph.push_back("SF"); QHash::iterator it; for (it = graphlist.begin(); it != graphlist.end(); ++it) { if (skipgraph.contains(it.key())) continue; it.value()->AddLayer(new gXGrid()); } gLineChart *l; l=new gLineChart(CPAP_FlowRate,COLOR_Black,false,false); gLineOverlaySummary *los=new gLineOverlaySummary(tr("Selection AHI"),5,-4); AddCPAP(l); gGraph *FRW = graphlist[schema::channel[CPAP_FlowRate].code()]; // Draw layer is important... spans first.. // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_CSR, COLOR_CSR, STR_TR_CSR, FT_Span))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_LargeLeak, COLOR_LargeLeak, STR_TR_LL, FT_Span))); //FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Ramp, COLOR_Ramp, schema::channel[CPAP_Ramp].label(), FT_Span))); // Then the graph itself FRW->AddLayer(l); // Then the LineOverlaySummaries // FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Hypopnea,COLOR_Hypopnea,STR_TR_H)))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_PressurePulse,COLOR_PressurePulse,STR_TR_PP,FT_Dot))); // //FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Pressure, COLOR_White,STR_TR_P,FT_Dot))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0B,COLOR_Blue,"0B",FT_Dot))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0E,COLOR_DarkRed,"0E",FT_Dot))); // gLineOverlayBar * rera = new gLineOverlayBar(CPAP_RERA, COLOR_RERA, STR_TR_RE); // if (p_profile->general->calculateRDI()) { // FRW->AddLayer(AddCPAP(los->add(rera))); // } else { // FRW->AddLayer(AddCPAP(rera)); // } // FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Apnea, COLOR_Apnea, STR_TR_UA)))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_VSnore, COLOR_VibratorySnore, STR_TR_VS))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_FlowLimit, COLOR_FlowLimit, STR_TR_FL))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_SensAwake, COLOR_SensAwake, STR_TR_SA))); // FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Obstructive, COLOR_Obstructive, STR_TR_OA)))); // FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_ClearAirway, COLOR_ClearAirway, STR_TR_CA)))); // if (p_profile->cpap->userEventFlagging()) { // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_UserFlag1, COLOR_Yellow, tr("U1"),FT_Bar))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_UserFlag2, COLOR_Orange, tr("U2"),FT_Bar))); // FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_UserFlag3, COLOR_Brown, tr("U3"),FT_Bar))); // } // FRW->AddLayer(AddCPAP(new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2))); FRW->AddLayer(AddOXI(new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2))); //FRW->AddLayer(AddOXI(new gLineOverlayBar(OXI_PulseChange, COLOR_PulseChange, STR_TR_PC,FT_Dot))); // FRW->AddLayer(AddCPAP(los)); bool square=p_profile->appearance->squareWavePlots(); gLineChart *pc=new gLineChart(CPAP_Pressure, COLOR_Pressure, square); graphlist[schema::channel[CPAP_Pressure].code()]->AddLayer(AddCPAP(pc)); graphlist[schema::channel[CPAP_Pressure].code()]->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Ramp, COLOR_Ramp, schema::channel[CPAP_Ramp].label(), FT_Span))); pc->addPlot(CPAP_EPAP, COLOR_EPAP, square); pc->addPlot(CPAP_IPAPLo, COLOR_IPAPLo, square); pc->addPlot(CPAP_IPAP, COLOR_IPAP, square); pc->addPlot(CPAP_IPAPHi, COLOR_IPAPHi, square); if (p_profile->general->calculateRDI()) { AHI->AddLayer(AddCPAP(new gLineChart(CPAP_RDI, COLOR_RDI, square))); // AHI->AddLayer(AddCPAP(new AHIChart(QColor("#37a24b")))); } else { AHI->AddLayer(AddCPAP(new gLineChart(CPAP_AHI, COLOR_AHI, square))); } // this is class wide because the leak redline can be reset in preferences.. // Better way would be having a search for linechart layers in graphlist[...] gLineChart *leakchart=new gLineChart(CPAP_Leak, COLOR_LeakTotal, square); graphlist[schema::channel[CPAP_Leak].code()]->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_LargeLeak, COLOR_LargeLeak, STR_TR_LL, FT_Span))); leakchart->addPlot(CPAP_LeakTotal, COLOR_Leak, square); leakchart->addPlot(CPAP_MaxLeak, COLOR_MaxLeak, square); schema::channel[CPAP_Leak].setUpperThresholdColor(Qt::red); schema::channel[CPAP_Leak].setLowerThresholdColor(Qt::green); graphlist[schema::channel[CPAP_Leak].code()]->AddLayer(AddCPAP(leakchart)); //LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_Leak, COLOR_Leak,square))); //LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_MaxLeak, COLOR_MaxLeak,square))); graphlist[schema::channel[CPAP_Snore].code()]->AddLayer(AddCPAP(new gLineChart(CPAP_Snore, COLOR_Snore, true))); graphlist[schema::channel[CPAP_PTB].code()]->AddLayer(AddCPAP(new gLineChart(CPAP_PTB, COLOR_PTB, square))); graphlist[schema::channel[CPAP_Test1].code()]->AddLayer(AddCPAP(new gLineChart(CPAP_Test1, COLOR_PTB, false))); gLineChart *lc = nullptr; graphlist[schema::channel[CPAP_MaskPressure].code()]->AddLayer(AddCPAP(new gLineChart(CPAP_MaskPressure, COLOR_MaskPressure, false))); graphlist[schema::channel[CPAP_RespRate].code()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_RespRate, COLOR_RespRate, square))); graphlist[schema::channel[POS_Inclination].code()]->AddLayer(AddPOS(new gLineChart(POS_Inclination))); graphlist[schema::channel[POS_Orientation].code()]->AddLayer(AddPOS(new gLineChart(POS_Orientation))); graphlist[schema::channel[CPAP_MinuteVent].code()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_MinuteVent, COLOR_MinuteVent, square))); lc->addPlot(CPAP_TgMV,COLOR_TgMV,square); graphlist[schema::channel[CPAP_TidalVolume].code()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_TidalVolume,COLOR_TidalVolume,square))); //lc->addPlot(CPAP_Test2,COLOR_DarkYellow,square); //graphlist[schema::channel[CPAP_TidalVolume].code()]->AddLayer(AddCPAP(new gLineChart("TidalVolume2",COLOR_Magenta,square))); graphlist[schema::channel[CPAP_FLG].code()]->AddLayer(AddCPAP(new gLineChart(CPAP_FLG, COLOR_FLG, true))); //graphlist[schema::channel[CPAP_RespiratoryEvent].code()]->AddLayer(AddCPAP(new gLineChart(CPAP_RespiratoryEvent,COLOR_Magenta,true))); graphlist[schema::channel[CPAP_IE].code()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_IE, COLOR_IE, square))); graphlist[schema::channel[CPAP_Te].code()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_Te, COLOR_Te, square))); graphlist[schema::channel[CPAP_Ti].code()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_Ti, COLOR_Ti, square))); //lc->addPlot(CPAP_Test2,COLOR:DarkYellow,square); graphlist[schema::channel[ZEO_SleepStage].code()]->AddLayer(AddSTAGE(new gLineChart(ZEO_SleepStage, COLOR_SleepStage, true))); gLineOverlaySummary *los1=new gLineOverlaySummary(STR_UNIT_EventsPerHour,5,-4); gLineOverlaySummary *los2=new gLineOverlaySummary(STR_UNIT_EventsPerHour,5,-4); graphlist[schema::channel[OXI_Pulse].code()]->AddLayer(AddOXI(los1->add(new gLineOverlayBar(OXI_PulseChange, COLOR_PulseChange, STR_TR_PC,FT_Span)))); graphlist[schema::channel[OXI_Pulse].code()]->AddLayer(AddOXI(los1)); graphlist[schema::channel[OXI_SPO2].code()]->AddLayer(AddOXI(los2->add(new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2,FT_Span)))); graphlist[schema::channel[OXI_SPO2].code()]->AddLayer(AddOXI(los2)); graphlist[schema::channel[OXI_Pulse].code()]->AddLayer(AddOXI(new gLineChart(OXI_Pulse, COLOR_Pulse, square))); graphlist[schema::channel[OXI_SPO2].code()]->AddLayer(AddOXI(new gLineChart(OXI_SPO2, COLOR_SPO2, true))); graphlist[schema::channel[OXI_Plethy].code()]->AddLayer(AddOXI(new gLineChart(OXI_Plethy, COLOR_Plethy,false))); // Fix me gLineOverlaySummary *los3=new gLineOverlaySummary(STR_UNIT_EventsPerHour,5,-4); graphlist["INTPULSE"]->AddLayer(AddCPAP(los3->add(new gLineOverlayBar(OXI_PulseChange, COLOR_PulseChange, STR_TR_PC,FT_Span)))); graphlist["INTPULSE"]->AddLayer(AddCPAP(los3)); graphlist["INTPULSE"]->AddLayer(AddCPAP(new gLineChart(OXI_Pulse, COLOR_Pulse, square))); gLineOverlaySummary *los4=new gLineOverlaySummary(STR_UNIT_EventsPerHour,5,-4); graphlist["INTSPO2"]->AddLayer(AddCPAP(los4->add(new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2,FT_Span)))); graphlist["INTSPO2"]->AddLayer(AddCPAP(los4)); graphlist["INTSPO2"]->AddLayer(AddCPAP(new gLineChart(OXI_SPO2, COLOR_SPO2, true))); graphlist[schema::channel[CPAP_PTB].code()]->setForceMaxY(100); graphlist[schema::channel[OXI_SPO2].code()]->setForceMaxY(100); for (it = graphlist.begin(); it != graphlist.end(); ++it) { if (skipgraph.contains(it.key())) continue; it.value()->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); it.value()->AddLayer(new gXAxis(),LayerBottom,0,20); } if (p_profile->cpap->showLeakRedline()) { schema::channel[CPAP_Leak].setUpperThreshold(p_profile->cpap->leakRedline()); } else { schema::channel[CPAP_Leak].setUpperThreshold(0); // switch it off } layout->layout(); QTextCharFormat format = ui->calendar->weekdayTextFormat(Qt::Saturday); format.setForeground(QBrush(COLOR_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(webView); webView->settings()->setFontSize(QWebSettings::DefaultFontSize,QApplication::font().pointSize()); webView->page()->setLinkDelegationPolicy(QWebPage::DelegateAllLinks); connect(webView,SIGNAL(linkClicked(QUrl)),this,SLOT(Link_clicked(QUrl))); int ews=p_profile->general->eventWindowSize(); ui->evViewSlider->setValue(ews); ui->evViewLCD->display(ews); GraphView->LoadSettings("Daily"); icon_on=new QIcon(":/icons/session-on.png"); icon_off=new QIcon(":/icons/session-off.png"); ui->splitter->setVisible(false); if (p_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->setEmptyText(STR_TR_NoData); previous_date=QDate(); ui->calButton->setChecked(p_profile->appearance->calendarVisible() ? Qt::Checked : Qt::Unchecked); on_calButton_toggled(p_profile->appearance->calendarVisible()); } Daily::~Daily() { // disconnect(sessbar, SIGNAL(toggledSession(Session*)), this, SLOT(doToggleSession(Session*))); // Save any last minute changes.. // delete splitter; delete ui; delete icon_on; delete icon_off; } void Daily::showEvent(QShowEvent *) { RedrawGraphs(); } void Daily::closeEvent(QCloseEvent *event) { disconnect(webView,SIGNAL(linkClicked(QUrl)),this,SLOT(Link_clicked(QUrl))); if (previous_date.isValid()) Unload(previous_date); GraphView->SaveSettings("Daily"); QWidget::closeEvent(event); event->accept(); } void Daily::doToggleSession(Session * sess) { Q_UNUSED(sess) sess->setEnabled(!sess->enabled()); // sess->StoreSummary(); Day *day=p_profile->GetDay(previous_date,MT_CPAP); if (day) { day->machine->Save(); this->LoadDate(previous_date); } } 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=nullptr; if (code=="togglecpapsession") { // Enable/Disable CPAP session day=p_profile->GetDay(previous_date,MT_CPAP); Session *sess=day->find(sid); if (!sess) return; int i=webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-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); webView->page()->mainFrame()->setScrollBarValue(Qt::Vertical, webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-i); return; } else if (code=="toggleoxisession") { // Enable/Disable Oximetry session day=p_profile->GetDay(previous_date,MT_OXIMETER); Session *sess=day->find(sid); if (!sess) return; int i=webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-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); webView->page()->mainFrame()->setScrollBarValue(Qt::Vertical, webView->page()->mainFrame()->scrollBarMaximum(Qt::Vertical)-i); return; } else if (code=="cpap") { day=p_profile->GetDay(previous_date,MT_CPAP); } else if (code=="oxi") { //day=p_profile->GetDay(previous_date,MT_OXIMETER); //Session *sess=day->machine->sessionlist[sid]; return; } else if (code=="event") { QList list=ui->treeWidget->findItems(schema::channel[sid].fullname(),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].fullname()),"",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() { GraphView->setDay(nullptr); ui->splitter->setVisible(true); QDate d; if (previous_date.isValid()) { d=previous_date; // Unload(d); } d=p_profile->LastDay(); if (!d.isValid()) { d=ui->calendar->selectedDate(); } on_calendar_currentPageChanged(d.year(),d.month()); // this fires a signal which unloads the old and loads the new ui->calendar->blockSignals(true); ui->calendar->setSelectedDate(d); ui->calendar->blockSignals(false); Load(d); ui->calButton->setText(ui->calendar->selectedDate().toString(Qt::TextDate)); graphView()->redraw(); } 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=nullptr; QHash mcroot; QHash mccnt; int total_events=0; bool userflags=p_profile->cpap->userEventFlagging(); qint64 drift=0, clockdrift=p_profile->cpap->clockDrift()*1000L; for (QList::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_UserFlag3) && (code!=CPAP_NRI) && (code!=CPAP_LeakFlag) && (code!=CPAP_LargeLeak) && (code!=CPAP_ExP) && (code!=CPAP_FlowLimit) && (code!=CPAP_SensAwake) && (code!=CPAP_PressurePulse) && (code!=CPAP_VSnore2) && (code!=CPAP_VSnore)) continue; if (!userflags && ((code==CPAP_UserFlag1) || (code==CPAP_UserFlag2) || (code==CPAP_UserFlag3))) continue; drift=((*s)->machine()->type() == MT_CPAP) ? clockdrift : 0; QTreeWidgetItem *mcr; if (mcroot.find(code)==mcroot.end()) { int cnt=day->count(code); if (!cnt) continue; // If no events than don't bother showing.. total_events+=cnt; QString st=schema::channel[code].fullname(); if (st.isEmpty()) { st=QString("Fixme %1").arg(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;zraw(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()); } QTreeWidgetItem * start = new QTreeWidgetItem(QStringList(tr("Session Start Times"))); QTreeWidgetItem * end = new QTreeWidgetItem(QStringList(tr("Session End Times"))); tree->insertTopLevelItem(cnt++ , start); tree->insertTopLevelItem(cnt++ , end); for (QList::iterator s=day->begin(); s!=day->end(); ++s) { QDateTime st = QDateTime::fromMSecsSinceEpoch((*s)->first()); QDateTime et = QDateTime::fromMSecsSinceEpoch((*s)->last()); QTreeWidgetItem * item = new QTreeWidgetItem(QStringList(st.toString("HH:mm:ss"))); item->setData(0,Qt::UserRole, (*s)->first()); start->addChild(item); item = new QTreeWidgetItem(QStringList(et.toString("HH:mm:ss"))); item->setData(0,Qt::UserRole, (*s)->last()); end->addChild(item); } //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; QTextCharFormat stageday; cpaponly.setForeground(QBrush(COLOR_Blue, Qt::SolidPattern)); cpaponly.setFontWeight(QFont::Normal); cpapjour.setForeground(QBrush(COLOR_Blue, Qt::SolidPattern)); cpapjour.setFontWeight(QFont::Bold); oxiday.setForeground(QBrush(COLOR_Red, Qt::SolidPattern)); oxiday.setFontWeight(QFont::Normal); oxicpap.setForeground(QBrush(COLOR_Red, Qt::SolidPattern)); oxicpap.setFontWeight(QFont::Bold); stageday.setForeground(QBrush(COLOR_Magenta, Qt::SolidPattern)); stageday.setFontWeight(QFont::Bold); jourday.setForeground(QBrush(COLOR_Black, Qt::SolidPattern)); jourday.setFontWeight(QFont::Bold); nodata.setForeground(QBrush(COLOR_Black, Qt::SolidPattern)); nodata.setFontWeight(QFont::Normal); bool hascpap=p_profile->GetDay(date,MT_CPAP)!=nullptr; bool hasoxi=p_profile->GetDay(date,MT_OXIMETER)!=nullptr; bool hasjournal=p_profile->GetDay(date,MT_JOURNAL)!=nullptr; bool hasstage=p_profile->GetDay(date,MT_SLEEPSTAGE)!=nullptr; bool haspos=p_profile->GetDay(date,MT_POSITION)!=nullptr; if (hascpap) { if (hasoxi) { ui->calendar->setDateTextFormat(date,oxicpap); } else if (hasjournal) { ui->calendar->setDateTextFormat(date,cpapjour); } else if (hasstage || haspos) { ui->calendar->setDateTextFormat(date,stageday); } else { ui->calendar->setDateTextFormat(date,cpaponly); } } else if (hasoxi) { ui->calendar->setDateTextFormat(date,oxiday); } else if (hasjournal) { ui->calendar->setDateTextFormat(date,jourday); } else if (hasstage) { ui->calendar->setDateTextFormat(date,oxiday); } else if (haspos) { ui->calendar->setDateTextFormat(date,oxiday); } else { ui->calendar->setDateTextFormat(date,nodata); } ui->calendar->setHorizontalHeaderFormat(QCalendarWidget::ShortDayNames); } void Daily::LoadDate(QDate date) { if (!date.isValid()) { qDebug() << "LoadDate called with invalid date"; return; } 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() { graphView()->releaseKeyboard(); QTime time; time_t unload_time, load_time, other_time; time.start(); this->setCursor(Qt::BusyCursor); if (previous_date.isValid()) { // GraphView->fadeOut(); Unload(previous_date); } unload_time=time.restart(); //bool fadedir=previous_date < ui->calendar->selectedDate(); ZombieMeterMoved=false; Load(ui->calendar->selectedDate()); load_time=time.restart(); //GraphView->fadeIn(fadedir); GraphView->redraw(); ui->calButton->setText(ui->calendar->selectedDate().toString(Qt::TextDate)); ui->calendar->setFocus(Qt::ActiveWindowFocusReason); if (p_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); other_time=time.restart(); qDebug() << "Page change time (in ms): Unload ="<getDaily()->sessionBar(); } qDebug() << "Request for unknown MyWebPage plugin"; return new QWidget(); } MyWebView::MyWebView(QWidget *parent): QWebView(parent), m_page(this) { // Set the page of our own PageView class, MyPageView, // because only objects of this class will handle // object-tags correctly. setPage(&m_page); } QString Daily::getSessionInformation(Day * cpap, Day * oxi, Day * stage, Day * posit) { QString html; QList list; if (cpap) list.push_back(cpap); if (oxi) list.push_back(oxi); if (stage) list.push_back(stage); if (posit) list.push_back(posit); 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 (QList::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->loaderName()+QString(":#%1").arg((*s)->session(),8,10,QChar('0')); #define DEBUG_SESSIONS #ifdef DEBUG_SESSIONS tooltip += " "+QString::number(len)+"s"; #endif // 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; case MT_POSITION: type="stage"; html+=tr("Position Sensor 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+=""; if ((cpap && cpap->settingExists(CPAP_BrokenSummary))) { html+="
%1
 
"+tr("Machine Settings Unavailable")+"

\n"; return html; } QMap other; QHash::iterator it = cpap->sessions.at(0)->settings.begin(); QHash::iterator it_end = cpap->sessions.at(0)->settings.end(); QMap first; for (; it != it_end; ++it) { ChannelID code = it.key(); if ((code <= 1) || (code == RMS9_MaskOnTime)) continue; schema::Channel & chan = schema::channel[code]; QString data; if (chan.datatype() == schema::LOOKUP) { data = chan.option(it.value().toInt()); } else if (chan.datatype() == schema::BOOL) { data = (it.value().toBool() ? STR_TR_Yes : STR_TR_No); } else if (chan.datatype() == schema::DOUBLE) { data = QString().number(it.value().toDouble(),'f',1) + " "+chan.units(); } else { data = it.value().toString() + " "+ chan.units(); } QString tmp = QString("%1%2%3") .arg(schema::channel[code].label()) .arg(schema::channel[code].description()) .arg(data); if ((code == CPAP_Mode) || (code == CPAP_IPAP) || (code == CPAP_EPAP) || (code == CPAP_IPAPHi) || (code == CPAP_EPAPHi) || (code == CPAP_IPAPLo) || (code == CPAP_EPAPLo) || (code == CPAP_PressureMin) || (code == CPAP_PressureMax) || (code == CPAP_Pressure) || (code == CPAP_PSMin) || (code == CPAP_PSMax) || (code == CPAP_PS)) { first[code] = tmp; } else { other[schema::channel[code].label()] = tmp; } } ChannelID order[] = { CPAP_Mode, CPAP_Pressure, CPAP_PressureMin, CPAP_PressureMax, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAPHi, CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_PS, CPAP_PSMin, CPAP_PSMax }; int os = sizeof(order) / sizeof(ChannelID); for (int i=0 ;i < os; ++i) { if (first.contains(order[i])) html += first[order[i]]; } for (QMap::iterator it = other.begin(); it != other.end(); ++it) { html += it.value(); } /* ChannelID pr_level_chan = NoChannel; ChannelID pr_mode_chan = NoChannel; ChannelID hum_stat_chan = NoChannel; ChannelID hum_level_chan = NoChannel; CPAPLoader * loader = dynamic_cast(cpap->machine->loader()); if (loader) { pr_level_chan = loader->PresReliefLevel(); pr_mode_chan = loader->PresReliefMode(); hum_stat_chan = loader->HumidifierConnected(); hum_level_chan = loader->HumidifierLevel(); } if ((pr_level_chan != NoChannel) && (cpap->settingExists(pr_level_chan))) { QString flexstr = cpap->getPressureRelief(); html+=QString("%1%2%3") .arg(schema::channel[pr_mode_chan].label()) .arg(schema::channel[pr_mode_chan].description()) .arg(flexstr); } if (cpap->settingExists(hum_level_chan)) { int humid=round(cpap->settings_wavg(hum_level_chan)); html+=QString(""+schema::channel[hum_level_chan].label()+"%1%2") .arg(schema::channel[hum_level_chan].description()) .arg(humid == 0 ? STR_GEN_Off : "x"+QString::number(humid)); } */ html+=""; 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->brand()+" "+oxi->machine->series()+"
 
%1: %2 (%3%)
%1: %2 (%3%)
%1: %2%
\n"; html+="
\n"; } return html; } QString Daily::getCPAPInformation(Day * cpap) { QString html; if (!cpap) return html; MachineInfo info = cpap->machine->getInfo(); html="\n"; html+="\n"; //CPAPMode mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode); html+="\n"; if ((cpap && cpap->settingExists(CPAP_BrokenSummary))) { html+="\n"; html+=QString("").arg(""+STR_MessageBox_PleaseNote+": "+ tr("This day has missing pressure, mode and settings data.")); } html+="
"+info.series+" "+info.model+""; QString tooltip=(info.brand+"\n"+info.series+" "+info.modelnumber+"\n"+info.serial); tooltip=tooltip.replace(" "," "); html+=tooltip; html+="
"; html+=tr("PAP Mode: %1
").arg(cpap->getCPAPMode()); html+= cpap->getPressureSettings(); html+="
 
%1
\n"; html+="
\n"; return html; } QString Daily::getStatisticsInfo(Day * cpap,Day * oxi,Day *pos) { QList list; list.push_back(cpap); list.push_back(oxi); list.push_back(pos); int mididx=p_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=p_profile->general->prefCalcPercentile()/100.0; SummaryType ST_max=p_profile->general->prefCalcMax() ? ST_PERC : ST_MAX; const EventDataType maxperc=0.995F; QString midname; if (ST_mid==ST_WAVG) midname=STR_TR_WAvg; else if (ST_mid==ST_AVG) midname=STR_TR_Avg; else if (ST_mid==ST_PERC) midname=STR_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%2").arg(percentile*100,0,'f',0).arg(STR_UNIT_Percentage)) .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, POS_Inclination, POS_Orientation }; 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) || (cpap && cpap->summaryOnly()))) { html+="\n"; html+=QString("").arg(""+STR_MessageBox_PleaseNote+" "+ tr("This day just contains summary data, only limited information is available .")); } else if (cpap && p_profile->cpap->showLeakRedline()) { float rlt = cpap->timeAboveThreshold(CPAP_Leak, p_profile->cpap->leakRedline()) / 60.0; float pc = 100.0 / cpap->hours() * rlt; html+=""; html+="").arg(pc, 0, 'f', 3); } if (cpap) { int l = cpap->sum(CPAP_Ramp); if (l > 0) { html+="").arg(l / 3600, 2, 10, QChar('0')).arg((l / 60) % 60, 2, 10, QChar('0')).arg(l % 60, 2, 10, QChar('0')); float v = (cpap->hours() - (float(l) / 3600.0)); int q = v * 3600.0; html+="").arg(q / 3600, 2, 10, QChar('0')).arg((q / 60) % 60, 2, 10, QChar('0')).arg(q % 60, 2, 10, QChar('0')); EventDataType hc = cpap->count(CPAP_Hypopnea) - cpap->countInsideSpan(CPAP_Ramp, CPAP_Hypopnea); EventDataType oc = cpap->count(CPAP_Obstructive) - cpap->countInsideSpan(CPAP_Ramp, CPAP_Obstructive); EventDataType tc = cpap->count(CPAP_Hypopnea) + cpap->count(CPAP_Obstructive); EventDataType ahi = (hc+oc) / v; html+="").arg(ahi, 0, 'f', 2); } } html+="
%1
%1%2%3%4%5
%1%6%2%3%4%5
 
%1
 
"+tr("Time over leak redline")+ QString("%1%
"+tr("Total ramp time")+ QString("%1:%2:%3
"+tr("Time outside of ramp")+ QString("%1:%2:%3
"+tr("AHI excluding ramp")+ QString("%1
\n"; html+="
\n"; return html; } QString Daily::getEventBreakdown(Day * cpap) { Q_UNUSED(cpap) QString html; html+="\n"; html+="
"; return html; } QString Daily::getSleepTime(Day * cpap, Day * oxi) { QString html; Day * day=nullptr; 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) { dateDisplay->setText(""+date.toString(Qt::SystemLocaleLongDate)+""); previous_date=date; Day *cpap=p_profile->GetDay(date,MT_CPAP); Day *oxi=p_profile->GetDay(date,MT_OXIMETER); Day *stage=p_profile->GetDay(date,MT_SLEEPSTAGE); Day *posit=p_profile->GetDay(date,MT_POSITION); if (!p_profile->session->cacheSessions()) { // Getting trashed on purge last day... // lastcpapday can get purged and be invalid if (lastcpapday && (lastcpapday!=cpap)) { for (QList::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, but its timestamps are too different, so the Graphs will not be linked."),"",3000); gr=1; } else gr=0; (*GraphView)[schema::channel[OXI_Pulse].code()]->setGroup(gr); (*GraphView)[schema::channel[OXI_SPO2].code()]->setGroup(gr); (*GraphView)[schema::channel[OXI_Plethy].code()]->setGroup(gr); } lastcpapday=cpap; QString html="" "" "" "" ""; QString tmp; UpdateOXIGraphs(oxi); UpdateCPAPGraphs(cpap); UpdateSTAGEGraphs(stage); UpdatePOSGraphs(posit); UpdateEventsTree(ui->treeWidget,cpap); // FIXME: // Generating entire statistics because bookmarks may have changed.. (This updates the side panel too) mainwin->GenerateStatistics(); snapGV->setDay(cpap); GraphView->ResetBounds(false); if (!cpap && !oxi) { scrollbar->hide(); } else { scrollbar->show(); } QString modestr; CPAPMode mode=MODE_UNKNOWN; QString a; bool isBrick=false; updateGraphCombo(); ui->eventsCombo->clear(); if (cpap) { QMap flags; for (int i=0; i< cpap->size(); ++i) { Session * sess = cpap->sessions.at(i); QHash >::iterator it; for (it = sess->eventlist.begin(); it != sess->eventlist.end(); ++it) { ChannelID code = it.key(); schema::Channel * chan = &schema::channel[code]; const QString & str = chan->label(); if (flags.contains(str)) continue; if ((chan->type() == schema::FLAG) || (chan->type() == schema::MINOR_FLAG) || (chan->type() == schema::SPAN)) { flags[str] = chan; } } } QMap::iterator fit; int c = 0; for (fit = flags.begin(); fit != flags.end(); ++fit) { ui->eventsCombo->addItem(fit.value()->enabled() ? *icon_on : * icon_off, fit.key(), fit.value()->id()); ui->eventsCombo->setItemData(c, fit.value()->fullname(), Qt::ToolTipRole); c++; } } if (cpap) { //QHash > list; float hours=cpap->hours(); if (GraphView->isEmpty() && (hours>0)) { if (!p_profile->hasChannel(CPAP_Obstructive) && !p_profile->hasChannel(CPAP_Hypopnea)) { GraphView->setEmptyText(tr("No Graphs :(")); isBrick=true; } } mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode); modestr=schema::channel[CPAP_Mode].m_options[mode]; EventDataType ahi=(cpap->count(CPAP_Obstructive)+cpap->count(CPAP_Hypopnea)+cpap->count(CPAP_ClearAirway)+cpap->count(CPAP_Apnea)); if (p_profile->general->calculateRDI()) ahi+=cpap->count(CPAP_RERA); ahi/=hours; EventDataType csr,uai,oai,hi,cai,rei,fli,sai,nri,lki,vs,vs2,exp,lk2,uf1, uf2; if (!isBrick && hours>0) { html+="\n"; ChannelID ahichan=CPAP_AHI; QString ahiname=STR_TR_AHI; if (p_profile->general->calculateRDI()) { ahichan=CPAP_RDI; ahiname=STR_TR_RDI; } html+=""; html+=QString("\n") .arg("#F88017").arg(COLOR_Text.name()).arg(ahiname).arg(schema::channel[ahichan].fullname()).arg(ahi,0,'f',2); html+="\n"; html+="
%3%4   %5
\n"; html+=getCPAPInformation(cpap); html+=getSleepTime(cpap,oxi); struct ChannelInfo { ChannelID id; QColor color; QColor color2; EventDataType value; }; ChannelInfo chans[]={ { CPAP_Hypopnea, COLOR_Hypopnea, Qt::white, hi=(cpap->count(CPAP_ExP)+cpap->count(CPAP_Hypopnea))/hours }, { CPAP_Obstructive, COLOR_Obstructive, Qt::black, oai=cpap->count(CPAP_Obstructive)/hours }, { CPAP_Apnea, COLOR_Apnea, Qt::black, uai=cpap->count(CPAP_Apnea)/hours }, { CPAP_ClearAirway, COLOR_ClearAirway, Qt::black, cai=cpap->count(CPAP_ClearAirway)/hours }, { CPAP_NRI, COLOR_NRI, Qt::black, nri=cpap->count(CPAP_NRI)/hours }, { CPAP_FlowLimit, COLOR_FlowLimit, Qt::white, fli=cpap->count(CPAP_FlowLimit)/hours }, { CPAP_SensAwake, COLOR_SensAwake, Qt::black, sai=cpap->count(CPAP_SensAwake)/hours }, { CPAP_ExP, COLOR_ExP, Qt::black, exp=cpap->count(CPAP_ExP)/hours }, { CPAP_RERA, COLOR_RERA, Qt::black, rei=cpap->count(CPAP_RERA)/hours }, { CPAP_VSnore, COLOR_VibratorySnore, Qt::black, vs=cpap->count(CPAP_VSnore)/cpap->hours() }, { CPAP_VSnore2, COLOR_VibratorySnore, Qt::black, vs2=cpap->count(CPAP_VSnore2)/cpap->hours() }, { CPAP_LeakFlag, COLOR_LeakFlag, Qt::black, lki=cpap->count(CPAP_LeakFlag)/hours }, { CPAP_LargeLeak, COLOR_LargeLeak, Qt::black, lk2=(100.0/cpap->hours())*(cpap->sum(CPAP_LargeLeak)/3600.0) }, { CPAP_CSR, COLOR_CSR, Qt::black, csr=(100.0/cpap->hours())*(cpap->sum(CPAP_CSR)/3600.0) }, { CPAP_UserFlag1, COLOR_UserFlag1, Qt::black, uf1=cpap->count(CPAP_UserFlag1)/hours }, { CPAP_UserFlag2, COLOR_UserFlag2, Qt::black, uf2=cpap->count(CPAP_UserFlag2)/hours }, }; int numchans=sizeof(chans)/sizeof(ChannelInfo); html+="\n"; for (int i=0;ichannelHasData(chans[i].id)) continue; if ((cpap->machine->loaderName() == STR_MACH_PRS1) && (chans[i].id == CPAP_VSnore)) continue; html+=QString("\n") .arg(chans[i].color.name()).arg(chans[i].color2.name()).arg(schema::channel[chans[i].id].fullname()).arg(chans[i].value,0,'f',2).arg(chans[i].id); // keep in case tooltips are needed //html+=QString("\n") // .arg(chans[i].color.name()).arg(chans[i].color2.name()).arg(chans[i].name).arg(schema::channel[chans[i].id].description()).arg(chans[i].value,0,'f',2).arg(chans[i].id); } html+="
%3%4
%3%4%5
"; html+="\n"; // Show Event Breakdown pie chart if ((hours > 0) && p_profile->appearance->graphSnapshots()) { // AHI Pie Chart if ((oai+hi+cai+uai+rei+fli+sai)>0) { html+=""; html+=QString("").arg(tr("Event Breakdown")); eventBreakdownPie()->setShowTitle(false); int w=155; int h=155; QPixmap pixmap=eventBreakdownPie()->renderPixmap(w,h,false); if (!pixmap.isNull()) { QByteArray byteArray; QBuffer buffer(&byteArray); // use buffer to store pixmap into byteArray buffer.open(QIODevice::WriteOnly); pixmap.save(&buffer, "PNG"); html += "\n").arg(w).arg(h); } else { html += "\n"; } } else if (cpap->channelHasData(CPAP_Obstructive) || cpap->channelHasData(CPAP_Hypopnea) || cpap->channelHasData(CPAP_ClearAirway) || cpap->channelHasData(CPAP_RERA) || cpap->channelHasData(CPAP_Apnea) || cpap->channelHasData(CPAP_FlowLimit) || cpap->channelHasData(CPAP_SensAwake) ) { html += "\n"; } } html+="
 
%1
Unable to display Pie Chart on this system
\n"; } else { html+="\n"; if (!isBrick) { html+="\n"; if (cpap->size()>0) { html+=""; html+="\n"; } else { html+=""; html+="\n"; } } else { // machine is a brick html+=""; html+="\n"; html+="\n"; } html+="\n"; html+="
 

"+tr("Sessions all off!")+"

"+tr("Sessions exist for this day but are switched off.")+"

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

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

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

"+tr("Sorry, your machine only provides compliance data.")+"
"+tr("Complain to your Equipment Provider!")+"
 
\n"; } html+="
\n"; } // if (!CPAP) else html+=getSleepTime(cpap,oxi); if ((cpap && !isBrick && (cpap->hours()>0)) || oxi || posit) { html+=getStatisticsInfo(cpap,oxi,posit); } else { if (cpap && cpap->hours()==0) { } else { html+="\n"; html+="\n"; html+="\n"; html+="
"+tr("No data available")+"
 
\n"; html+="
\n"; } } html+=getOximeterInformation(oxi); html+=getMachineSettings(cpap); html+=getSessionInformation(cpap,oxi,stage,posit); html+=""; QColor cols[]={ QColor("gold"), QColor("light blue"), }; const int maxcolors=sizeof(cols)/sizeof(QColor); QList::iterator i; // WebView trashes it without asking.. :( if (cpap) { sessbar=new SessionBar(this); sessbar->setMouseTracking(true); connect(sessbar, SIGNAL(sessionClicked(Session*)), this, SLOT(doToggleSession(Session*))); int c=0; for (i=cpap->begin();i!=cpap->end();++i) { Session * s=*i; sessbar->add(s, cols[c % maxcolors]); c++; } } else sessbar=nullptr; //sessbar->update(); 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 (p_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=p_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); qint64 clockdrift=p_profile->cpap->clockDrift()*1000L,drift; Day * dday=p_profile->GetDay(previous_date,MT_CPAP); drift=(dday!=nullptr) ? clockdrift : 0; 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 (p_profile->general->unitSystem()==US_Archiac) { 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::clearLastDay() { lastcpapday=nullptr; } void Daily::Unload(QDate date) { if (!date.isValid()) { date = getDate(); if (!date.isValid()) { graphView()->setDay(nullptr); return; } } webView->setHtml(""); 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()) { // blah.. was updating overview graphs here.. Was too slow. } Machine *jm=p_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(COLOR_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=p_profile->GetMachine(MT_JOURNAL); if (!m) { m=new Machine(0); MachineInfo info; info.loadername = "Journal"; info.serial = m->hexid(); info.brand = "Journal"; info.type = MT_JOURNAL; m->setInfo(info); p_profile->AddMachine(m); } Session *sess=new Session(m,0); qint64 st,et; Day *cday=p_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+3600000L; } sess->set_first(st); sess->set_last(et); sess->SetChanged(true); m->AddSession(sess); return sess; } Session * Daily::GetJournalSession(QDate date) // Get the first journal session { Day *journal=p_profile->GetDay(date,MT_JOURNAL); if (!journal) return nullptr; //CreateJournalSession(date); QList::iterator s; s=journal->begin(); if (s!=journal->end()) return *s; return nullptr; } 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::UpdateSTAGEGraphs(Day *day) { //if (!day) return; if (day) { day->OpenEvents(); } for (QList::iterator g=STAGEData.begin();g!=STAGEData.end();g++) { (*g)->SetDay(day); } } void Daily::UpdatePOSGraphs(Day *day) { //if (!day) return; if (day) { day->OpenEvents(); } for (QList::iterator g=POSData.begin();g!=POSData.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() { // setting this here, because it needs to be done when preferences change if (p_profile->cpap->showLeakRedline()) { schema::channel[CPAP_Leak].setUpperThreshold(p_profile->cpap->leakRedline()); } else { schema::channel[CPAP_Leak].setUpperThreshold(0); // switch it off } 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(p_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_GRAPH_SleepFlags); 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 (!p_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 (p_profile->GetDay(d)) { LoadDate(d); break; } } } } void Daily::on_nextDayButton_clicked() { if (!p_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 (p_profile->GetDay(d)) { LoadDate(d); break; } } } } void Daily::on_calButton_toggled(bool checked) { bool b=checked; ui->calendarFrame->setVisible(b); p_profile->appearance->setCalendarVisible(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 > p_profile->LastDay()) d=p_profile->LastDay(); LoadDate(d); } void Daily::on_evViewSlider_valueChanged(int value) { ui->evViewLCD->display(value); p_profile->general->setEventWindowSize(value); int winsize=value*60; gGraph *g=GraphView->findGraph(STR_GRAPH_SleepFlags); 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; // qint64 clockdrift=p_profile->cpap->clockDrift()*1000L,drift; // Day * dday=p_profile->GetDay(previous_date,MT_CPAP); // drift=(dday!=nullptr) ? clockdrift : 0; 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=0,et2=0,st3,et3; Day * day=p_profile->GetGoodDay(previous_date,MT_CPAP); if (day) { st2=day->first(); et2=day->last(); } Day * oxi=p_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(tr("This bookmarked is in a currently disabled area..")); return; } if (stet2) et=et2; GraphView->SetXBounds(st,et); GraphView->redraw(); } void Daily::addBookmark(qint64 st, qint64 et, QString text) { ui->bookmarkTable->blockSignals(true); QDateTime d=QDateTime::fromTime_t(st/1000L); int row=ui->bookmarkTable->rowCount(); ui->bookmarkTable->insertRow(row); QTableWidgetItem *tw=new QTableWidgetItem(text); 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); qint64 clockdrift=p_profile->cpap->clockDrift()*1000L,drift; Day * day=p_profile->GetDay(previous_date,MT_CPAP); drift=(day!=nullptr) ? clockdrift : 0; // Counter CPAP clock drift for storage, in case user changes it later on // This won't fix the text string names.. tw->setData(Qt::UserRole,st-drift); tw->setData(Qt::UserRole+1,et-drift); ui->bookmarkTable->blockSignals(false); update_Bookmarks(); mainwin->updateFavourites(); } void Daily::on_addBookmarkButton_clicked() { qint64 st,et; GraphView->GetXBounds(st,et); QDateTime d=QDateTime::fromTime_t(st/1000L); addBookmark(st,et, tr("Bookmark at %1").arg(d.time().toString("HH:mm:ss"))); } 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); if (mainwin->getOverview()) mainwin->getOverview()->ResetGraph("Zombie"); } void Daily::on_bookmarkTable_itemChanged(QTableWidgetItem *item) { Q_UNUSED(item); update_Bookmarks(); } void Daily::on_weightSpinBox_valueChanged(double arg1) { // Update the BMI display double kg; if (p_profile->general->unitSystem()==US_Archiac) { kg=((arg1*pound_convert) + (ui->ouncesSpinBox->value()*ounce_convert)) / 1000.0; } else kg=arg1; double height=p_profile->user->height()/100.0; if ((height>0) && (kg>0)) { double bmi=kg/(height * height); ui->BMI->display(bmi); ui->BMI->setVisible(true); } } void Daily::on_weightSpinBox_editingFinished() { double arg1=ui->weightSpinBox->value(); double height=p_profile->user->height()/100.0; Session *journal=GetJournalSession(previous_date); if (!journal) { journal=CreateJournalSession(previous_date); } double kg; if (p_profile->general->unitSystem()==US_Archiac) { kg=((arg1*pound_convert) + (ui->ouncesSpinBox->value()*ounce_convert)) / 1000.0; } else { kg=arg1; } journal->settings[Journal_Weight]=kg; gGraphView *gv=mainwin->getOverview()->graphView(); gGraph *g; if (gv) { g=gv->findGraph(STR_GRAPH_Weight); if (g) g->setDay(nullptr); } 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_GRAPH_BMI); if (g) g->setDay(nullptr); } } journal->SetChanged(true); } void Daily::on_ouncesSpinBox_valueChanged(int arg1) { // just update for BMI display double height=p_profile->user->height()/100.0; double kg=((ui->weightSpinBox->value()*pound_convert) + (arg1*ounce_convert)) / 1000.0; if ((height>0) && (kg>0)) { double bmi=kg/(height * height); ui->BMI->display(bmi); ui->BMI->setVisible(true); } } void Daily::on_ouncesSpinBox_editingFinished() { double arg1=ui->ouncesSpinBox->value(); Session *journal=GetJournalSession(previous_date); if (!journal) { journal=CreateJournalSession(previous_date); } double height=p_profile->user->height()/100.0; double kg=((ui->weightSpinBox->value()*pound_convert) + (arg1*ounce_convert)) / 1000.0; journal->settings[Journal_Weight]=kg; gGraph *g; if (mainwin->getOverview()) { g=mainwin->getOverview()->graphView()->findGraph(STR_GRAPH_Weight); if (g) g->setDay(nullptr); } 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_GRAPH_BMI); if (g) g->setDay(nullptr); } } journal->SetChanged(true); if (mainwin->getOverview()) mainwin->getOverview()->ResetGraph(STR_GRAPH_Weight); } QString Daily::GetDetailsText() { webView->triggerPageAction(QWebPage::SelectAll); QString text=webView->page()->selectedText(); webView->triggerPageAction(QWebPage::MoveToEndOfDocument); webView->triggerPageAction(QWebPage::SelectEndOfDocument); return text; } void Daily::on_graphCombo_activated(int index) { if (index<0) return; gGraph *g; QString s; s=ui->graphCombo->currentText(); bool b=!ui->graphCombo->itemData(index,Qt::UserRole).toBool(); ui->graphCombo->setItemData(index,b,Qt::UserRole); if (b) { ui->graphCombo->setItemIcon(index,*icon_on); } else { ui->graphCombo->setItemIcon(index,*icon_off); } g=GraphView->findGraphTitle(s); g->setVisible(b); updateCube(); GraphView->updateScale(); GraphView->redraw(); } void Daily::updateCube() { //brick.. if ((GraphView->visibleGraphs()==0)) { ui->toggleGraphs->setArrowType(Qt::UpArrow); ui->toggleGraphs->setToolTip(tr("Show all graphs")); ui->toggleGraphs->blockSignals(true); ui->toggleGraphs->setChecked(true); ui->toggleGraphs->blockSignals(false); if (ui->graphCombo->count() > 0) { GraphView->setEmptyText(tr("No Graphs On!")); } else { if (!p_profile->GetGoodDay(getDate(), MT_CPAP) && !p_profile->GetGoodDay(getDate(), MT_OXIMETER) && !p_profile->GetGoodDay(getDate(), MT_SLEEPSTAGE) && !p_profile->GetGoodDay(getDate(), MT_POSITION)) { GraphView->setEmptyText(STR_TR_NoData); } else { GraphView->setEmptyText(tr("Summary Only :(")); } } } else { ui->toggleGraphs->setArrowType(Qt::DownArrow); ui->toggleGraphs->setToolTip(tr("Hide all graphs")); ui->toggleGraphs->blockSignals(true); ui->toggleGraphs->setChecked(false); ui->toggleGraphs->blockSignals(false); } } void Daily::on_toggleGraphs_clicked(bool checked) { QString s; QIcon *icon=checked ? icon_off : icon_on; for (int i=0;igraphCombo->count();i++) { s=ui->graphCombo->itemText(i); ui->graphCombo->setItemIcon(i,*icon); ui->graphCombo->setItemData(i,!checked,Qt::UserRole); } for (int i=0;isize();i++) { (*GraphView)[i]->setVisible(!checked); } updateCube(); GraphView->updateScale(); GraphView->redraw(); } void Daily::updateGraphCombo() { ui->graphCombo->clear(); gGraph *g; for (int i=0;isize();i++) { g=(*GraphView)[i]; if (g->isEmpty()) continue; if (g->visible()) { ui->graphCombo->addItem(*icon_on,g->title(),true); } else { ui->graphCombo->addItem(*icon_off,g->title(),false); } } ui->graphCombo->setCurrentIndex(0); updateCube(); } void Daily::on_zoomFullyOut_clicked() { GraphView->ResetBounds(true); GraphView->redraw(); } void Daily::on_resetLayoutButton_clicked() { GraphView->resetLayout(); } void Daily::on_eventsCombo_activated(int index) { if (index<0) return; ChannelID code = ui->eventsCombo->itemData(index, Qt::UserRole).toUInt(); schema::Channel * chan = &schema::channel[code]; bool b = !chan->enabled(); chan->setEnabled(b); ui->eventsCombo->setItemIcon(index,b ? *icon_on : *icon_off); GraphView->redraw(); } void Daily::on_toggleEvents_clicked(bool checked) { QString s; QIcon *icon=checked ? icon_on : icon_off; ui->toggleEvents->setArrowType(checked ? Qt::DownArrow : Qt::UpArrow); ui->toggleEvents->setToolTip(checked ? tr("Hide all events") : tr("Show all events")); // ui->toggleEvents->blockSignals(true); // ui->toggleEvents->setChecked(false); // ui->toggleEvents->blockSignals(false); for (int i=0;ieventsCombo->count();i++) { // s=ui->eventsCombo->itemText(i); ui->eventsCombo->setItemIcon(i,*icon); ChannelID code = ui->eventsCombo->itemData(i).toUInt(); schema::channel[code].setEnabled(checked); } updateCube(); GraphView->redraw(); }