diff --git a/Graphs/gBarChart.cpp b/Graphs/gBarChart.cpp index 7a5613f6..affae664 100644 --- a/Graphs/gBarChart.cpp +++ b/Graphs/gBarChart.cpp @@ -18,7 +18,8 @@ SummaryChart::SummaryChart(Profile *p,QString label,GraphType type) addGLBuf(quads=new GLBuffer(color,20000,GL_QUADS)); addGLBuf(lines=new GLBuffer(color,20000,GL_LINES)); quads->forceAntiAlias(true); - //lines->setSize(2); + lines->setSize(2); + lines->forceAntiAlias(false); m_empty=true; hl_day=-1; } @@ -263,13 +264,14 @@ void SummaryChart::paint(gGraph & w,int left, int top, int width, int height) quads->add(x2,py-h,col2); quads->add(x2,py,col2); if (barw>2) { - lines->add(x1,py,x1,py-h,blk); - lines->add(x1,py-h,x2,py-h,blk); - lines->add(x1,py,x2,py,blk); - lines->add(x2,py,x2,py-h,blk); + outlines->add(x1,py,x1,py-h,blk); + outlines->add(x1,py-h,x2,py-h,blk); + outlines->add(x1,py,x2,py,blk); + outlines->add(x2,py,x2,py-h,blk); } // if (bar py-=h; } else if (m_graphtype==GT_LINE) { // if (m_graphtype==GT_BAR + col.setAlpha(128); short px2=px+barw; short py2=top+height-1-h; if (lastdaygood) { @@ -295,7 +297,7 @@ void SummaryChart::paint(gGraph & w,int left, int top, int width, int height) lines->scissor(left,w.flipY(top+height+2),width+1,height+1); // Draw Ledgend - px=left+width-10; + px=left+width; py=top+10; QString a; int x,y; diff --git a/Graphs/gGraphView.cpp b/Graphs/gGraphView.cpp index 57861a4a..1ef83d8d 100644 --- a/Graphs/gGraphView.cpp +++ b/Graphs/gGraphView.cpp @@ -675,6 +675,13 @@ void gGraphView::queGraph(gGraph * g,int left, int top, int width, int height) m_drawlist.push_back(g); dl_mutex.unlock(); } +void gGraphView::TrashGraphs() +{ + for (int i=0;isetGroup(group); m_graphs.push_back(g); - updateScrollBar(); + // updateScrollBar(); } } float gGraphView::totalHeight() @@ -1499,13 +1506,15 @@ void gGraphView::paintGL() queGraph(m_graphs[i],px,py,width()-titleWidth,h); - // draw the splitter handle - QColor ca=QColor(128,128,128,255); - backlines->add(0, py+h, w, py+h, ca); - ca=QColor(192,192,192,255); - backlines->add(0, py+h+1, w, py+h+1, ca); - ca=QColor(90,90,90,255); - backlines->add(0, py+h+2, w, py+h+2, ca); + if (m_showsplitter) { + // draw the splitter handle + QColor ca=QColor(128,128,128,255); + backlines->add(0, py+h, w, py+h, ca); + ca=QColor(192,192,192,255); + backlines->add(0, py+h+1, w, py+h+1, ca); + ca=QColor(90,90,90,255); + backlines->add(0, py+h+2, w, py+h+2, ca); + } } py=ceil(py+h+graphSpacer); diff --git a/Graphs/gGraphView.h b/Graphs/gGraphView.h index 77e36a60..dc1b2ae2 100644 --- a/Graphs/gGraphView.h +++ b/Graphs/gGraphView.h @@ -363,9 +363,11 @@ public: bool useThreads() { return m_idealthreads>1; } GLBuffer * lines, * backlines, *quads; + void TrashGraphs(); gGraph * popGraph(); QVector m_threads; - + void hideSplitter() { m_showsplitter=false; } + void showSplitter() { m_showsplitter=true; } protected: int m_idealthreads; Day * m_day; @@ -418,6 +420,7 @@ protected: //volatile int m_threadsrunning; QString m_emptytext; + bool m_showsplitter; signals: diff --git a/SleepLib/loader_plugins/prs1_loader.cpp b/SleepLib/loader_plugins/prs1_loader.cpp index 9191367f..9f052f23 100644 --- a/SleepLib/loader_plugins/prs1_loader.cpp +++ b/SleepLib/loader_plugins/prs1_loader.cpp @@ -343,7 +343,7 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile) sess->avg(a[i]); sess->wavg(a[i]); sess->p90(a[i]); */ - sess->cph(a[i]); + //sess->cph(a[i]); } } diff --git a/SleepyHeadQT.pro b/SleepyHeadQT.pro index a142c4de..9b0c2c35 100644 --- a/SleepyHeadQT.pro +++ b/SleepyHeadQT.pro @@ -53,7 +53,8 @@ SOURCES += main.cpp\ qextserialport/qextserialport.cpp \ preferencesdialog.cpp \ Graphs/gGraphView.cpp \ - Graphs/gStatsLine.cpp + Graphs/gStatsLine.cpp \ + report.cpp unix:SOURCES += qextserialport/posix_qextserialport.cpp unix:!macx:SOURCES += qextserialport/qextserialenumerator_unix.cpp @@ -103,7 +104,8 @@ HEADERS += \ qextserialport/qextserialenumerator.h \ preferencesdialog.h \ Graphs/gGraphView.h \ - Graphs/gStatsLine.h + Graphs/gStatsLine.h \ + report.h FORMS += \ @@ -111,7 +113,8 @@ FORMS += \ overview.ui \ mainwindow.ui \ oximetry.ui \ - preferencesdialog.ui + preferencesdialog.ui \ + report.ui RESOURCES += \ Resources.qrc diff --git a/mainwindow.cpp b/mainwindow.cpp index 7946dbb4..7f888666 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -137,6 +137,7 @@ MainWindow::MainWindow(QWidget *parent) : if (!pref.Exists("AlwaysShowOverlayBars")) pref["AlwaysShowOverlayBars"]=true; ui->actionOverlay_Bars->setChecked(pref["AlwaysShowOverlayBars"].toBool()); + ui->tabWidget->setCurrentWidget(ui->welcome); netmanager = new QNetworkAccessManager(this); @@ -157,6 +158,10 @@ MainWindow::~MainWindow() oximetry->close(); delete oximetry; } + if (report) { + report->close(); + delete report; + } DoneGraphs(); Profiles::Done(); mainwin=NULL; @@ -182,11 +187,16 @@ void MainWindow::Startup() oximetry=new Oximetry(ui->tabWidget,daily->SharedWidget()); ui->tabWidget->insertTab(3,oximetry,tr("Oximetry")); + report=new Report(ui->tabWidget,daily->SharedWidget(),daily,overview); + ui->tabWidget->insertTab(4,report,tr("Overview Report")); + + if (daily) daily->ReloadGraphs(); if (overview) { overview->ReloadGraphs(); } + //if (report) report->Reload(); qprogress->hide(); qstatus->setText(""); diff --git a/mainwindow.h b/mainwindow.h index ebd5994f..3a6fc6e3 100644 --- a/mainwindow.h +++ b/mainwindow.h @@ -14,6 +14,7 @@ #include "daily.h" #include "overview.h" #include "oximetry.h" +#include "report.h" const int major_version=0; const int minor_version=8; @@ -26,6 +27,7 @@ namespace Ui { extern QStatusBar *qstatusbar; class Daily; +class Report; class MainWindow : public QMainWindow { Q_OBJECT @@ -96,6 +98,7 @@ private: Daily * daily; Overview * overview; Oximetry * oximetry; + Report * report; bool first_load; Profile *profile; QNetworkAccessManager *netmanager; diff --git a/overview.cpp b/overview.cpp index a9892a0c..94bf110d 100644 --- a/overview.cpp +++ b/overview.cpp @@ -83,8 +83,8 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : pr=new SummaryChart(profile,"cmH2O",GT_LINE); - pr->addSlice(CPAP_Pressure,QColor("orange"),ST_MIN); pr->addSlice(CPAP_Pressure,QColor("dark green"),ST_WAVG); + pr->addSlice(CPAP_Pressure,QColor("orange"),ST_MIN); pr->addSlice(CPAP_Pressure,QColor("red"),ST_MAX); pr->addSlice(CPAP_EPAP,QColor("light green"),ST_MIN); pr->addSlice(CPAP_IPAP,QColor("light blue"),ST_MAX); @@ -97,8 +97,8 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : PR->AddLayer(new gXGrid()); lk=new SummaryChart(profile,"Avg Leak",GT_LINE); - lk->addSlice(CPAP_Leak,QColor("dark blue"),ST_WAVG); lk->addSlice(CPAP_Leak,QColor("dark grey"),ST_90P); + lk->addSlice(CPAP_Leak,QColor("dark blue"),ST_WAVG); //lk->addSlice(CPAP_Leak,QColor("dark yellow")); //pr->addSlice(CPAP_IPAP,QColor("red")); LK->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); diff --git a/overview.h b/overview.h index 2a0c72d9..5491b81e 100644 --- a/overview.h +++ b/overview.h @@ -30,6 +30,8 @@ public: void ReloadGraphs(); //void UpdateGraphs(); + gGraph *AHI,*UC,*PR,*LK; + SummaryChart *bc,*uc,*pr,*lk; private slots: /* void on_drStart_dateChanged(const QDate &date); @@ -51,11 +53,6 @@ private: void UpdateHTML(); //SessionTimes *session_times; - gGraph *AHI,*UC,*PR,*LK; - SummaryChart *bc; - SummaryChart *uc; - SummaryChart *pr; - SummaryChart *lk; //,*PRESSURE,*LEAK,*SESSTIMES; //Layer *prmax,*prmin,*iap,*eap,*pr,*sesstime; diff --git a/preferencesdialog.cpp b/preferencesdialog.cpp index 548d0845..678ebb1c 100644 --- a/preferencesdialog.cpp +++ b/preferencesdialog.cpp @@ -26,6 +26,7 @@ PreferencesDialog::PreferencesDialog(QWidget *parent) : ui->heightEdit->setValue((*profile)["Height"].toDouble(&ok)); ui->dobEdit->setDate((*profile)["DOB"].toDate()); int i=ui->unitCombo->findText((*profile)["UnitSystem"].toString()); + if (i<0) i=0; ui->unitCombo->setCurrentIndex(i); i=ui->timeZoneCombo->findText((*profile)["TimeZone"].toString()); diff --git a/report.cpp b/report.cpp new file mode 100644 index 00000000..6878a503 --- /dev/null +++ b/report.cpp @@ -0,0 +1,232 @@ +#include "report.h" +#include "ui_report.h" +#include +#include +#include +#include +#include +#include +#include + +Report::Report(QWidget *parent, gGraphView * shared, Daily * daily, Overview * overview) : + QWidget(parent), + ui(new Ui::Report), + m_daily(daily), + m_overview(overview) +{ + ui->setupUi(this); + QString prof=pref["Profile"].toString(); + profile=Profiles::Get(prof); + if (!profile) { + QMessageBox::critical(this,"Profile Error",QString("Couldn't get profile '%1'.. Have to abort!").arg(pref["Profile"].toString())); + exit(-1); + } + GraphView=new gGraphView(this); + + //GraphView->AddGraph(overview->AHI); + GraphView->hide(); + ui->startDate->setDate(profile->FirstDay()); + ui->endDate->setDate(profile->LastDay()); + + // Create a new graph, but reuse the layers.. + int default_height=200; + UC=new gGraph(GraphView,"Usage",default_height,0); + /*uc=new SummaryChart(profile,"Hours",GT_BAR); + uc->addSlice(EmptyChannel,QColor("green"),ST_HOURS); */ + UC->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); + gXAxis *gx=new gXAxis(); + gx->setUtcFix(true); + UC->AddLayer(gx,LayerBottom,0,gXAxis::Margin); + UC->AddLayer(m_overview->uc); + UC->AddLayer(new gXGrid()); + + AHI=new gGraph(GraphView,"AHI",default_height,0); + + /* bc=new SummaryChart(profile,"AHI",GT_BAR); + bc->addSlice(CPAP_Hypopnea,QColor("blue"),ST_CPH); + bc->addSlice(CPAP_Apnea,QColor("dark green"),ST_CPH); + bc->addSlice(CPAP_Obstructive,QColor("#40c0ff"),ST_CPH); + bc->addSlice(CPAP_ClearAirway,QColor("purple"),ST_CPH);*/ + AHI->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); + gx=new gXAxis(); + gx->setUtcFix(true); + AHI->AddLayer(gx,LayerBottom,0,gXAxis::Margin); + AHI->AddLayer(m_overview->bc); + AHI->AddLayer(new gXGrid()); + + PR=new gGraph(GraphView,"Pressure",default_height,0); + /*pr=new SummaryChart(profile,"cmH2O",GT_LINE); + pr->addSlice(CPAP_Pressure,QColor("orange"),ST_MIN); + pr->addSlice(CPAP_Pressure,QColor("red"),ST_MAX); + pr->addSlice(CPAP_EPAP,QColor("light green"),ST_MIN); + pr->addSlice(CPAP_IPAP,QColor("light blue"),ST_MAX); */ + PR->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); + gx=new gXAxis(); + gx->setUtcFix(true); + PR->AddLayer(gx,LayerBottom,0,gXAxis::Margin); + PR->AddLayer(m_overview->pr); + PR->AddLayer(new gXGrid()); + + LK=new gGraph(GraphView,"Leaks",default_height,0); + /*lk=new SummaryChart(profile,"Avg Leak",GT_LINE); + lk->addSlice(CPAP_Leak,QColor("dark blue"),ST_WAVG); + lk->addSlice(CPAP_Leak,QColor("dark grey"),ST_90P); */ + //lk->addSlice(CPAP_Leak,QColor("dark yellow")); + //pr->addSlice(CPAP_IPAP,QColor("red")); + LK->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); + gx=new gXAxis(); + gx->setUtcFix(true); + LK->AddLayer(gx,LayerBottom,0,gXAxis::Margin); + LK->AddLayer(m_overview->lk); + LK->AddLayer(new gXGrid()); + + GraphView->hideSplitter(); + //ui->webView->hide(); + m_ready=false; +// Reload(); +} + +Report::~Report() +{ + delete ui; +} +void Report::showEvent (QShowEvent * event) +{ + QTimer::singleShot(0,this,SLOT(on_refreshButton_clicked())); +} + +void Report::resizeEvent(QResizeEvent *event) +{ + QWidget::resizeEvent(event); + GraphView->setMinimumSize(event->size().width()-20,240); + GraphView->setMaximumSize(event->size().width()-20,240); + Reload(); +} + +QPixmap Report::Snapshot(gGraph * graph) +{ + QDateTime d1(ui->startDate->date(),QTime(0,0,0),Qt::UTC); + qint64 first=qint64(d1.toTime_t())*1000L; + QDateTime d2(ui->endDate->date(),QTime(0,0,0),Qt::UTC); + qint64 last=qint64(d2.toTime_t())*1000L; + + GraphView->TrashGraphs(); + GraphView->AddGraph(graph); + GraphView->ResetBounds(); + GraphView->SetXBounds(first,last); + + int w=this->width()-20; + QPixmap pixmap=GraphView->renderPixmap(w,240,false); //gwwidth,gwheight,false); + + + return pixmap; +} + +void Report::Reload() +{ + if (!m_ready) return; + + //UC->ResetBounds(); + QString html="" + "" + "" + "" + "
"; + html+="

