diff --git a/Resources.qrc b/Resources.qrc index 09eb9209..3d845e87 100644 --- a/Resources.qrc +++ b/Resources.qrc @@ -15,5 +15,6 @@ docs/usage.html icons/oximeter.png docs/0.0.gif + docs/template_overview.sht diff --git a/SleepyHeadQT.pro b/SleepyHeadQT.pro index 9b0c2c35..cfe5dbac 100644 --- a/SleepyHeadQT.pro +++ b/SleepyHeadQT.pro @@ -121,4 +121,5 @@ RESOURCES += \ OTHER_FILES += \ docs/index.html \ - docs/usage.html + docs/usage.html \ + docs/template_overview.sht diff --git a/docs/template_overview.sht b/docs/template_overview.sht new file mode 100644 index 00000000..11e6287f --- /dev/null +++ b/docs/template_overview.sht @@ -0,0 +1,50 @@ + + + + + + + + + + + + +

CPAP Overview

+ + + + + +
+ + + + + +
Name:{{Profile.FirstName}} {{Profile.LastName}}
Address:{{Profile.Address}}
Phone:{{Profile.Phone}}
Email:{{Profile.EmailAddress}}
+ + + + +
Gender:{{Profile.Gender}}
Age:{{local.Age}} years
Height:{{Profile.Height}}{{local.DistanceMeasure}}
+
+
+ +
SleepyHead v{{pref.VersionString}} +
http://sleepyhead.sf.net +
+ Reporting from {{local.start}} to {{local.end}} +
+