CPAP Overview

"; + html+="
"; + + //html+="This is a temporary scratch pad tab so I can see what's going on while designing printing code. These graphs are images, and not controllable."; + if (!((*profile).Exists("FirstName") && (*profile).Exists("LastName"))) html+="

Please edit your profile (in Preferences)

"; else { + html+=""; + } + if ((*profile).Exists("Address")&& !(*profile)["Address"].toString().isEmpty()) { + QString address=(*profile)["Address"].toString().replace("\n","
"); + html+=""; + } + if ((*profile).Exists("Phone") && !(*profile)["Phone"].toString().isEmpty()) { + html+=""; + } + if ((*profile).Exists("EmailAddress") && !(*profile)["EmailAddress"].toString().isEmpty()) { + html+=""; + } + html+="
Name:"+(*profile)["FirstName"].toString()+" "+(*profile)["LastName"].toString()+"
Address:"+address+"
Phone:"+(*profile)["Phone"].toString()+"
Email:"+(*profile)["EmailAddress"].toString()+"
"; + if ((*profile).Exists("Gender")) { + QString gender=(*profile)["Gender"].toBool() ? "Male" : "Female"; + html+=""; + } + if ((*profile).Exists("DOB") && !(*profile)["DOB"].toString().isEmpty()) { + QDate dob=(*profile)["DOB"].toDate(); + //html+=""; + QDateTime d1(dob,QTime(0,0,0)); + QDateTime d2(QDate::currentDate(),QTime(0,0,0)); + int years=d1.daysTo(d2)/365.25; + html+=""; + + } + if ((*profile).Exists("Height") && !(*profile)["Height"].toString().isEmpty()) { + html+=""; + } + + html+="
Gender:"+gender+"
D.O.B.:"+dob.toString()+"
Age:"+QString::number(years)+" years
Height:"+(*profile)["Height"].toString(); + if (!(*profile).Exists("UnitSystem")) { + (*profile)["UnitSystem"]="Metric"; + } + if ((*profile)["UnitSystem"].toString()=="Metric") html+="cm"; else html+="inches"; + html+="
"; + html+="