+{{graph.AHI}} +{{graph.Usage}} +{{graph.Leaks}} +{{graph.Pressure}} +{{graph.%PB}} + + diff --git a/preferencesdialog.cpp b/preferencesdialog.cpp index 7fcd5c11..d92beacf 100644 --- a/preferencesdialog.cpp +++ b/preferencesdialog.cpp @@ -17,12 +17,16 @@ PreferencesDialog::PreferencesDialog(QWidget *parent,Profile * _profile) : ui->addressEdit->appendPlainText((*profile)["Address"].toString()); ui->emailEdit->setText((*profile)["EmailAddress"].toString()); ui->phoneEdit->setText((*profile)["Phone"].toString()); - bool gender; - if ((*profile).Exists("Gender")) { - gender=(*profile)["Gender"].toBool(); - } else gender=true; // the true gender.. How sexist.. ;) + QString gender; + + if ((*profile).Exists("Gender")) { + gender=(*profile)["Gender"].toString().toLower(); + } else gender="male"; + + // I know this looks sexist.. This was originally a boolean.. :) + if ((gender=="male") || (gender=="true")) ui->genderMale->setChecked(true); + else if ((gender=="female") || (gender=="false")) ui->genderFemale->setChecked(true); - if (gender) ui->genderMale->setChecked(true); else ui->genderFemale->setChecked(true); bool ok; ui->heightEdit->setValue((*profile)["Height"].toDouble(&ok)); @@ -148,7 +152,9 @@ void PreferencesDialog::Save() { (*profile)["FirstName"]=ui->firstNameEdit->text(); (*profile)["LastName"]=ui->lastNameEdit->text(); - (*profile)["Gender"]=ui->genderMale->isChecked(); + if (ui->genderMale->isChecked()) + (*profile)["Gender"]="male"; + else (*profile)["Gender"]="female"; (*profile)["Height"]=ui->heightEdit->value(); (*profile)["DOB"]=ui->dobEdit->date(); (*profile)["EmailAddress"]=ui->emailEdit->text(); diff --git a/report.cpp b/report.cpp index 858b075b..fc3d7042 100644 --- a/report.cpp +++ b/report.cpp @@ -4,9 +4,11 @@ #include #include #include -#include +//#include #include #include +#include +#include Report::Report(QWidget *parent, Profile * _profile, gGraphView * shared, Overview * overview) : QWidget(parent), @@ -24,64 +26,57 @@ Report::Report(QWidget *parent, Profile * _profile, gGraphView * shared, Overvie GraphView->setMaximumSize(graph_print_width,graph_print_height); GraphView->setMinimumSize(graph_print_width,graph_print_height); - GraphView->hide(); - // Reusing the layer data from overview screen, // (Can't reuse the graphs objects without breaking things) - UC=new gGraph(GraphView,"Usage",graph_print_height,0); + graphs["Usage"]=UC=new gGraph(GraphView,"Usage",graph_print_height,0); UC->AddLayer(m_overview->uc); - AHI=new gGraph(GraphView,"AHI",graph_print_height,0); + graphs["AHI"]=AHI=new gGraph(GraphView,"AHI",graph_print_height,0); AHI->AddLayer(m_overview->bc); - PR=new gGraph(GraphView,"Pressure",graph_print_height,0); + graphs["Pressure"]=PR=new gGraph(GraphView,"Pressure",graph_print_height,0); PR->AddLayer(m_overview->pr); - LK=new gGraph(GraphView,"Leaks",graph_print_height,0); + graphs["Leaks"]=LK=new gGraph(GraphView,"Leaks",graph_print_height,0); LK->AddLayer(m_overview->lk); - NPB=new gGraph(GraphView,"% in PB",graph_print_height,0); + graphs["%PB"]=NPB=new gGraph(GraphView,"% in PB",graph_print_height,0); NPB->AddLayer(m_overview->npb); - graphs.push_back(AHI); - graphs.push_back(UC); - graphs.push_back(PR); - graphs.push_back(LK); - graphs.push_back(NPB); - gXAxis *gx; - for (int i=0;iAddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); - gx=new gXAxis(); + for (QHash::iterator g=graphs.begin();g!=graphs.end();g++) { + gGraph *gr=g.value(); + gr->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); + gXAxis *gx=new gXAxis(); gx->setUtcFix(true); - graphs[i]->AddLayer(gx,LayerBottom,0,gXAxis::Margin); - graphs[i]->AddLayer(new gXGrid()); + gr->AddLayer(gx,LayerBottom,0,gXAxis::Margin); + gr->AddLayer(new gXGrid()); } - GraphView->hideSplitter(); - //ui->webView->hide(); m_ready=false; - ReloadGraphs(); -// Reload(); } Report::~Report() { + for (QHash::iterator g=graphs.begin();g!=graphs.end();g++) { + delete g.value(); + } delete ui; } void Report::ReloadGraphs() { - for (int i=0;isetDay(NULL); + + for (QHash::iterator g=graphs.begin();g!=graphs.end();g++) { + g.value()->setDay(NULL); } startDate=profile->FirstDay(); endDate=profile->LastDay(); - for (int i=0;iResetBounds(); + for (QHash::iterator g=graphs.begin();g!=graphs.end();g++) { + g.value()->ResetBounds(); } m_ready=true; @@ -104,83 +99,109 @@ QPixmap Report::Snapshot(gGraph * graph) return pixmap; } +QString Report::ParseTemplate(QString input) +{ + QString output; + + QRegExp rx("\\{\\{(.*)\\}\\}"); + rx.setMinimal(true); + int lastpos=0,pos=0; + + while ((pos=rx.indexIn(input,pos))!=-1) { + output+=input.mid(lastpos,pos-lastpos); + QString block=rx.cap(1); + + QString code=block.section(".",0,0).toLower(); + QString key=block.section(".",1,-1); + QHash * pr=NULL; + if (code=="profile") { + pr=&profile->p_preferences; + } else if (code=="pref") { + pr=&pref.p_preferences; + } else if (code=="local") { + pr=&locals; + } + + QString value; + if (pr) { + if (pr->contains(key)) { + if ((*pr)[key].type()==QVariant::String){ + value=(*pr)[key].toString(); + value.replace("\n","
"); + output+=value; + } else if ((*pr)[key].type()==QVariant::Double){ + bool ok; + value=QString::number((*pr)[key].toDouble(&ok),'f',2); + if (ok) output+=value; else output+="[NaN]"; + } else if ((*pr)[key].type()==QVariant::Int){ + bool ok; + value=QString::number((*pr)[key].toInt(&ok)); + if (ok) output+=value; else output+="[NaN]"; + } else if ((*pr)[key].type()==QVariant::Date){ + value=(*pr)[key].toDate().toString(); + output+=value; + } else { + qDebug() << "Unknown key type" << (*pr)[key].typeName() << " in " << code << "." << key << "in template"; + } + } else { + qDebug() << "Key not found" << code << "." << key << "in template"; + } + } else if (code=="graph") { + if (graphs.contains(key)) { + if (!graphs[key]->isEmpty()) { + QPixmap pixmap=Snapshot(graphs[key]); + QByteArray byteArray; + QBuffer buffer(&byteArray); // use buffer to store pixmap into byteArray + buffer.open(QIODevice::WriteOnly); + pixmap.save(&buffer, "PNG"); + //html += "
\n"; // + output += "
\n"; + } + + } else { + qDebug() << "Graph not found" << key << "in template"; + } + } + pos+=rx.matchedLength(); + lastpos=pos; + } + output+=input.mid(lastpos); // will just return the input if no tags are used + return output; + +} + void Report::GenerateReport(QDate start, QDate end) { if (!m_ready) return; startDate=start; endDate=end; - //UC->ResetBounds(); - QString html="" - "" - "" - "" - "" - "" - "