SleepyHead v"+pref["VersionString"].toString()+"
 
" + "Reporting from "+ui->startDate->date().toString()+" to "+ui->endDate->date().toString()+"
" + "
"; + + + QVector graphs; + graphs.push_back(AHI); + graphs.push_back(UC); + graphs.push_back(PR); + graphs.push_back(LK); + + for (int i=0;i\n"; // + } + + html+=""; + ui->webView->setHtml(html); +} + +void Report::on_refreshButton_clicked() +{ + m_ready=true; + Reload(); +} + +void Report::on_startDate_dateChanged(const QDate &date) +{ + Reload(); +} + +void Report::on_endDate_dateChanged(const QDate &date) +{ + Reload(); +} + +void Report::on_printButton_clicked() +{ + QPrinter printer; + QPrintDialog *dialog = new QPrintDialog(&printer); + //printer.setPrinterName("Print to File (PDF)"); + //printer.setOutputFormat(QPrinter::PdfFormat); + printer.setPrintRange(QPrinter::AllPages); + printer.setOrientation(QPrinter::Portrait); + printer.setPaperSize(QPrinter::A4); + printer.setResolution(QPrinter::HighResolution); + printer.setFullPage(false); + printer.setNumCopies(1); + //printer.setOutputFileName("printYou.pdf"); + if ( dialog->exec() == QDialog::Accepted) { + ui->webView->print(&printer); + } +} diff --git a/report.h b/report.h new file mode 100644 index 00000000..379d3181 --- /dev/null +++ b/report.h @@ -0,0 +1,48 @@ +#ifndef REPORT_H +#define REPORT_H + +#include +#include "SleepLib/profiles.h" +#include "Graphs/gGraphView.h" +#include "daily.h" +#include "overview.h" + +namespace Ui { + class Report; +} + +class Daily; +class Report : public QWidget +{ + Q_OBJECT + +public: + explicit Report(QWidget *parent, gGraphView * shared, Daily * daily, Overview * overview); + ~Report(); + void Reload(); + QPixmap Snapshot(gGraph * graph); +protected: + virtual void showEvent (QShowEvent * event); +private slots: + void on_refreshButton_clicked(); + + void on_startDate_dateChanged(const QDate &date); + + void on_endDate_dateChanged(const QDate &date); + + void on_printButton_clicked(); + +private: + Ui::Report *ui; + Profile * profile; + Daily * m_daily; + Overview * m_overview; + gGraphView * shared; + gGraphView * GraphView; + gGraph *AHI,*UC,*PR,*LK; + SummaryChart *bc,*uc,*pr,*lk; + bool m_ready; + virtual void resizeEvent(QResizeEvent *); +}; + +#endif // REPORT_H diff --git a/report.ui b/report.ui new file mode 100644 index 00000000..5079680d --- /dev/null +++ b/report.ui @@ -0,0 +1,115 @@ + + + Report + + + + 0 + 0 + 549 + 338 + + + + Form + + + + 0 + + + 0 + + + + + + about:blank + + + + + + + + + + Start + + + + + + + d/MM/yyyy + + + true + + + + + + + End + + + + + + + d/MM/yyyy + + + true + + + + + + + &Refresh + + + + + + + Temporary Page + + + + + + + Qt::Horizontal + + + + 40 + 20 + + + + + + + + &Print + + + + + + + + + + QWebView + QWidget +
QtWebKit/QWebView
+
+
+ + +