CPAP Overview

" - "" - "" - "
" - ""; - if (!((*profile).Exists("FirstName") && (*profile).Exists("LastName"))) html+="

Please edit your profile

"; 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+=""; - } + locals["start"]=startDate; + locals["end"]=endDate; + locals["width"]=graph_print_width-10; + 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+=""; - + locals["Age"]=years; + } + if (!(*profile).Exists("UnitSystem")) { + (*profile)["UnitSystem"]="Metric"; } if ((*profile).Exists("Height") && !(*profile)["Height"].toString().isEmpty()) { - html+=""; + if ((*profile)["UnitSystem"].toString()=="Metric") + locals["DistanceMeasure"]="cm"; + else locals["DistanceMeasure"]="inches"; } + QFile file(":/docs/template_overview.sht"); + file.open(QIODevice::ReadOnly); + QString html=file.readAll(); - 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+="
" - "
" - "
SleepyHead v"+pref["VersionString"].toString()+ - "
http://sleepyhead.sf.net
" - "Reporting from "+startDate.toString()+" to "+endDate.toString()+"" - "
" - "
 
"; + QString output=ParseTemplate(html); - - - for (int i=0;iisEmpty()) continue; - QPixmap pixmap=Snapshot(graphs[i]); - QByteArray byteArray; - QBuffer buffer(&byteArray); // use buffer to store pixmap into byteArray - buffer.open(QIODevice::WriteOnly); - pixmap.save(&buffer, "PNG"); - //html += "
\n"; // - html += "
\n"; // - } - - html+=""; - ui->webView->setHtml(html); + ui->webView->setHtml(output); } void Report::Print() diff --git a/report.h b/report.h index 061fe100..09455904 100644 --- a/report.h +++ b/report.h @@ -25,6 +25,8 @@ public: ~Report(); void GenerateReport(QDate start, QDate end); void ReloadGraphs(); + QString ParseTemplate(QString input); + QPixmap Snapshot(gGraph * graph); void Print(); @@ -36,7 +38,8 @@ private: gGraphView * GraphView; gGraph *AHI,*UC,*PR,*LK,*NPB; SummaryChart *bc,*uc,*pr,*lk,*npb; - QVector graphs; + QHash locals; + QHash graphs; QDate startDate; QDate endDate;