From 1e466ca6bf67a9717ead829048cbc357f041bd7e Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Tue, 10 Sep 2013 01:56:02 +1000 Subject: [PATCH] Translation preparation work, Oximetry time selection queries --- Graphs/gGraphView.cpp | 2 + SleepLib/common.h | 162 +- SleepyHeadQT.pro | 10 +- Translations/sleepyhead_nl.ts | 4277 +++++++++++++++++++++++++++++++++ daily.cpp | 158 +- docs/release_notes.html | 28 +- exportcsv.cpp | 2 +- main.cpp | 38 +- mainwindow.cpp | 255 +- overview.cpp | 52 +- oximetry.cpp | 107 +- oximetry.h | 5 +- preferencesdialog.cpp | 6 +- quazip/quazipfile.cpp | 2 +- 14 files changed, 4796 insertions(+), 308 deletions(-) create mode 100644 Translations/sleepyhead_nl.ts diff --git a/Graphs/gGraphView.cpp b/Graphs/gGraphView.cpp index 12209ef7..96f27ba6 100644 --- a/Graphs/gGraphView.cpp +++ b/Graphs/gGraphView.cpp @@ -2277,6 +2277,8 @@ gGraphView::~gGraphView() } delete masterlock; #endif + + // Note: This will cause a crash if two graphs accidentally have the same name for (int i=0;i #include +#include "version.h" #if QT_VERSION >= QT_VERSION_CHECK(4,8,0) @@ -37,18 +38,9 @@ QString weightString(float kg, UnitSystem us=US_Undefined); //! \brief Mercilessly trash a directory bool removeDir(const QString & path); - -const QString STR_UNIT_CM=QObject::tr("cm"); -const QString STR_UNIT_INCH=QObject::tr("\""); -const QString STR_UNIT_FOOT=QObject::tr("ft"); -const QString STR_UNIT_POUND=QObject::tr("lb"); -const QString STR_UNIT_OUNCE=QObject::tr("oz"); -const QString STR_UNIT_KG=QObject::tr("Kg"); -const QString STR_UNIT_CMH2O=QObject::tr("cmH2O"); -const QString STR_UNIT_Hours=QObject::tr("Hours"); - -const QString STR_MESSAGE_ERROR=QObject::tr("Error"); -const QString STR_MESSAGE_WARNING=QObject::tr("Warning"); +/////////////////////////////////////////////////////////////////////////////////////////////// +// Preference Name Strings +/////////////////////////////////////////////////////////////////////////////////////////////// const QString STR_GEN_Profile="Profile"; const QString STR_GEN_SkipLogin="SkipLoginScreen"; @@ -81,18 +73,154 @@ const QString STR_MACH_MSeries="MSeries"; const QString STR_MACH_CMS50="CMS50"; const QString STR_MACH_ZEO="Zeo"; -const QString STR_TR_BMI=QObject::tr("BMI"); +const QString STR_PREF_VersionString="VersionString"; + +/////////////////////////////////////////////////////////////////////////////////////////////// +// Commonly used translatable text strings +/////////////////////////////////////////////////////////////////////////////////////////////// + +const QString STR_UNIT_CM=QObject::tr("cm"); +const QString STR_UNIT_INCH=QObject::tr("\""); +const QString STR_UNIT_FOOT=QObject::tr("ft"); +const QString STR_UNIT_POUND=QObject::tr("lb"); +const QString STR_UNIT_OUNCE=QObject::tr("oz"); +const QString STR_UNIT_KG=QObject::tr("Kg"); +const QString STR_UNIT_CMH2O=QObject::tr("cmH2O"); +const QString STR_UNIT_Hours=QObject::tr("Hours"); + +const QString STR_UNIT_BPM=QObject::tr("bpm"); // Beats per Minute +const QString STR_UNIT_LPM=QObject::tr("L/m"); // Litres per Minute + +const QString STR_MESSAGE_ERROR=QObject::tr("Error"); +const QString STR_MESSAGE_WARNING=QObject::tr("Warning"); + +const QString STR_TR_BMI=QObject::tr("BMI"); // Short form of Body Mass Index const QString STR_TR_Weight=QObject::tr("Weight"); -const QString STR_TR_PulseRate=QObject::tr("Pulse Rate"); +const QString STR_TR_Zombie=QObject::tr("Zombie"); +const QString STR_TR_PulseRate=QObject::tr("Pulse Rate"); // Pulse / Heart rate const QString STR_TR_SpO2=QObject::tr("SpO2"); -const QString STR_TR_Plethy=QObject::tr("Plethy"); -const QString STR_TR_FlowRate=QObject::tr("Flow Rate"); +const QString STR_TR_Plethy=QObject::tr("Plethy"); // Plethysomogram const QString STR_TR_Pressure=QObject::tr("Pressure"); const QString STR_TR_Daily=QObject::tr("Daily"); const QString STR_TR_Overview=QObject::tr("Overview"); const QString STR_TR_Oximetry=QObject::tr("Oximetry"); - +const QString STR_TR_Oximeter=QObject::tr("Oximeter"); const QString STR_TR_EventFlags=QObject::tr("Event Flags"); +// Machine type names. +const QString STR_TR_CPAP=QObject::tr("CPAP"); // Constant Positive Airway Pressure +const QString STR_TR_BIPAP=QObject::tr("BiPAP"); // Bi-Level Positive Airway Pressure +const QString STR_TR_BiLevel=QObject::tr("Bi-Level"); // Another name for BiPAP +const QString STR_TR_EPAP=QObject::tr("EPAP"); // Expiratory Positive Airway Pressure +const QString STR_TR_IPAP=QObject::tr("IPAP"); // Inspiratory Positive Airway Pressure +const QString STR_TR_IPAPLo=QObject::tr("IPAPLo"); // Inspiratory Positive Airway Pressure, Low +const QString STR_TR_IPAPHi=QObject::tr("IPAPHi"); // Inspiratory Positive Airway Pressure, High +const QString STR_TR_APAP=QObject::tr("APAP"); // Automatic Positive Airway Pressure +const QString STR_TR_ASV=QObject::tr("ASV"); // Assisted Servo Ventilator +const QString STR_TR_STASV=QObject::tr("ST/ASV"); + +const QString STR_TR_Humidifier=QObject::tr("Humidifier"); + +const QString STR_TR_H=QObject::tr("H"); // Short form of Hypopnea +const QString STR_TR_OA=QObject::tr("OA"); // Short form of Obstructive Apnea +const QString STR_TR_UA=QObject::tr("A"); // Short form of Unspecified Apnea +const QString STR_TR_CA=QObject::tr("CA"); // Short form of Clear Airway Apnea +const QString STR_TR_FL=QObject::tr("FL"); // Short form of Flow Limitation +const QString STR_TR_LE=QObject::tr("LE"); // Short form of Leak Event +const QString STR_TR_EP=QObject::tr("EP"); // Short form of Expiratory Puff +const QString STR_TR_VS=QObject::tr("VS"); // Short form of Vibratory Snore +const QString STR_TR_VS2=QObject::tr("VS2"); // Short form of Secondary Vibratory Snore (Some Philips Respironics Machines have two sources) +const QString STR_TR_RERA=QObject::tr("RERA"); // Acronym for Respiratory Effort Related Arousal +const QString STR_TR_PP=QObject::tr("PP"); // Short form for Pressure Pulse +const QString STR_TR_RE=QObject::tr("RE"); // Short form of Respiratory Effort Related Arousal +const QString STR_TR_NRI=QObject::tr("NRI"); // Sorry I Forgot.. it's a flag on Intellipap machines +const QString STR_TR_O2=QObject::tr("O2"); // SpO2 Desaturation +const QString STR_TR_PC=QObject::tr("PC"); // Short form for Pulse Change + + + +const QString STR_TR_PS=QObject::tr("PS"); // Short form of Pressure Support +const QString STR_TR_AHI=QObject::tr("AHI"); // Short form of Apnea Hypopnea Index +const QString STR_TR_RDI=QObject::tr("RDI"); // Short form of Respiratory Distress Index +const QString STR_TR_AI=QObject::tr("AI"); // Short form of Apnea Index +const QString STR_TR_HI=QObject::tr("HI"); // Short form of Hypopnea Index +const QString STR_TR_UAI=QObject::tr("UAI"); // Short form of Uncatagorized Apnea Index +const QString STR_TR_CAI=QObject::tr("CAI"); // Short form of Clear Airway Index +const QString STR_TR_FLI=QObject::tr("FLI"); // Short form of Flow Limitation Index +const QString STR_TR_REI=QObject::tr("REI"); // Short form of RERA Index +const QString STR_TR_EPI=QObject::tr("EPI"); // Short form of Expiratory Puff Index +const QString STR_TR_CSR=QObject::tr("ÇSR"); // Short form of Cheyne Stokes Respiration +const QString STR_TR_PB=QObject::tr("PB"); // Short form of Periodic Breathing + + +// Graph Titles +const QString STR_TR_IE=QObject::tr("IE"); // Inspiratory Expiratory Ratio +const QString STR_TR_InspTime=QObject::tr("Insp. Time"); // Inspiratory Time +const QString STR_TR_ExpTime=QObject::tr("Exp. Time"); // Expiratory Time +const QString STR_TR_RespEvent=QObject::tr("Resp. Event"); // Respiratory Event +const QString STR_TR_FlowLimitation=QObject::tr("Flow Limitation"); +const QString STR_TR_FlowLimit=QObject::tr("Flow Limit"); +const QString STR_TR_PatTrigBreath=QObject::tr("Pat. Trig. Breath"); // Patient Triggered Breath +const QString STR_TR_TgtMinVent=QObject::tr("Tgt. Min. Vent"); // Target Minute Ventilation +const QString STR_TR_TargetVent=QObject::tr("Target Vent."); // Target Ventilation +const QString STR_TR_MinuteVent=QObject::tr("Minute Vent."); // Minute Ventilation +const QString STR_TR_TidalVolume=QObject::tr("Tidal Volume"); +const QString STR_TR_RespRate=QObject::tr("Resp. Rate"); // Respiratory Rate +const QString STR_TR_Snore=QObject::tr("Snore"); +const QString STR_TR_Leak=QObject::tr("Leak"); +const QString STR_TR_Leaks=QObject::tr("Leaks"); +const QString STR_TR_TotalLeaks=QObject::tr("Total Leaks"); +const QString STR_TR_UnintentionalLeaks=QObject::tr("Unintentional Leaks"); +const QString STR_TR_MaskPressure=QObject::tr("MaskPressure"); +const QString STR_TR_FlowRate=QObject::tr("Flow Rate"); +const QString STR_TR_SleepStage=QObject::tr("Sleep Stage"); +const QString STR_TR_Usage=QObject::tr("Usage"); +const QString STR_TR_Sessions=QObject::tr("Sessions"); +const QString STR_TR_PrRelief=QObject::tr("Pr. Relief"); // Pressure Relief + +const QString STR_TR_NoData=QObject::tr("No Data"); +const QString STR_TR_Bookmarks=QObject::tr("Bookmarks"); +const QString STR_TR_SleepyHead=QObject::tr("SleepyHead"); +const QString STR_TR_SleepyHeadVersion=STR_TR_SleepyHead+" v"+VersionString; + +const QString STR_TR_Mode=QObject::tr("Mode"); +const QString STR_TR_Model=QObject::tr("Model"); +const QString STR_TR_Brand=QObject::tr("Brand"); +const QString STR_TR_Serial=QObject::tr("Serial"); +const QString STR_TR_Machine=QObject::tr("Machine"); +const QString STR_TR_Channel=QObject::tr("Channel"); +const QString STR_TR_Settings=QObject::tr("Settings"); + +const QString STR_TR_Name=QObject::tr("Name"); +const QString STR_TR_DOB=QObject::tr("DOB"); // Date of Birth +const QString STR_TR_Phone=QObject::tr("Phone"); +const QString STR_TR_Address=QObject::tr("Address"); +const QString STR_TR_Email=QObject::tr("Email"); +const QString STR_TR_PatientID=QObject::tr("Patient ID"); +const QString STR_TR_Date=QObject::tr("Date"); + +const QString STR_TR_BedTime=QObject::tr("Bedtime"); +const QString STR_TR_WakeUp=QObject::tr("Wake-up"); +const QString STR_TR_MaskTime=QObject::tr("Mask Time"); +const QString STR_TR_Unknown=QObject::tr("Unknown"); +const QString STR_TR_None=QObject::tr("None"); + +const QString STR_TR_First=QObject::tr("First"); +const QString STR_TR_Last=QObject::tr("Last"); +const QString STR_TR_Start=QObject::tr("Start"); +const QString STR_TR_End=QObject::tr("End"); +const QString STR_TR_On=QObject::tr("On"); +const QString STR_TR_Off=QObject::tr("Off"); + +const QString STR_TR_Min=QObject::tr("Min"); // Minimum +const QString STR_TR_Max=QObject::tr("Max"); // Maximum + +const QString STR_TR_Average=QObject::tr("Average"); +const QString STR_TR_Median=QObject::tr("Median"); +const QString STR_TR_Avg=QObject::tr("Avg"); // Average +const QString STR_TR_WAvg=QObject::tr("W-Avg"); // Weighted Average + + + #endif // COMMON_H diff --git a/SleepyHeadQT.pro b/SleepyHeadQT.pro index f4c2c88e..a80a10ee 100644 --- a/SleepyHeadQT.pro +++ b/SleepyHeadQT.pro @@ -183,6 +183,10 @@ FORMS += \ exportcsv.ui \ UpdaterWindow.ui +TRANSLATIONS += \ + Translations/sleepyhead_nl.ts \ + Translations/sleepyhead_fr.ts + RESOURCES += \ Resources.qrc @@ -203,7 +207,11 @@ OTHER_FILES += \ docs/changelog.txt \ docs/update_notes.html - +mac { + TransFiles.files = Translations + TransFiles.path = Contents/MacOS + QMAKE_BUNDLE_DATA += TransFiles +} diff --git a/Translations/sleepyhead_nl.ts b/Translations/sleepyhead_nl.ts new file mode 100644 index 00000000..311aee54 --- /dev/null +++ b/Translations/sleepyhead_nl.ts @@ -0,0 +1,4277 @@ + + + + + Daily + + + Form + + + + + Go to the previous day + + + + + Prev + + + + + Show or hide the calender + + + + + + ... + + + + + Go to the next day + + + + + Next + + + + + Go to the most recent day with data records + + + + + Details + + + + + about:blank + + + + + Events + + + + + View Size + + + + + + Notes + + + + + Journal + + + + + Small + + + + + Medium + + + + + Big + + + + + Color + + + + + i + + + + + Ctrl+I + + + + + u + + + + + B + + + + + Ctrl+B + + + + + Zombie + + + + + I'm feeling... + + + + + Weight + + + + + Awesome + + + + + B.M.I. + + + + + Bookmarks + + + + + Add Bookmark + + + + + Starts + + + + + Remove Bookmark + + + + + Zoom fully out + + + + + 100% + + + + + Reset the graph heights to uniform sizes + + + + + Reset + + + + + Drop down this list to show/hide available graphs. + + + + + Flow Rate + + + + + + RDI + + + + + + AHI + + + + + Mask Pressure + + + + + Pressure + + + + + + Leak + + + + + Snore + + + + + Resp. Rate + + + + + Tidal Volume + + + + + Minute Vent. + + + + + Flow Limitation + + + + + Pat. Trig. Breath + + + + + Resp. Event + + + + + Insp. Time + + + + + Exp. Time + + + + + IE + + + + + + Sleep Stage + + + + + Breakdown + + + + + events + + + + + + + H + + + + + + + A + + + + + + + OA + + + + + + + CA + + + + + + + + RE + + + + + NR + + + + + + + FL + + + + + PB + + + + + E + + + + + L + + + + + + NRI + + + + + + VS + + + + + VS2 + + + + + UF1 + + + + + UF2 + + + + + UF3 + + + + + Selection AHI + + + + + CSR + + + + + PR + + + + + U1 + + + + + U2 + + + + + U3 + + + + + + O2 + + + + + + Events/hour + + + + + PD + + + + + No Data + + + + + No %1 events are recorded this day + + + + + %1 event + + + + + %1 events + + + + + Oximetry data exists for this day, however it's timestamps are too different, so the Graphs will not be linked. + + + + + No Graphs :( + + + + + + CPAP + + + + + APAP + + + + + Bi-Level + + + + + ASV + + + + + Unknown + + + + + + Date + + + + + Sleep + + + + + Wake + + + + + Hypopnea + + + + + Apnea + + + + + Obstructive + + + + + + Flow Limit + + + + + Clear Airway + + + + + User Flags + + + + + RERA + + + + + + VSnore + + + + + VSnore2 + + + + + PB/CSR + + + + + Exh&nbsp;Puff + + + + + Event Breakdown + + + + + Sessions all off! + + + + + Sessions exist for this day but are switched off. + + + + + Impossibly short session + + + + + Zero hours?? + + + + + BRICK :( + + + + + Sorry, your machine does not record data. + + + + + Complain to your Equipment Provider! + + + + + + Avg + + + + + Med + + + + + Statistics + + + + + Channel + + + + + Min + + + + + %1% + + + + + Max + + + + + <b>Please Note:</b> This day just contains summary data, only limited information is available . + + + + + No data available + + + + + Oximeter Information + + + + + SpO2 Desaturations + + + + + Pulse Change events + + + + + SpO2 Baseline Used + + + + + Machine Settings + + + + + Pr. Relief + + + + + Humidifier + + + + + Session Information + + + + + SessionID + + + + + On + + + + + Start + + + + + End + + + + + CPAP Sessions + + + + + Oximetry Sessions + + + + + Oximeter + + + + + Sleep Stage Sessions + + + + + One or more waveform record for this session had faulty source data. Some waveform overlay points may not match up correctly. + + + + + Pick a Colour + + + + + Show all graphs + + + + + No Graphs On! + + + + + Hide all graphs + + + + + ExportCSV + + + Export as CSV + + + + + Dates: + + + + + Resolution: + + + + + Details + + + + + Sessions + + + + + Daily + + + + + Filename: + + + + + Cancel + + + + + Export + + + + + Start: + + + + + End: + + + + + Quick Range: + + + + + + + Most Recent Day + + + + + + Last Week + + + + + + Last Fortnight + + + + + + Last Month + + + + + + Last 6 Months + + + + + + Last Year + + + + + + Everything + + + + + + Custom + + + + + ... + + + + + SleepyHead_ + + + + + Details_ + + + + + Sessions_ + + + + + Summary_ + + + + + Select file to export to + + + + + CSV Files (*.csv) + + + + + DateTime + + + + + + Session + + + + + Event + + + + + Data/Duration + + + + + + Date + + + + + Session Count + + + + + + Start + + + + + + End + + + + + + Total Time + + + + + + AHI + + + + + Count + + + + + Avg + + + + + %1% + + + + + MainWindow + + + + SleepyHead + SlaapKop + + + + &Statistics + + + + + + + about:blank + + + + + &Help Browser + + + + + + + + ... + + + + + qrc:/docs/index.html + + + + + &Navigation + + + + + Statistics + + + + + Daily + Dagelijks + + + + Overview + Overzicht + + + + Oximetry + + + + + Import + + + + + Help + + + + + &Bookmarks + &Bladwijzer + + + + &Records + &Archief + + + + &File + + + + + &View + + + + + &Help + + + + + &Data + &Gegevens + + + + &Advanced + + + + + &Purge CPAP Data + + + + + &Import Data + + + + + Shift+F2 + + + + + &Preferences + &Voorkeuren + + + + &Profiles + &Profiel + + + + E&xit + &Uitgaan + + + + View &Daily + + + + + F5 + + + + + View &Overview + + + + + F6 + + + + + View &Welcome + + + + + + F4 + + + + + - + + + + + Ctrl+Tab + + + + + Use &AntiAliasing + + + + + &About SleepyHead + &Over SlaapKop + + + + &Fullscreen Toggle + + + + + F11 + + + + + Show Debug Pane + + + + + &Reset Graph Layout + + + + + Check for &Updates + + + + + Take &Screenshot + + + + + F12 + + + + + View O&ximetry + + + + + F7 + + + + + Print &Report + + + + + &Edit Profile + + + + + &Link Graph Groups + + + + + Exp&ort + + + + + Online Users &Guide + + + + + &Frequently Asked Questions + + + + + &Rebuild Oximetry Indices + + + + + Change &User + + + + + &Current Selected Day + + + + + All data for current CPAP machine + + + + + Right &Sidebar + + + + + F8 + + + + + View S&ummary + + + + + Import &ZEO Data + + + + + Import RemStar &MSeries Data + + + + + &Support Sleepyhead Development + + + + + Sleep Disorder Terms &Glossary + + + + + Loading Data + + + + + Importing Data + + + + + RDI + + + + + AHI + + + + + No CPAP data available. + + + + + %1 day of CPAP Data, on %2. + + + + + %1 days of CPAP Data, between %2 and %3 + + + + + + Details + + + + + + Most Recent + + + + + + Last 7 Days + + + + + + Last 30 Days + + + + + + Last 6 months + + + + + + Last Year + + + + + RERA Index + + + + + Flow Limit Index + + + + + Hours per Night + + + + + Min EPAP + + + + + %1% EPAP + + + + + Max IPAP + + + + + %1% IPAP + + + + + Average Pressure + + + + + %1% Pressure + + + + + + Pressure + + + + + Average %1 + + + + + %1% %2 + + + + + Oximetry Summary + + + + + %1 day of Oximetry Data, on %2. + + + + + %1 days of Oximetry Data, between %2 and %3 + + + + + Average SpO2 + + + + + Minimum SpO2 + + + + + SpO2 Events / Hour + + + + + % of time in SpO2 Events + + + + + Average Pulse Rate + + + + + Minimum Pulse Rate + + + + + Maximum Pulse Rate + + + + + Pulse Change Events / Hour + + + + + Usage Information + + + + + Total Days + Totaal dagen + + + + Compliant Days + Naleving dagen + + + + Days AHI &gt;5.0 + + + + + Best&nbsp;%1 + + + + + Worst&nbsp;%1 + Slechtst&nbsp;%1 + + + + + + CPAP + + + + + + + APAP + + + + + + + Bi-Level + + + + + + ST/ASV + + + + + Best RX Setting + + + + + + + Mode + + + + + Worst RX Setting + + + + + + EPAP + + + + + IPAPLo + + + + + IPAPHi + + + + + PS Min + + + + + PS Max + + + + + IPAP + + + + + PS + + + + + Min Pres. + + + + + Max Pres. + + + + + First + + + + + Last + + + + + Days + + + + + FL + + + + + Machine + + + + + Pr. Rel. + Drukontlastings + + + + + %5 %1% EPAP=%2<br/>%3% IPAP=%4 + + + + + %3 %1% Pressure=%2 + + + + + Brand + Merk + + + + Model + Ser + + + + Serial + Volgnummer + + + + First Use + + + + + Last Use + + + + + Loading + Het Laden + + + + <html><body><div align='center'><h2>SleepyHead v%1.%2.%3-%4 (%8)</h2>Build Date: %5 %6<br/>%7<br/>Data Folder: %9<hr>Copyright &copy;2011 Mark Watkins (jedimark) <br> +<a href='http://sleepyhead.sourceforge.net'>http://sleepyhead.sourceforge.net</a> <hr>This software is released under the GNU Public License <br><i>This software comes with absolutely no warranty, either express of implied. It comes with no guarantee of fitness for any particular purpose. No guarantees are made regarding the accuracy of any data this program displays.</div></body></html> + + + + + About SleepyHead + Over SlaapKop + + + + There are no graphs visible to print + + + + + Bookmarks + Bladwijzers + + + + Would you like to show bookmarked areas in this report? + + + + + This make take some time to complete.. +Please don't touch anything until it's done. + + + + + Printing %1 Report + + + + + %1 Report + + + + + Name: %1, %2 + + + + + + DOB: %1 + + + + + + Patient ID: %1 + + + + + + Phone: %1 + + + + + + Email: %1 + + + + + + +Address: +%1 + + + + + Mask Time: + + + + + Bedtime: + + + + + Wake-up: + + + + + Machine: + + + + + +Mode: + + + + + ASV + + + + + Unknown + + + + + RDI %1 + + + + + + AHI %1 + + + + + + AI=%1 HI=%2 CAI=%3 + + + + + REI=%1 VSI=%2 FLI=%3 PB/CSR=%4% + + + + + UAI=%1 + + + + + NRI=%1 LKI=%2 EPI=%3 + + + + + Weight %1 + + + + + BMI %1 + + + + + Zombie %1/10 + + + + + Reporting from %1 to %2 + + + + + Reporting data goes here + + + + + Entire Day's Flow Waveform + + + + + SleepyHead v%1 - http://sleepyhead.sourceforge.net + + + + + Page %1 of %2 + + + + + + Gah! + + + + + + If you can read this, the restart command didn't work. Your going to have to do it yourself manually. + + + + + Are you sure? + + + + + Are you sure you want to purge all CPAP data for the following machine: + + + + + + Loading Event Data + + + + + + Recalculating Summaries + + + + + NewProfile + + + Edit User Profile + + + + + Language + Taal + + + + English + Engels + + + + Data Folder + + + + + Shows the directory where SleepyHead data will be stored. + + + + + Click here to choose where to store SleepyHead data. + + + + + ... + + + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans'; font-size:10pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600; font-style:italic;">Welcome to SleepyHead</span></p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">This software is being designed to help you review data related to your CPAP treatment.</p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">It's intended as merely a data viewer, and not a substitute for competent medical guidance from your Doctor. </p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">This software has been released freely under the <a href="qrc:/LICENSE.txt"><span style=" text-decoration: underline; color:#0000ff;">GNU Public License</span></a>, and comes with no warranty, and without ANY claims to fitness for any purpose.</p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Accuracy of any data displayed is not and can not be guaranteed. </p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><br />Any reports generated are for PERSONAL USE ONLY, and not fit for compliance purposes.</p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">The author will not be held liable for <span style=" text-decoration: underline;">anything</span> related to the use or misuse of this software. </p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Use at your own risk. </p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">This software is copyright ©2011 Mark Watkins </p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p></body></html> + + + + + I agree to all the conditions above. + + + + + User Information + Gebruikersinformatie + + + + User Name + Gebruikersnaam + + + + Keep the kids out.. Nothing more.. This isn't meant to be uber security. + + + + + Password Protect Profile + + + + + Password + + + + + ...twice... + + + + + Locale Settings + + + + + Country + Land + + + + TimeZone + + + + + DST Zone + + + + + Personal Information (for reports) + + + + + First Name + Voornaam + + + + Last Name + Achternaam + + + + D.O.B. + + + + + Gender + Geslacht + + + + Male + Man + + + + Female + Vrouw + + + + Height + Lengte + + + + metric + + + + + archiac + + + + + Contact Information + + + + + + Address + + + + + + Email + + + + + + Phone + + + + + CPAP Treatment Information + + + + + Date Diagnosed + + + + + Untreated AHI + + + + + CPAP Mode + + + + + CPAP + + + + + APAP + + + + + Bi-Level + + + + + ASV + + + + + RX Pressure + + + + + Doctors / Clinic Information + + + + + Doctors Name + + + + + Practice Name + + + + + Patient ID + + + + + SleepyHead + + + + + TextLabel + + + + + &Cancel + + + + + &Back + + + + + + + &Next + + + + + Select Country + + + + + Empty Username + + + + + Passwords don't match + + + + + Profile Changes + + + + + Accept and save this information? + + + + + &Finish + + + + + Overview + + + Form + + + + + Range: + + + + + Last Week + + + + + Last Two Weeks + + + + + Last Month + + + + + Last Two Months + + + + + Last Three Months + + + + + Last 6 Months + + + + + Last Year + + + + + Everything + + + + + Custom + + + + + Start: + + + + + End: + + + + + Reset view to selected date range + + + + + + ... + + + + + Toggle Graph Visibility + + + + + Drop down to see list of graphs to switch on/off. + + + + + Graphs + + + + + + RDI + + + + + + AHI + + + + + Apnea +Hypopnea +Index + + + + + Usage + + + + + Usage +(hours) + + + + + Flow Limit + + + + + Session Times + + + + + Session Times +(hours) + + + + + Pressure +(cmH2O) + + + + + Settings + + + + + + Leaks + + + + + Unintentional Leaks +(L/min) + + + + + + Total Leaks + + + + + Total Leaks +(L/min) + + + + + % in PB + + + + + Periodic +Breathing +(% of night) + + + + + Peak RDI + + + + + Peak RDI +Shows RDI Clusters +(RDI/hr) + + + + + Peak AHI + + + + + Peak AHI +Shows AHI Clusters +(AHI/hr) + + + + + Resp. Rate + + + + + Respiratory +Rate +(breaths/min) + + + + + Tidal Volume + + + + + Tidal +Volume +(ml) + + + + + Minute Vent. + + + + + Minute +Ventilation +(L/min) + + + + + Target Vent. + + + + + Target +Ventilation +(L/min) + + + + + Pat. Trig. Br. + + + + + Patient +Triggered +Breaths +(%) + + + + + + Sessions + + + + + Sessions +(count) + + + + + + Pulse Rate + + + + + Pulse Rate +(bpm) + + + + + Oxygen Saturation +(%) + + + + + Body +Mass +Index + + + + + Zombie + + + + + How you felt +(0-10) + + + + + Events/Hr + + + + + Zombie Meter + + + + + FL + + + + + breaths/min + + + + + L/b + + + + + + L/m + + + + + %PTB + + + + + % PB + + + + + Show all graphs + + + + + No Graphs On! + + + + + Hide all graphs + + + + + Oximetry + + + Form + + + + + Date + + + + + d/MM/yy h:mm:ss AP + + + + + R&eset + + + + + SpO2 + + + + + Pulse + + + + + ... + + + + + &Open .spo/R File + + + + + Serial &Import + + + + + &Start Live + + + + + Serial Port + + + + + &Rescan Ports + + + + + Control + + + + + No Oximetry Data + + + + + Oximetry live recording has been terminated due to timeout. + + + + + + &Start + + + + + Save Session? + + + + + Creating a new oximetry session will destroy the old one. +Would you like to save it first? + + + + + Save + + + + + Destroy It + + + + + Cancel + + + + + Please Wait + + + + + Oximetry Error! + +Something is wrong with the device connection. + + + + + &Stop + + + + + Oximeter Error + +The device has not responded.. Make sure it's switched on. + + + + + + Ready + + + + + Check Oximeter is Ready + + + + + Oximeter Error + +The device did not respond.. Make sure it's switched on. + + + + + Please make sure your oximeter is switched on, and in the right mode to transmit data. + + + + + Oximeter Error! + + + + + PreferencesDialog + + + Preferences + + + + + &Import + + + + + Session Settings + + + + + Combine Close Sessions + + + + + + Minutes + + + + + Multiple sessions closer together than this value will be kept on the same day. + + + + + + Ignore Short Sessions + + + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Cantarell'; font-size:11pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Sessions shorter in duration than this will not be displayed<span style=" font-style:italic;">.</span></p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-style:italic;"></p></body></html> + + + + + Day Split Time + + + + + Sessions starting before this time will go to the previous calendar day. + + + + + Keep session data in memory to speed up revisiting days. + + + + + Cache Session Data (uses more system memory) + + + + + Session Storage Options + + + + + This maintains a backup of SD-card data for ResMed machines, + +ResMed machines delete high resolution data older than 7 days, +and graph data older than 30 days.. + +Sleepyhead can keep a copy of this data if you ever need to reinstall. +(Highly recomended, unless your short on disk space or don't care about the graph data) + + + + + Create SD Card Backups during Import (only for ResMed so far, highly recommended) + + + + + This makes SleepyHead's data take around half as much space. +But it makes import and day changing take longer.. +If you've got a new computer with a small solid state disk, this is a good option. + + + + + Compress Session Data (makes SleepyHead data smaller, but day changing slower.) + + + + + Compress ResMed (EDF) backups to save disk space. +Backed up EDF files are stored in the .gz format, +which is common on Mac & Linux platforms.. + +SleepyHead can import from this compressed backup directory natively.. +To use with ResScan will require the .gz files to be uncompressed first.. + + + + + Compress SD Card Backups (slower first import, but makes backups smaller) + + + + + The following options affect the amount of disk space SleepyHead uses, and all have an effect on how long import takes. + + + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans'; font-size:10pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-style:italic;">Changing SD Backup compression options doesn't automatically recompress backup data. </span></p></body></html> + + + + + &CPAP + + + + + CPAP Mask Information + + + + + Mask Type + + + + + Generic mask type. Select the one that's closest to your mask. + + + + + Description + + + + + The name of your mask, or at least the name you call it. + + + + + Method of unintentional leaks calculation if not provided by your machine. +Note: Statistical Model is experimental. + + + + + Mask Profile + + + + + Statistical Model + + + + + Leak calcs + + + + + Started Using + + + + + The date you started using this mask + + + + + Leak Profile + + + + + Pressure + + + + + Leak + + + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans'; font-size:10pt; font-weight:400; font-style:italic;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600; font-style:normal;">Note: </span>Leak profiles currently does not work yet..</p></body></html> + + + + + Shows Respiratory Disturbance Index instead of Apnea/Hypopnea Index (RDI=AHI + RERA) + + + + + Use RDI instead of AHI (PRS1 only) + + + + + Don't show any compliance information + + + + + Show Compliance + + + + + Regard days with under this usage as "incompliant". 4 hours is usually considered compliant. + + + + + hours + + + + + as over + + + + + of usage per night + + + + + Enable/disable experimental event flagging enhancements. +It allows detecting borderline events, and some the machine missed. +This option must be enabled before import, otherwise a purge is required. + + + + + Custom User Event Flagging + + + + + Flow Restriction + + + + + Percentage of restriction in airflow from the median value. +A value of 20% works well for detecting apneas. + + + + + + % + + + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans'; font-size:10pt; font-weight:400; font-style:italic;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Custom flagging is an experimental method of detecting events missed by the machine. They are <span style=" text-decoration: underline;">not</span> included in AHI.</p></body></html> + + + + + Duration of airflow restriction + + + + + + + + s + + + + + Event Duration + + + + + Allow duplicates near machine events. + + + + + AHI/Hour Graph Settings + + + + + Window + + + + + Adjusts the amount of data considered for each point in the AHI/Hour graph. +Defaults to 60 minutes.. Highly recommend it's left at this value. + + + + + minutes + + + + + Reset the counter to zero at beginning of each (time) window. + + + + + Zero Reset + + + + + CPAP Clock Drift + + + + + Don't touch this unless you know your CPAP clock is out. +Try to sync it to your PC's clock (which should be synced to a timeserver) + + + + + seconds + + + + + &Events + + + + + Not entirely sure if this will get to live or not.. + + + + + Show + + + + + Colour + + + + + Event + + + + + ID + + + + + Graphs + + + + + Search + + + + + Filters the graph list. Simply start typing the name of the graph your looking for. + + + + + &Defaults + + + + + Double click on the (Y-axis) min/max values to edit them + + + + + &Oximetry + + + + + Use Oximetry + + + + + Type + + + + + Contec CMS50 + + + + + Overpriced ResMed S9 Oximeter + + + + + Tries to forces the oximetry data to link with CPAP when possible. + + + + + Link Oximetry and CPAP graphs + + + + + Flag changes in oximetry stats + + + + + SPO2 + + + + + Percentage drop in oxygen saturation + + + + + Pulse + + + + + Sudden change in Pulse Rate of at least this amount + + + + + bpm + + + + + Minimum duration of drop in oxygen saturation + + + + + Minimum duration of pulse change event. + + + + + Discard chunks under + + + + + Small chunks of oximetry data under this amount will be discarded. + + + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans'; font-size:10pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Syncing Oximetry and CPAP Data</span></p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">CMS50 data imported from SpO2Review (from .spoR files) or the serial import method does <span style=" font-weight:600; text-decoration: underline;">not</span> have the correct timestamp needed to sync.</p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Live view mode (using a serial cable) is one way to acheive an accurate sync on CMS50 oximeters, but does not counter for CPAP clock drift.</p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">If you start your Oximeters recording mode at <span style=" font-style:italic;">exactly </span>the same time you start your CPAP machine, you can now also achieve sync. </p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">The serial import process takes the starting time from last nights first CPAP session. (Remember to import your CPAP data first!)</p></body></html> + + + + + &General + + + + + + General Settings + + + + + Daily view navigation buttons will skip over days without data records + + + + + Skip over Empty Days + + + + + Allow use of multiple CPU cores where available to improve performance. +Mainly affects the importer. + + + + + Enable Multithreading + + + + + Bypass the login screen and load the most recent User Profile + + + + + Skip Login Screen + + + + + Changes to the following settings needs a restart, but not a recalc. + + + + + Preferred Calculation Methods + + + + + Middle Calculations + + + + + Upper Percentile + + + + + For consistancy, ResMed users should use 95% here, +as this is the only value available on summary-only days. + + + + + Median is recommended for ResMed users. + + + + + Median + + + + + Weighted Average + + + + + Normal Average + + + + + ResMed users probably should use 99th Percentile for visual consistency. + + + + + True Maximum + + + + + 99% Percentile + + + + + Maximum Calcs + + + + + Import Locations + + + + + Add + + + + + Remove + + + + + Automatically Check For Updates + + + + + Check for new version every + + + + + Sourceforge hosts this project for free.. Please be considerate of their resources.. + + + + + days. + + + + + &Check for Updates now + + + + + Last Checked For Updates: + + + + + TextLabel + + + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans'; font-size:10pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">If your interested in helping test new features and bugfixes early, click here.</p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-style:italic;">But please be warned this will sometimes mean breaky code..</span></p> +<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p></body></html> + + + + + I want to try experimental and test builds (Advanced users only please.) + + + + + &Appearance + + + + + Overlay Flags + + + + + The visual method of displaying waveform overlay flags. + + + + + + Standard Bars + + + + + Top & Bottom Markers + + + + + Graph Height + + + + + Default display height of graphs in pixels + + + + + How long you want the tooltips to stay visible. + + + + + Tooltip Timeout + + + + + Anti-Aliasing applies smoothing to graph plots.. +Certain plots look more attractive with this on. +This also affects printed reports. + +Try it and see if you like it. + + + + + Use Anti-Aliasing + + + + + Makes certain plots look more "square waved". + + + + + Square Wave Plots + + + + + Allows graphs to be "screenshotted" for display purposes. +The Event Breakdown PIE chart uses this method, as does +the printing code. +Unfortunately some older computers/versions of Qt can cause +this application to be unstable with this feature enabled. + + + + + Show event breakdown pie chart + + + + + Turn on/off the spinning "context" cube. +It really doesn't use that much resources.. :) + + + + + Animations && Fancy Stuff + + + + + Application Fonts + + + + + Font + + + + + Size + + + + + Bold + + + + + Italic + + + + + Application + + + + + Graph Text + + + + + Graph Titles + + + + + Big Text + + + + + Details + + + + + &Cancel + + + + + &Ok + + + + + Nasal Pillows + + + + + Data Reindex Required + + + + + A data reindexing proceedure is required to apply these changes. This operation may take a couple of minutes to complete. + +Are you sure you want to make these changes? + + + + + Restart Required + + + + + One or more of the changes you have made will require this application to be restarted, +in order for these changes to come into effect. + +Would you like do this now? + + + + + Add this Location to the Import List + + + + + Daily Graphs + + + + + Overview Graphs + + + + + Graph + + + + + Min + + + + + Max + + + + + N/A + + + + + Oximetry Graphs + + + + + Confirmation + + + + + Are you sure you want to reset your graph preferences to the defaults? + + + + + ProfileSelect + + + Select Profile + + + + + &Quit + + + + + New Profile + + + + + &Select User + + + + + Open Profile + + + + + Edit Profile + + + + + Delete Profile + + + + + + Enter Password for %1 + + + + + + + Incorrect Password + + + + + You entered the password wrong too many times. + + + + + + + Question + + + + + Are you sure you want to trash the profile "%1"? + + + + + Double Checking: + +Do you really want "%1" profile to be obliterated? + + + + + Okay, I am about to totally OBLITERATE the profile "%1" and all it's contained data.. + +Don't say you weren't warned. :-p + + + + + WTH??? + + + + + If you can read this you need to delete this profile directory manually (It's under %1) + + + + + Meheh... If your trying to delete because you forgot the password, your going the wrong way about it. Read the docs. + +Signed: Nasty Programmer + + + + + Whoops. + + + + + There was an error deleting the profile directory.. You need to manually remove %1 + + + + + Enter Password + + + + + You entered an Incorrect Password too many times. Exiting! + + + + + QObject + + + No Data + + + + + cm + + + + + " + + + + + ft + + + + + lb + + + + + oz + + + + + Kg + + + + + cmH2O + + + + + Hours + + + + + Error + + + + + Warning + + + + + On + + + + + Off + + + + + BMI + + + + + Weight + + + + + Pulse Rate + + + + + SpO2 + + + + + Plethy + + + + + Flow Rate + + + + + Pressure + + + + + Daily + + + + + Overview + + + + + Oximetry + + + + + Event Flags + + + + + Windows User + + + + + Software changes have been made that require the reimporting of the following machines data: + + + + + + + I can automatically purge this data for you, or you can cancel now and continue to run in a previous version. + + + + + + + Would you like me to purge this data this for you so you can run the new version? + + + + + Machine Database Changes + + + + + Purge Failed + + + + + Sorry, I could not purge this data, which means this version of SleepyHead can't start.. SleepyHead's Data folder needs to be removed manually + +This folder currently resides at the following location: + + + + + + SleepyHead Release Notes + + + + + + &Ok, get on with it.. + + + + + SleepyHead Update Notes + + + + + SleepyHead + + + + + Unspecified + + + + + Nasal Pillows + + + + + Hybrid F/F Mask + + + + + Nasal Interface + + + + + Full-Face Mask + + + + + QuaZipFilePrivate + + + ZIP/UNZIP API error %1 + + + + + Report + + + Form + + + + + about:blank + + + + + UpdaterWindow + + + SleepyHead Updater + + + + + A new version of $APP is available + + + + + Version Information + + + + + Release Notes + + + + + about:blank + + + + + Build Notes + + + + + Maybe &Later + + + + + &Upgrade Now + + + + + Please wait while updates are downloaded and installed... + + + + + Updates + + + + + Component + + + + + Version + + + + + Size + + + + + Progress + + + + + Log + + + + + Downloading & Installing Updates + + + + + &Finished + + + + + Checking for SleepyHead Updates + + + + + + Requesting + + + + + Saving as + + + + + XML update structure parsed cleanly + + + + + No updates were found for your platform. + + + + + + SleepyHead Updates + + + + + No new updates were found for your platform. + + + + + SleepyHead v%1, codename "%2" + + + + + platform notes + + + + + A new version of SleepyHead is available! + + + + + Shiny new <b>v%1</b> is available. You're running old and busted v%2 + + + + + An update for SleepyHead is available. + + + + + Version <b>%1</b> is available. You're currently running v%1 + + + + + SleepyHead v%1 build notes + + + + + Update to QtLibs (v%1) + + + + + There was an error parsing the XML Update file. + + + + + %1 bytes received + + + + + Redirected to + + + + + File size mismatch for %1 + + + + + File integrity check failed for %1 + + + + + Extracting + + + + + You might need to reinstall manually. Sorry :( + + + + + Ugh.. Something went wrong with unzipping. + + + + + Failed + + + + + Download Complete + + + + + There was an error completing a network request: + +( + + + + + Update Complete! + + + + + Updates Complete. SleepyHead needs to restart now, click Finished to do so. + + + + + Update Failed :( + + + + + Download Error. Sorry, try again later. + + + + + Downloading & Installing Updates... + + + + + Please wait while downloading and installing updates. + + + + diff --git a/daily.cpp b/daily.cpp index 0a0070d1..6c56fc3a 100644 --- a/daily.cpp +++ b/daily.cpp @@ -91,29 +91,29 @@ Daily::Daily(QWidget *parent,gGraphView * shared) int default_height=PROFILE.appearance->graphHeight(); SF=new gGraph(GraphView,STR_TR_EventFlags,STR_TR_EventFlags,default_height); - FRW=new gGraph(GraphView,tr("Flow Rate"),schema::channel[CPAP_FlowRate].description()+"\n("+schema::channel[CPAP_FlowRate].units()+")",default_height); + FRW=new gGraph(GraphView,STR_TR_FlowRate,schema::channel[CPAP_FlowRate].description()+"\n("+schema::channel[CPAP_FlowRate].units()+")",default_height); if (PROFILE.general->calculateRDI()) { - AHI=new gGraph(GraphView,tr("RDI"),schema::channel[CPAP_RDI].description()+"\n("+schema::channel[CPAP_RDI].units()+")",default_height); - } else AHI=new gGraph(GraphView,tr("AHI"),schema::channel[CPAP_AHI].description()+"\n("+schema::channel[CPAP_AHI].units()+")",default_height); + AHI=new gGraph(GraphView,STR_TR_RDI,schema::channel[CPAP_RDI].description()+"\n("+schema::channel[CPAP_RDI].units()+")",default_height); + } else AHI=new gGraph(GraphView,STR_TR_AHI,schema::channel[CPAP_AHI].description()+"\n("+schema::channel[CPAP_AHI].units()+")",default_height); - MP=new gGraph(GraphView,tr("Mask Pressure"),schema::channel[CPAP_MaskPressure].description()+"\n("+schema::channel[CPAP_MaskPressure].units()+")",default_height); - PRD=new gGraph(GraphView,tr("Pressure"),schema::channel[CPAP_Pressure].description()+"\n("+schema::channel[CPAP_Pressure].units()+")",default_height); - LEAK=new gGraph(GraphView,tr("Leak"),schema::channel[CPAP_Leak].description()+"\n("+schema::channel[CPAP_Leak].units()+")",default_height); - SNORE=new gGraph(GraphView,tr("Snore"),schema::channel[CPAP_Snore].description()+"\n("+schema::channel[CPAP_Snore].units()+")",default_height); - RR=new gGraph(GraphView,tr("Resp. Rate"),schema::channel[CPAP_RespRate].description()+"\n("+schema::channel[CPAP_RespRate].units()+")",default_height); - TV=new gGraph(GraphView,tr("Tidal Volume"),schema::channel[CPAP_TidalVolume].description()+"\n("+schema::channel[CPAP_TidalVolume].units()+")",default_height); - MV=new gGraph(GraphView,tr("Minute Vent."),schema::channel[CPAP_MinuteVent].description()+"\n("+schema::channel[CPAP_MinuteVent].units()+")",default_height); - //TgMV=new gGraph(GraphView,tr("Tgt. Min. Vent"),schema::channel[CPAP_TgMV].description()+"\n("+schema::channel[CPAP_TgMV].units()+")",default_height); - FLG=new gGraph(GraphView,tr("Flow Limitation"),schema::channel[CPAP_FLG].description()+"\n("+schema::channel[CPAP_FLG].units()+")",default_height); - PTB=new gGraph(GraphView,tr("Pat. Trig. Breath"),schema::channel[CPAP_PTB].description()+"\n("+schema::channel[CPAP_PTB].units()+")",default_height); - RE=new gGraph(GraphView,tr("Resp. Event"),schema::channel[CPAP_RespEvent].description()+"\n("+schema::channel[CPAP_RespEvent].units()+")",default_height); - TI=new gGraph(GraphView,tr("Insp. Time"),schema::channel[CPAP_Ti].description()+"\n("+schema::channel[CPAP_Ti].units()+")",default_height); - TE=new gGraph(GraphView,tr("Exp. Time"),schema::channel[CPAP_Te].description()+"\n("+schema::channel[CPAP_Te].units()+")",default_height); - IE=new gGraph(GraphView,tr("IE"),schema::channel[CPAP_IE].description()+"\n("+schema::channel[CPAP_IE].units()+")",default_height); + MP=new gGraph(GraphView,STR_TR_MaskPressure,schema::channel[CPAP_MaskPressure].description()+"\n("+schema::channel[CPAP_MaskPressure].units()+")",default_height); + PRD=new gGraph(GraphView,STR_TR_Pressure,schema::channel[CPAP_Pressure].description()+"\n("+schema::channel[CPAP_Pressure].units()+")",default_height); + LEAK=new gGraph(GraphView,STR_TR_Leak,schema::channel[CPAP_Leak].description()+"\n("+schema::channel[CPAP_Leak].units()+")",default_height); + SNORE=new gGraph(GraphView,STR_TR_Snore,schema::channel[CPAP_Snore].description()+"\n("+schema::channel[CPAP_Snore].units()+")",default_height); + RR=new gGraph(GraphView,STR_TR_RespRate,schema::channel[CPAP_RespRate].description()+"\n("+schema::channel[CPAP_RespRate].units()+")",default_height); + TV=new gGraph(GraphView,STR_TR_TidalVolume,schema::channel[CPAP_TidalVolume].description()+"\n("+schema::channel[CPAP_TidalVolume].units()+")",default_height); + MV=new gGraph(GraphView,STR_TR_MinuteVent,schema::channel[CPAP_MinuteVent].description()+"\n("+schema::channel[CPAP_MinuteVent].units()+")",default_height); + //TgMV=new gGraph(GraphView,STR_TR_TgtMinVent,schema::channel[CPAP_TgMV].description()+"\n("+schema::channel[CPAP_TgMV].units()+")",default_height); + FLG=new gGraph(GraphView,STR_TR_FlowLimitation,schema::channel[CPAP_FLG].description()+"\n("+schema::channel[CPAP_FLG].units()+")",default_height); + PTB=new gGraph(GraphView,STR_TR_PatTrigBreath,schema::channel[CPAP_PTB].description()+"\n("+schema::channel[CPAP_PTB].units()+")",default_height); + RE=new gGraph(GraphView,STR_TR_RespEvent,schema::channel[CPAP_RespEvent].description()+"\n("+schema::channel[CPAP_RespEvent].units()+")",default_height); + TI=new gGraph(GraphView,STR_TR_InspTime,schema::channel[CPAP_Ti].description()+"\n("+schema::channel[CPAP_Ti].units()+")",default_height); + TE=new gGraph(GraphView,STR_TR_ExpTime,schema::channel[CPAP_Te].description()+"\n("+schema::channel[CPAP_Te].units()+")",default_height); + IE=new gGraph(GraphView,STR_TR_IE,schema::channel[CPAP_IE].description()+"\n("+schema::channel[CPAP_IE].units()+")",default_height); - STAGE=new gGraph(GraphView,tr("Sleep Stage"),schema::channel[ZEO_SleepStage].description()+"\n("+schema::channel[ZEO_SleepStage].units()+")",default_height); + STAGE=new gGraph(GraphView,STR_TR_SleepStage,schema::channel[ZEO_SleepStage].description()+"\n("+schema::channel[ZEO_SleepStage].units()+")",default_height); int oxigrp=PROFILE.ExistsAndTrue("SyncOximetry") ? 0 : 1; PULSE=new gGraph(GraphView,STR_TR_PulseRate,schema::channel[OXI_Pulse].description()+"\n("+schema::channel[OXI_Pulse].units()+")",default_height,oxigrp); SPO2=new gGraph(GraphView,STR_TR_SpO2,schema::channel[OXI_SPO2].description()+"\n("+schema::channel[OXI_SPO2].units()+")",default_height,oxigrp); @@ -145,18 +145,18 @@ Daily::Daily(QWidget *parent,gGraphView * shared) gFlagsGroup *fg=new gFlagsGroup(); SF->AddLayer(AddCPAP(fg)); - fg->AddLayer((new gFlagsLine(CPAP_CSR,QColor("light green"),tr("PB"),false,FT_Span))); - fg->AddLayer((new gFlagsLine(CPAP_ClearAirway,QColor("purple"),tr("CA"),false))); - fg->AddLayer((new gFlagsLine(CPAP_Obstructive,QColor("#40c0ff"),tr("OA"),true))); - fg->AddLayer((new gFlagsLine(CPAP_Apnea,QColor("dark green"),tr("A")))); - fg->AddLayer((new gFlagsLine(CPAP_Hypopnea,QColor("blue"),tr("H"),true))); - fg->AddLayer((new gFlagsLine(CPAP_ExP,QColor("dark cyan"),tr("E"),false))); - fg->AddLayer((new gFlagsLine(CPAP_LeakFlag,QColor("dark blue"),tr("L"),false))); - fg->AddLayer((new gFlagsLine(CPAP_NRI,QColor("dark magenta"),tr("NRI"),false))); - fg->AddLayer((new gFlagsLine(CPAP_FlowLimit,QColor("black"),tr("FL")))); - fg->AddLayer((new gFlagsLine(CPAP_RERA,QColor("gold"),tr("RE")))); - fg->AddLayer((new gFlagsLine(CPAP_VSnore,QColor("red"),tr("VS")))); - fg->AddLayer((new gFlagsLine(CPAP_VSnore2,QColor("red"),tr("VS2")))); + fg->AddLayer((new gFlagsLine(CPAP_CSR,QColor("light green"),STR_TR_PB,false,FT_Span))); + fg->AddLayer((new gFlagsLine(CPAP_ClearAirway,QColor("purple"),STR_TR_CA,false))); + fg->AddLayer((new gFlagsLine(CPAP_Obstructive,QColor("#40c0ff"),STR_TR_OA,true))); + fg->AddLayer((new gFlagsLine(CPAP_Apnea,QColor("dark green"),STR_TR_UA))); + fg->AddLayer((new gFlagsLine(CPAP_Hypopnea,QColor("blue"),STR_TR_H,true))); + fg->AddLayer((new gFlagsLine(CPAP_ExP,QColor("dark cyan"),STR_TR_EP,false))); + fg->AddLayer((new gFlagsLine(CPAP_LeakFlag,QColor("dark blue"),STR_TR_LE,false))); + fg->AddLayer((new gFlagsLine(CPAP_NRI,QColor("dark magenta"),STR_TR_NRI,false))); + fg->AddLayer((new gFlagsLine(CPAP_FlowLimit,QColor("black"),STR_TR_FL))); + fg->AddLayer((new gFlagsLine(CPAP_RERA,QColor("gold"),STR_TR_RE))); + fg->AddLayer((new gFlagsLine(CPAP_VSnore,QColor("red"),STR_TR_VS))); + fg->AddLayer((new gFlagsLine(CPAP_VSnore2,QColor("red"),STR_TR_VS2))); if (PROFILE.cpap->userEventFlagging()) { fg->AddLayer((new gFlagsLine(CPAP_UserFlag1,QColor("yellow"),tr("UF1")))); fg->AddLayer((new gFlagsLine(CPAP_UserFlag2,QColor("green"),tr("UF2")))); @@ -175,33 +175,33 @@ Daily::Daily(QWidget *parent,gGraphView * shared) gLineOverlaySummary *los=new gLineOverlaySummary(tr("Selection AHI"),5,-4); AddCPAP(l); FRW->AddLayer(new gXGrid()); - FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_CSR,QColor("light green"),tr("CSR"),FT_Span))); + FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_CSR,QColor("light green"),STR_TR_CSR,FT_Span))); FRW->AddLayer(l); FRW->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); FRW->AddLayer(new gXAxis(),LayerBottom,0,20); - FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Hypopnea,QColor("blue"),tr("H"))))); - FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_PressurePulse,QColor("red"),tr("PR"),FT_Dot))); + FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Hypopnea,QColor("blue"),STR_TR_H)))); + FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_PressurePulse,QColor("red"),STR_TR_PP,FT_Dot))); //FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Pressure,QColor("white"),tr("P"),FT_Dot))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0B,QColor("blue"),"0B",FT_Dot))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_10,QColor("orange"),"10",FT_Dot))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0E,QColor("dark red"),"0E",FT_Dot))); if (PROFILE.general->calculateRDI()) - FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_RERA,QColor("gold"),tr("RE"))))); + FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_RERA,QColor("gold"),STR_TR_RE)))); else - FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_RERA,QColor("gold"),tr("RE")))); + FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_RERA,QColor("gold"),STR_TR_RE))); - FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Apnea,QColor("dark green"),tr("A"))))); + FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Apnea,QColor("dark green"),STR_TR_UA)))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_VSnore,QColor("red"),tr("VS")))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_FlowLimit,QColor("black"),tr("FL")))); - FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Obstructive,QColor("#40c0ff"),tr("OA"))))); - FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_ClearAirway,QColor("purple"),tr("CA"))))); + FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Obstructive,QColor("#40c0ff"),STR_TR_OA)))); + FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_ClearAirway,QColor("purple"),STR_TR_CA)))); if (PROFILE.cpap->userEventFlagging()) { FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_UserFlag1,QColor("yellow"),tr("U1"),FT_Bar))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_UserFlag2,QColor("orange"),tr("U2"),FT_Bar))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_UserFlag3,QColor("brown"),tr("U3"),FT_Bar))); } - FRW->AddLayer(AddOXI(new gLineOverlayBar(OXI_SPO2Drop,QColor("red"),tr("O2")))); - //FRW->AddLayer(AddOXI(new gLineOverlayBar(OXI_PulseChange,QColor("blue"),tr("PC"),FT_Dot))); + FRW->AddLayer(AddOXI(new gLineOverlayBar(OXI_SPO2Drop,QColor("red"),STR_TR_O2))); + //FRW->AddLayer(AddOXI(new gLineOverlayBar(OXI_PulseChange,QColor("blue"),STR_TR_PC,FT_Dot))); FRW->AddLayer(AddCPAP(los)); @@ -278,9 +278,9 @@ Daily::Daily(QWidget *parent,gGraphView * shared) gLineOverlaySummary *los1=new gLineOverlaySummary(tr("Events/hour"),5,-4); gLineOverlaySummary *los2=new gLineOverlaySummary(tr("Events/hour"),5,-4); - PULSE->AddLayer(AddOXI(los1->add(new gLineOverlayBar(OXI_PulseChange,QColor("light gray"),tr("PD"),FT_Span)))); + PULSE->AddLayer(AddOXI(los1->add(new gLineOverlayBar(OXI_PulseChange,QColor("light gray"),STR_TR_PC,FT_Span)))); PULSE->AddLayer(AddOXI(los1)); - SPO2->AddLayer(AddOXI(los2->add(new gLineOverlayBar(OXI_SPO2Drop,QColor("light blue"),tr("O2"),FT_Span)))); + SPO2->AddLayer(AddOXI(los2->add(new gLineOverlayBar(OXI_SPO2Drop,QColor("light blue"),STR_TR_O2,FT_Span)))); SPO2->AddLayer(AddOXI(los2)); PULSE->AddLayer(AddOXI(new gLineChart(OXI_Pulse,Qt::red,square))); @@ -333,7 +333,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared) ui->weightSpinBox->setSuffix(STR_UNIT_KG); } GraphView->setCubeImage(images["nodata"]); - GraphView->setEmptyText(tr("No Data")); + GraphView->setEmptyText(STR_TR_NoData); previous_date=QDate(); } @@ -774,16 +774,16 @@ void Daily::Load(QDate date) if (mode==MODE_CPAP) { EventDataType min=round(cpap->settings_wavg(CPAP_Pressure)*2)/2.0; - html+=tr("CPAP")+" "+QString::number(min)+STR_UNIT_CMH2O; + html+=STR_TR_CPAP+" "+QString::number(min)+STR_UNIT_CMH2O; } else if (mode==MODE_APAP) { EventDataType min=cpap->settings_min(CPAP_PressureMin); EventDataType max=cpap->settings_max(CPAP_PressureMax); - html+=tr("APAP")+" "+QString::number(min)+"-"+QString::number(max)+STR_UNIT_CMH2O; + html+=STR_TR_APAP+" "+QString::number(min)+"-"+QString::number(max)+STR_UNIT_CMH2O; } else if (mode==MODE_BIPAP) { EventDataType epap=cpap->settings_min(CPAP_EPAP); EventDataType ipap=cpap->settings_max(CPAP_IPAP); EventDataType ps=cpap->settings_max(CPAP_PS); - html+=tr("Bi-Level")+QString("
EPAP: %1 IPAP: %2 %3
PS: %4") + html+=STR_TR_BiLevel+QString("
"+STR_TR_EPAP+": %1 "+STR_TR_IPAP+": %2 %3
"+STR_TR_PS+": %4") .arg(epap,0,'f',1).arg(ipap,0,'f',1).arg(STR_UNIT_CMH2O).arg(ps,0,'f',1); } else if (mode==MODE_ASV) { @@ -792,7 +792,7 @@ void Daily::Load(QDate date) EventDataType high=cpap->settings_max(CPAP_IPAPHi); EventDataType psl=cpap->settings_min(CPAP_PSMin); EventDataType psh=cpap->settings_max(CPAP_PSMax); - html+=tr("ASV")+QString("
EPAP: %1 IPAP: %2 - %3 %4
PS: %5 / %6") + html+=tr("ASV")+QString("
"+STR_TR_EPAP+": %1 "+STR_TR_IPAP+": %2 - %3 %4
"+STR_TR_PS+": %5 / %6") .arg(epap,0,'f',1) .arg(low,0,'f',1) .arg(high,0,'f',1) @@ -800,12 +800,12 @@ void Daily::Load(QDate date) .arg(psl,0,'f',1) .arg(psh,0,'f',1); } - else html+=tr("Unknown"); + else html+=STR_TR_Unknown; html+="\n"; if (hours>0) { - html+=""+tr("Date")+""+tr("Sleep")+""+tr("Wake")+""+STR_UNIT_Hours+""; + html+=""+STR_TR_Date+""+tr("Sleep")+""+tr("Wake")+""+STR_UNIT_Hours+""; int tt=qint64(cpap->total_time())/1000L; QDateTime date=QDateTime::fromTime_t(cpap->first()/1000L); QDateTime date2=QDateTime::fromTime_t(cpap->last()/1000L); @@ -826,10 +826,10 @@ void Daily::Load(QDate date) if (!isBrick && hours>0) { if (PROFILE.general->calculateRDI()) { html+=QString("%3%4 %5\n") - .arg("#F88017").arg("black").arg(tr("RDI")).arg(schema::channel[CPAP_RDI].description()).arg(ahi,0,'f',2); + .arg("#F88017").arg("black").arg(STR_TR_RDI).arg(schema::channel[CPAP_RDI].description()).arg(ahi,0,'f',2); } else { html+=QString("%3%4 %5\n") - .arg("#F88017").arg("black").arg(tr("AHI")).arg(schema::channel[CPAP_AHI].description()).arg(ahi,0,'f',2); + .arg("#F88017").arg("black").arg(STR_TR_AHI).arg(schema::channel[CPAP_AHI].description()).arg(ahi,0,'f',2); } if (cpap->machine->GetClass()==STR_MACH_ResMed || cpap->machine->GetClass()==STR_MACH_FPIcon) { @@ -857,7 +857,7 @@ void Daily::Load(QDate date) if (cpap->machine->GetClass()==STR_MACH_Intellipap) { html+=QString("%3%4%5%\n") - .arg(schema::channel[CPAP_NRI].defaultColor().name()).arg("black").arg(tr("NRI")).arg(schema::channel[CPAP_NRI].description()).arg(nri,0,'f',2).arg(CPAP_NRI); + .arg(schema::channel[CPAP_NRI].defaultColor().name()).arg("black").arg(STR_TR_NRI).arg(schema::channel[CPAP_NRI].description()).arg(nri,0,'f',2).arg(CPAP_NRI); } if (PROFILE.cpap->userEventFlagging()) { EventDataType uf1=cpap->count(CPAP_UserFlag1) / cpap->hours(); @@ -874,7 +874,7 @@ void Daily::Load(QDate date) if (cpap->machine->GetClass()==STR_MACH_PRS1) { html+=""; html+=QString("\n") - .arg("#ffff80").arg("black").arg(tr("RERA")).arg(schema::channel[CPAP_RERA].description()).arg(rei,0,'f',2).arg(CPAP_RERA); + .arg("#ffff80").arg("black").arg(STR_TR_RERA).arg(schema::channel[CPAP_RERA].description()).arg(rei,0,'f',2).arg(CPAP_RERA); if (mode>MODE_CPAP) { html+=QString("\n") .arg("#404040").arg("white").arg(tr("Flow Limit")).arg(schema::channel[CPAP_FlowLimit].description()).arg(fli,0,'f',2).arg(CPAP_FlowLimit); @@ -893,7 +893,7 @@ void Daily::Load(QDate date) } else if (cpap->machine->GetClass()==STR_MACH_Intellipap) { html+="
%3%4%5
%3%4%5
"; html+=QString("\n") - .arg("#40c0c0").arg("black").arg(tr("Leak")).arg(schema::channel[CPAP_LeakFlag].description()).arg(lki,0,'f',2).arg(CPAP_LeakFlag); + .arg("#40c0c0").arg("black").arg(STR_TR_Leak).arg(schema::channel[CPAP_LeakFlag].description()).arg(lki,0,'f',2).arg(CPAP_LeakFlag); html+=QString("\n") .arg("#ff4040").arg("black").arg(tr("VSnore")).arg(schema::channel[CPAP_VSnore].description()).arg(cpap->count(CPAP_VSnore)/cpap->hours(),0,'f',2).arg(CPAP_VSnore); @@ -970,11 +970,11 @@ void Daily::Load(QDate date) html+=QString("\n").arg(tr("Statistics")); html+="\n"; html+=QString("") - .arg(tr("Channel")) - .arg(tr("Min")) + .arg(STR_TR_Channel) + .arg(STR_TR_Min) .arg(midname) .arg(tr("%1%").arg(percentile*100,0,'f',0)) - .arg(tr("Max")); + .arg(STR_TR_Max); ChannelID chans[]={ CPAP_Pressure,CPAP_EPAP,CPAP_IPAP,CPAP_PS,CPAP_PTB, CPAP_MinuteVent, CPAP_RespRate, CPAP_RespEvent,CPAP_FLG, @@ -1005,24 +1005,24 @@ void Daily::Load(QDate date) med=cpap->percentile(code,0.5); tmp=cpap->wavg(code); if (tmp>0 || mx==0) { - tooltip+=QString("
W-Avg: %1").arg(tmp,0,'f',2); + tooltip+=QString("
"+STR_TR_WAvg+": %1").arg(tmp,0,'f',2); } } else if (ST_mid==ST_WAVG) { med=cpap->wavg(code); tmp=cpap->percentile(code,0.5); if (tmp>0 || mx==0) { - tooltip+=QString("
Median: %1").arg(tmp,0,'f',2); + tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); } } else if (ST_mid==ST_AVG) { med=cpap->avg(code); tmp=cpap->percentile(code,0.5); if (tmp>0 || mx==0) { - tooltip+=QString("
Median: %1").arg(tmp,0,'f',2); + tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); } } html+=QString("") - //.arg(QString("%3%2") //"+tr("RDI")+""+ + //.arg(QString("%3%2") //"+STR_TR_RDI+""+ //.arg(QString::number(code)).arg(tooltip).arg(schema::channel[code].label())) .arg(schema::channel[code].label()) .arg(mn,0,'f',2) @@ -1045,19 +1045,19 @@ void Daily::Load(QDate date) med=oxi->percentile(code,0.5); tmp=oxi->wavg(code); if (tmp>0 || mx==0) { - tooltip+=QString("
W-Avg: %1").arg(tmp,0,'f',2); + tooltip+=QString("
"+STR_TR_WAvg+": %1").arg(tmp,0,'f',2); } } else if (ST_mid==ST_WAVG) { med=oxi->wavg(code); tmp=oxi->percentile(code,0.5); if (tmp>0 || mx==0) { - tooltip+=QString("
Median: %1").arg(tmp,0,'f',2); + tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); } } else if (ST_mid==ST_AVG) { med=oxi->avg(code); tmp=oxi->percentile(code,0.5); if (tmp>0 || mx==0) { - tooltip+=QString("
Median: %1").arg(tmp,0,'f',2); + tooltip+=QString("
"+STR_TR_Median+": %1").arg(tmp,0,'f',2); } } @@ -1105,16 +1105,15 @@ void Daily::Load(QDate date) html+="
"; int i=cpap->settings_max(CPAP_PresReliefType); int j=cpap->settings_max(CPAP_PresReliefSet); - QString flexstr=(i>1) ? schema::channel[CPAP_PresReliefType].option(i)+" x"+QString::number(j) : "None"; + QString flexstr=(i>1) ? schema::channel[CPAP_PresReliefType].option(i)+" x"+QString::number(j) : STR_TR_None; html+=QString("") - .arg(tr("Pr. Relief")) + .arg(STR_TR_PrRelief) .arg(schema::channel[CPAP_PresReliefType].description()) .arg(flexstr); QString mclass=cpap->machine->GetClass(); if (mclass==STR_MACH_PRS1 || mclass==STR_MACH_FPIcon) { int humid=round(cpap->settings_wavg(CPAP_HumidSetting)); - html+=QString("") - .arg(tr("Humidifier")) + html+=QString("") .arg(schema::channel[CPAP_HumidSetting].description()) .arg(humid==0 ? STR_GEN_Off : "x"+QString::number(humid)); } @@ -1125,19 +1124,14 @@ void Daily::Load(QDate date) if (cpap || oxi) { html+="
%3%4%5%
%3%4%5
%1

%1%2%3%4%5
%1%6%2%3%4%5

%1%2%3
%1%2%3
"+STR_TR_Humidifier+"%1%2
"; html+=""; - html+=QString("").arg(tr("Session Information")); + html+=QString(""); html+=""; QDateTime fd,ld; bool corrupted_waveform=false; QString tooltip; - html+=QString("") - .arg(tr("SessionID")) - .arg(tr("On")) - .arg(tr("Date")) - .arg(tr("Start")) - .arg(tr("End")); + html+=QString(""); if (cpap) { - html+=QString("").arg(tr("CPAP Sessions")); + html+=QString(""); for (QVector::iterator s=cpap->begin();s!=cpap->end();s++) { fd=QDateTime::fromTime_t((*s)->first()/1000L); ld=QDateTime::fromTime_t((*s)->last()/1000L); @@ -1145,7 +1139,7 @@ void Daily::Load(QDate date) int h=len/3600; int m=(len/60) % 60; int s1=len % 60; - tooltip=cpap->machine->GetClass()+" "+tr("CPAP")+" "+QString().sprintf("%2ih, %2im, %2is",h,m,s1); + tooltip=cpap->machine->GetClass()+" "+STR_TR_CPAP+" "+QString().sprintf("%2ih, %2im, %2is",h,m,s1); // tooltip needs to lookup language.. :-/ QHash::iterator i=(*s)->settings.find(CPAP_BrokenWaveform); @@ -1167,7 +1161,7 @@ void Daily::Load(QDate date) } if (oxi) { - html+=QString("").arg(tr("Oximetry Sessions")); + html+=QString(""); for (QVector::iterator s=oxi->begin();s!=oxi->end();s++) { fd=QDateTime::fromTime_t((*s)->first()/1000L); ld=QDateTime::fromTime_t((*s)->last()/1000L); @@ -1175,7 +1169,7 @@ void Daily::Load(QDate date) int h=len/3600; int m=(len/60) % 60; int s1=len % 60; - tooltip=oxi->machine->GetClass()+" "+tr("Oximeter")+" "+QString().sprintf("%2ih, %2im, %2is",h,m,s1); + tooltip=oxi->machine->GetClass()+" "+STR_TR_Oximeter+" "+QString().sprintf("%2ih, %2im, %2is",h,m,s1); Session *sess=*s; if (!sess->settings.contains(SESSION_ENABLED)) { @@ -1712,7 +1706,7 @@ void Daily::on_bookmarkTable_itemClicked(QTableWidgetItem *item) et2=et3; } else if (!day) return; if ((etet2)) { - mainwin->Notify("This bookmarked is in a currently disabled area.."); + mainwin->Notify(tr("This bookmarked is in a currently disabled area..")); return; } @@ -1730,7 +1724,7 @@ void Daily::on_addBookmarkButton_clicked() QDateTime d=QDateTime::fromTime_t(st/1000L); int row=ui->bookmarkTable->rowCount(); ui->bookmarkTable->insertRow(row); - QTableWidgetItem *tw=new QTableWidgetItem("Bookmark at "+d.time().toString("HH:mm:ss")); + QTableWidgetItem *tw=new QTableWidgetItem(tr("Bookmark at %1").arg(d.time().toString("HH:mm:ss"))); QTableWidgetItem *dw=new QTableWidgetItem(d.time().toString("HH:mm:ss")); dw->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled); ui->bookmarkTable->setItem(row,0,dw); @@ -1950,7 +1944,7 @@ void Daily::updateCube() GraphView->setCubeImage(images["nographs"]); } else { - GraphView->setEmptyText("No Data"); + GraphView->setEmptyText(STR_TR_NoData); GraphView->setCubeImage(images["nodata"]); } } else { diff --git a/docs/release_notes.html b/docs/release_notes.html index 705fe861..a2439e18 100644 --- a/docs/release_notes.html +++ b/docs/release_notes.html @@ -1,14 +1,26 @@ -

SleepyHead v0.9.3 BETA

+

SleepyHead v0.9.4 BETA

Release Notes

-

Hi There!

-

Thanks for bearing with us this year - it has been a crazy year for some of us, and we're thankful for a few who have stepped up to help out. Anyone who wants to contribue to SleepyHead (whether with code, documentation, or just helping those with questions), please contact us or post one one of the lists/forums on our website.

-New features & bugs fixes in this Update:
+

Greetings!

+

For a lot of you, this is a minor interim release mainly to bring the various platform builds in sync again and fix a couple of crashes.
+To native Mac users this is a fairly large update because you missed out on the last one, if you weren't using the windows one in Parallels/etc.... (Thanks for being so patient!)

+ +

There is one caveat for this release on native Mac though, printing had to be disabled due to a change in MacOSX that broke something in the QT libraries which causes a consistent crash. +That bug is fixed in the next version of Qt, but something else broke there that's way worse.. Still waiting on upstream for a fix. +Hopefully next time I can switch it back on, but the feedback I've received indicates a lot of you would rather still have the software without it for now.
+ +

Oximetry importing support has been improved, so now you should be able to import directly from your CMS50X oximeters without messing around with the windows software.
+Please remember, for achieving the best sync, always start your oximeter at the same time as your CPAP machine!
+ +

I (Jedimark) am so very greatful for those who have stepped up to help out while I've been out of action. +Also like to thank

+ +New features & bugs fixes in this Update:
-
  • Improved Auto-Updater previously released in a test release.
  • +
  • Improved CMS50 Oximetry serial importing
  • Support for the PRS1 Series 60, in large part due to a patch from Keary Griffin.
  • A bunch of small PRS1 fixes, especially for the AutoSV.
  • Added Flow Limit to the summary and overview screens. This can be a useful indicator in certain types of apnea.
  • @@ -18,7 +30,9 @@

    Sleep Well, and have fun!

    -

    Mark Watkins (JediMark)
    -Richard Freeman (rich0)

    +

    The SleepyHead Team
    +Mark Watkins (JediMark)
    +Richard Freeman (rich0)
    +James Marshall (Breathe Jimbo)

    diff --git a/exportcsv.cpp b/exportcsv.cpp index a1b3a77d..cacc2c36 100644 --- a/exportcsv.cpp +++ b/exportcsv.cpp @@ -188,7 +188,7 @@ void ExportCSV::on_exportButton_clicked() for (int i=0;i #include #include +#include +#include #include "SleepLib/schema.h" #include "mainwindow.h" @@ -87,7 +89,7 @@ void initialize() void release_notes() { QDialog relnotes; - relnotes.setWindowTitle(QObject::tr("SleepyHead Release Notes")); + relnotes.setWindowTitle(STR_TR_SleepyHead+" "+QObject::tr("Release Notes")); QVBoxLayout layout(&relnotes); QWebView web(&relnotes); @@ -109,10 +111,10 @@ void release_notes() void build_notes() { QDialog relnotes; - relnotes.setWindowTitle(QObject::tr("SleepyHead Update Notes")); + relnotes.setWindowTitle(STR_TR_SleepyHead+" "+QObject::tr("SleepyHead Update Notes")); QVBoxLayout layout(&relnotes); QWebView web(&relnotes); - relnotes.setWindowTitle("SleepyHead v"+FullVersionString+" Update"); + relnotes.setWindowTitle(STR_TR_SleepyHead+" v"+FullVersionString+QObject::tr(" Update")); // Language??? web.load(QUrl("qrc:/docs/update_notes.html")); @@ -157,7 +159,27 @@ int main(int argc, char *argv[]) } } - a.setApplicationName("SleepyHead"); + QDir dir(QCoreApplication::applicationDirPath()+"/Translations/"); + dir.setFilter(QDir::Files); + dir.setNameFilters(QStringList("*.qm")); + + qDebug() << "Available Translations"; + QFileInfoList list=dir.entryInfoList(); + for (int i=0;i("Preference"); - PREF["AppName"]=QObject::tr("SleepyHead"); + PREF["AppName"]=STR_TR_SleepyHead; // Skip login screen, unless asked not to on the command line @@ -216,8 +238,8 @@ int main(int argc, char *argv[]) // Show New User wizard.. } else { - if (PREF.contains("VersionString")) { - QString V=PREF["VersionString"].toString(); + if (PREF.contains(STR_PREF_VersionString)) { + QString V=PREF[STR_PREF_VersionString].toString(); if (FullVersionString>V) { QString V2=V.section("-",0,0); @@ -245,7 +267,7 @@ int main(int argc, char *argv[]) } } } - PREF["VersionString"]=FullVersionString; + PREF[STR_PREF_VersionString]=FullVersionString; p_profile=Profiles::Get(PREF[STR_GEN_Profile].toString()); diff --git a/mainwindow.cpp b/mainwindow.cpp index d0359bdd..791df556 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -25,11 +25,15 @@ #include #include #include +#include #include // Custom loaders that don't autoscan.. #include + +#ifndef REMSTAR_M_SUPPORT #include +#endif #include "mainwindow.h" #include "ui_mainwindow.h" @@ -84,7 +88,7 @@ MainWindow::MainWindow(QWidget *parent) : QString version=FullVersionString; if (QString(GIT_BRANCH)!="master") version+=QString(" ")+QString(GIT_BRANCH); - this->setWindowTitle(tr("SleepyHead")+QString(" v%1 (Profile: %2)").arg(version).arg(PREF[STR_GEN_Profile].toString())); + this->setWindowTitle(tr("SleepyHead")+QString(" v%1 ("+tr("Profile")+": %2)").arg(version).arg(PREF[STR_GEN_Profile].toString())); //ui->tabWidget->setCurrentIndex(1); // Disable Screenshot on Mac Platform,as it doesn't work, and the system provides this functionality anyway. @@ -103,7 +107,7 @@ MainWindow::MainWindow(QWidget *parent) : qstatusbar=ui->statusbar; qprogress=new QProgressBar(this); qprogress->setMaximum(100); - qstatus2=new QLabel("Welcome",this); + qstatus2=new QLabel(tr("Welcome"),this); qstatus2->setFrameStyle(QFrame::Raised); qstatus2->setFrameShadow(QFrame::Sunken); qstatus2->setFrameShape(QFrame::Box); @@ -149,13 +153,13 @@ MainWindow::MainWindow(QWidget *parent) : systray->show(); systraymenu=new QMenu(this); systray->setContextMenu(systraymenu); - QAction *a=systraymenu->addAction("SleepyHead v"+VersionString); + QAction *a=systraymenu->addAction(STR_TR_SleepyHead+" v"+VersionString); a->setEnabled(false); systraymenu->addSeparator(); - systraymenu->addAction("&About",this,SLOT(on_action_About_triggered())); - systraymenu->addAction("Check for &Updates",this,SLOT(on_actionCheck_for_Updates_triggered())); + systraymenu->addAction(tr("&About"),this,SLOT(on_action_About_triggered())); + systraymenu->addAction(tr("Check for &Updates"),this,SLOT(on_actionCheck_for_Updates_triggered())); systraymenu->addSeparator(); - systraymenu->addAction("E&xit",this,SLOT(close())); + systraymenu->addAction(tr("E&xit"),this,SLOT(close())); } else { // if not available, the messages will popup in the taskbar systray=NULL; systraymenu=NULL; @@ -186,10 +190,13 @@ MainWindow::MainWindow(QWidget *parent) : //"border-top-left-radius: 8px;" //"border-top-right-radius: 8px;" - QString loadingtxt="
     
    %1
    "+tr("Session Information")+"

    %1%2%3%4%5
    "+tr("SessionID")+""+STR_TR_On+""+STR_TR_Date+""+STR_TR_Start+""+STR_TR_End+"
    %1
    "+tr("CPAP Sessions")+"
    %1
    "+tr("Oximetry Sessions")+"

    Loading...

    "; + QString loadingtxt="

    "+tr("Loading...")+"

    "; ui->summaryView->setHtml(loadingtxt); on_tabWidget_currentChanged(0); - //ui->actionImport_RemStar_MSeries_Data->setVisible(false); + +#ifndef REMSTAR_M_SUPPORT + ui->actionImport_RemStar_MSeries_Data->setVisible(false); +#endif } extern MainWindow *mainwin; MainWindow::~MainWindow() @@ -280,7 +287,7 @@ void MainWindow::Startup() void MainWindow::on_action_Import_Data_triggered() { if (m_inRecalculation) { - Notify("Access to Import has been blocked while recalculations are in progress."); + Notify(tr("Access to Import has been blocked while recalculations are in progress.")); return; } @@ -311,7 +318,7 @@ void MainWindow::on_action_Import_Data_triggered() if (importLocations.size()==0) { asknew=true; } else { - int res=QMessageBox::question(this,"Import from where?","Do you just want to Import from the usual (remembered) locations?\n","The Usual","New Location","Cancel",0,2); + int res=QMessageBox::question(this,tr("Import from where?"),tr("Do you just want to Import from the usual (remembered) locations?\n"),tr("The Usual"),tr("New Location"),tr("Cancel"),0,2); if (res==1) { asknew=true; } @@ -380,7 +387,7 @@ void MainWindow::on_action_Import_Data_triggered() if (overview) overview->ReloadGraphs(); on_summaryButton_clicked(); if (daily) daily->ReloadGraphs(); - if ((goodlocations.size()>0) && (QMessageBox::question(this,"Remember this Location?","Would you like to remember this import location for next time?\n"+newdir,QMessageBox::Yes,QMessageBox::No)==QMessageBox::Yes)) { + if ((goodlocations.size()>0) && (QMessageBox::question(this,tr("Remember this Location?"),tr("Would you like to remember this import location for next time?")+"\n"+newdir,QMessageBox::Yes,QMessageBox::No)==QMessageBox::Yes)) { for (int i=0;i" "" "
    " -"

    SleepyHead v"+VersionString+"

    This is a beta software and some functionality may not work as intended yet.
    Please report any bugs you find to SleepyHead's SourceForge page.
    " +"

    "+QObject::tr("SleepyHead")+" v"+VersionString+"

    "+QObject::tr("This is a beta software and some functionality may not work as intended yet.")+"
    "+QObject::tr("Please report any bugs you find to SleepyHead's SourceForge page.")+"
    " "" "
    "); } @@ -670,7 +677,7 @@ void MainWindow::on_summaryButton_clicked() if (mach.size()==0) { html+=""; QString datacard; - html+="

    Please Import Some Data

    SleepyHead is pretty much useless without it.

    It might be a good idea to check preferences first,
    as there are some options that affect import.

    First import can take a few minutes.

    "; + html+="

    "+tr("Please Import Some Data")+"

    "+tr("SleepyHead is pretty much useless without it.")+"

    "+tr("It might be a good idea to check preferences first,
    as there are some options that affect import.")+"

    "+tr("First import can take a few minutes.")+"

    "; html+=htmlFooter(); ui->summaryView->setHtml(html); return; @@ -693,18 +700,18 @@ void MainWindow::on_summaryButton_clicked() QString ahitxt; if (PROFILE.general->calculateRDI()) { - ahitxt=tr("RDI"); + ahitxt=STR_TR_RDI; } else { - ahitxt=tr("AHI"); + ahitxt=STR_TR_AHI; } int decimals=2; html+="
    "; html+=QString(""); if (cpapdays==0) { - html+=""; + html+=""; } else { - html+=QString("").arg(lastcpap.toString(Qt::SystemLocaleLongDate)); + html+=QString("").arg(lastcpap.toString(Qt::SystemLocaleLongDate)); if (cpap_machines.size()>0) { // html+=QString("").arg(tr("CPAP Summary")); @@ -766,7 +773,7 @@ void MainWindow::on_summaryButton_clicked() .arg(p_profile->calcMin(CPAP_EPAP,MT_CPAP,cpap6month,lastcpap),0,'f',decimals) .arg(p_profile->calcMin(CPAP_EPAP,MT_CPAP,cpapyear,lastcpap),0,'f',decimals); html+=QString("") - .arg(tr("%1% EPAP").arg(percentile*100.0,0,'f',0)) + .arg(QString("%1% "+STR_TR_EPAP).arg(percentile*100.0,0,'f',0)) .arg(p_profile->calcPercentile(CPAP_EPAP,percentile,MT_CPAP),0,'f',decimals) .arg(p_profile->calcPercentile(CPAP_EPAP,percentile,MT_CPAP,cpapweek,lastcpap),0,'f',decimals) .arg(p_profile->calcPercentile(CPAP_EPAP,percentile,MT_CPAP,cpapmonth,lastcpap),0,'f',decimals) @@ -780,7 +787,7 @@ void MainWindow::on_summaryButton_clicked() .arg(p_profile->calcMax(CPAP_IPAP,MT_CPAP,cpap6month,lastcpap),0,'f',decimals) .arg(p_profile->calcMax(CPAP_IPAP,MT_CPAP,cpapyear,lastcpap),0,'f',decimals); html+=QString("") - .arg(tr("%1% IPAP").arg(percentile*100.0,0,'f',0)) + .arg(QString("%1% "+STR_TR_IPAP).arg(percentile*100.0,0,'f',0)) .arg(p_profile->calcPercentile(CPAP_IPAP,percentile,MT_CPAP),0,'f',decimals) .arg(p_profile->calcPercentile(CPAP_IPAP,percentile,MT_CPAP,cpapweek,lastcpap),0,'f',decimals) .arg(p_profile->calcPercentile(CPAP_IPAP,percentile,MT_CPAP,cpapmonth,lastcpap),0,'f',decimals) @@ -1117,34 +1124,34 @@ void MainWindow::on_summaryButton_clicked() CPAPMode mode=(CPAPMode)(int)PROFILE.calcSettingsMax(CPAP_Mode,MT_CPAP,tmpRX[ls]->first,tmpRX[ls]->last); if (mode
    No CPAP Machine Data Imported
    "+tr("No CPAP Machine Data Imported")+"
    CPAP Statistics as of %1
    ")+tr("CPAP Statistics as of")+QString(" %1
    %1
    %1%2%3%4%5%6
    %1%2%3%4%5%6
    ") .arg(tr("Best RX Setting")); - recbox+=QString("") + recbox+=QString("") .arg(tmpRX[ls]->first.toString(Qt::ISODate)) .arg(tmpRX[ls]->last.toString(Qt::ISODate)) .arg(tmpRX[ls]->first.toString(Qt::SystemLocaleShortDate)) .arg(tmpRX[ls]->last.toString(Qt::SystemLocaleShortDate)); recbox+=QString("").arg(ahitxt).arg(tmpRX[ls]->ahi,0,'f',decimals); - recbox+=QString("").arg(tr("Mode")).arg(modestr); + recbox+=QString("").arg(STR_TR_Mode).arg(modestr); recbox+=QString("").arg(minstr).arg(tmpRX[ls]->min,0,'f',1).arg(STR_UNIT_CMH2O); if (!maxstr.isEmpty()) recbox+=QString("").arg(maxstr).arg(tmpRX[ls]->max,0,'f',1).arg(STR_UNIT_CMH2O); if (!maxhistr.isEmpty()) recbox+=QString("").arg(maxhistr).arg(tmpRX[ls]->maxhi,0,'f',1).arg(STR_UNIT_CMH2O); @@ -1154,33 +1161,33 @@ void MainWindow::on_summaryButton_clicked() mode=(CPAPMode)(int)PROFILE.calcSettingsMax(CPAP_Mode,MT_CPAP,tmpRX[0]->first,tmpRX[0]->last); if (mode
    %3
    Start
    End
    %3
    %4
    ")+STR_TR_Start+"
    "+STR_TR_End+QString("
    %3
    %4
    %1%2
    %1%2
    %1%2
    %1%2%3
    %1%2%3
    %1%2%3
    ") .arg(tr("Worst RX Setting")); - recbox+=QString("") + recbox+=QString("") .arg(tmpRX[0]->first.toString(Qt::ISODate)) .arg(tmpRX[0]->last.toString(Qt::ISODate)) .arg(tmpRX[0]->first.toString(Qt::SystemLocaleShortDate)) .arg(tmpRX[0]->last.toString(Qt::SystemLocaleShortDate)); recbox+=QString("").arg(ahitxt).arg(tmpRX[0]->ahi,0,'f',decimals); - recbox+=QString("").arg(tr("Mode")).arg(modestr); + recbox+=QString("").arg(STR_TR_Mode).arg(modestr); recbox+=QString("").arg(minstr).arg(tmpRX[0]->min,0,'f',1).arg(STR_UNIT_CMH2O); if (!maxstr.isEmpty()) recbox+=QString("").arg(maxstr).arg(tmpRX[0]->max,0,'f',1).arg(STR_UNIT_CMH2O); if (!maxhistr.isEmpty()) recbox+=QString("").arg(maxhistr).arg(tmpRX[0]->maxhi,0,'f',1).arg(STR_UNIT_CMH2O); @@ -1195,32 +1202,32 @@ void MainWindow::on_summaryButton_clicked() RXorder=true; qSort(rxchange.begin(),rxchange.end());*/ html+="
    "; - html+=QString("
    Changes to Prescription Settings"); + html+=QString("
    ")+tr("Changes to Prescription Settings")+""; html+=QString("
    %3
    Start
    End
    %3
    %4
    ")+STR_TR_Start+"
    "+STR_TR_End+QString("
    %3
    %4
    %1%2
    %1%2
    %1%2
    %1%2%3
    %1%2%3
    %1%2%3
    "); QString extratxt; if (cpapmode>=MODE_ASV) { extratxt=QString("") - .arg(tr("EPAP")).arg(tr("IPAPLo")).arg(tr("IPAPHi")).arg(tr("PS Min")).arg(tr("PS Max")); + .arg(STR_TR_EPAP).arg(STR_TR_IPAPLo).arg(STR_TR_IPAPHi).arg(tr("PS Min")).arg(tr("PS Max")); } else if (cpapmode>=MODE_BIPAP) { extratxt=QString("") - .arg(tr("EPAP")).arg(tr("IPAP")).arg(tr("PS")); + .arg(STR_TR_EPAP).arg(STR_TR_IPAP).arg(STR_TR_PS); } else if (cpapmode>MODE_CPAP) { extratxt=QString("") .arg(tr("Min Pres.")).arg(tr("Max Pres.")); } else { extratxt=QString("") - .arg(tr("Pressure")); + .arg(STR_TR_Pressure); } QString tooltip; html+=QString("%9") - .arg(tr("First")) - .arg(tr("Last")) + .arg(STR_TR_First) + .arg(STR_TR_Last) .arg(tr("Days")) .arg(ahitxt) .arg(tr("FL")) - .arg(tr("Machine")) - .arg(tr("Mode")) + .arg(STR_TR_Machine) + .arg(STR_TR_Mode) .arg(tr("Pr. Rel.")) .arg(extratxt); @@ -1251,30 +1258,21 @@ void MainWindow::on_summaryButton_clicked() extratxt=QString("") .arg(rx.max,0,'f',decimals).arg(rx.maxhi,0,'f',decimals).arg(rx.max-rx.min,0,'f',decimals).arg(rx.maxhi-rx.min,0,'f',decimals); - tooltip=tr("%5 %1% EPAP=%2
    %3% IPAP=%4") - .arg(percentile*100.0) - .arg(rx.per1,0,'f',decimals) - .arg(percentile*100.0) - .arg(rx.per2,0,'f',decimals) - .arg(machstr) - ; + tooltip=QString("%1 %2% ").arg(machstr).arg(percentile*100.0)+STR_TR_EPAP+ + QString("=%1
    %2% ").arg(rx.per1,0,'f',decimals).arg(percentile*100.0)+ + STR_TR_IPAP+QString("=%1").arg(rx.per2,0,'f',decimals); } else if (mode>=MODE_BIPAP) { extratxt=QString("") .arg(rx.max,0,'f',decimals).arg(rx.max-rx.min,0,'f',decimals); - tooltip=tr("%5 %1% EPAP=%2
    %3% IPAP=%4") + tooltip=QString("%1 %2% ").arg(machstr).arg(percentile*100.0)+ + STR_TR_EPAP+ + QString("=%1
    %2% ").arg(rx.per1,0,'f',decimals) .arg(percentile*100.0) - .arg(rx.per1,0,'f',decimals) - .arg(percentile*100.0) - .arg(rx.per2,0,'f',decimals) - .arg(machstr) - ; + +STR_TR_IPAP+QString("=%1").arg(rx.per2,0,'f',decimals); } else if (mode>MODE_CPAP) { extratxt=QString("").arg(rx.max,0,'f',decimals); - tooltip=tr("%3 %1% Pressure=%2") - .arg(percentile*100.0) - .arg(rx.per1,0,'f',decimals) - .arg(machstr) - ; + tooltip=QString("%1 %2% ").arg(machstr).arg(percentile*100.0)+STR_TR_Pressure+ + QString("=%2").arg(rx.per1,0,'f',decimals); } else { if (cpapmode>MODE_CPAP) { extratxt=""; @@ -1288,7 +1286,7 @@ void MainWindow::on_summaryButton_clicked() if (rx.prelset>0) { presrel=schema::channel[CPAP_PresReliefType].option(int(rx.prelief)); presrel+=QString(" x%1").arg(rx.prelset); - } else presrel="None"; + } else presrel=STR_TR_None; QString tooltipshow,tooltiphide; if (!tooltip.isEmpty()) { tooltipshow=QString("tooltip.show(\"%1\");").arg(tooltip); @@ -1311,19 +1309,19 @@ void MainWindow::on_summaryButton_clicked() .arg(tooltiphide); } html+="
    %1%2%3%4%5%1%2%3%1%2%1
    %1%2%3%4%5%6%7%8
    %1%2%3%4%1%2%1 
    "; - html+=QString("The above has a threshold which excludes day counts less than %1 from the best/worst highlighting
    ").arg(rxthresh); + html+=QString("")+tr("The above has a threshold which excludes day counts less than %1 from the best/worst highlighting").arg(rxthresh)+QString("
    "); html+=""; } if (mach.size()>0) { html+="
    "; - html+=QString("
    Machine Information"); + html+=QString("
    ")+tr("Machine Information")+""; html+=QString(""); html+=QString("") - .arg(tr("Brand")) - .arg(tr("Model")) - .arg(tr("Serial")) + .arg(STR_TR_Brand) + .arg(STR_TR_Model) + .arg(STR_TR_Serial) .arg(tr("First Use")) .arg(tr("Last Use")); Machine *m; @@ -1499,24 +1497,24 @@ void MainWindow::on_action_About_triggered() QString msg=QString( "" "
    %1%2%3%4%5
    " -"

    SleepyHead v%1.%2.%3-%4 (%8)

    Build Date: %5 %6
    %7
    Data Folder: %9


    " -"Copyright ©2012 Mark Watkins (jedimark)
    \n" -"This software is released under the GNU Public License v3.0
    " -"

    SleepyHead Project Page: http://sourceforge.net/projects/sleepyhead
    " -"SleepyHead Wiki: http://sleepyhead.sourceforge.net
    " -"Authors Twitter Feed: http://twitter.com/jedimark64

    " -"
    " -"

    The author wishes to express thanks to James Marshall and Rich Freeman for their assistance with this project.

    " -"

    This software comes with absolutely no warranty, either express of implied. It comes with no guarantee of fitness for any particular purpose. No guarantees are made regarding the accuracy of any data this program displays.

    " -"

    This is NOT medical software, it is merely a research tool that provides a visual interpretation of data recorded by supported devices. This software is NOT suitable for medical diagnosis, CPAP complaince reporting and other similar purposes.

    " -"

    The author and any associates of his accept NO responsibilty for damages, issues or non-issues resulting from the use or mis-use of this software
    Use this software entirely at your own risk.

    " -"

    If you find this free software to be of use, please consider supporting the development efforts by making a paypal donation to the Author

    " +"

    "+tr("SleepyHead")+" v%1.%2.%3-%4 (%8)

    "+tr("Build Date")+": %5 %6
    %7
    "+tr("Data Folder")+": %9


    "+ +tr("Copyright")+" ©2012 Mark Watkins (jedimark)
    \n"+ +tr("This software is released under the GNU Public License v3.0
    ")+ +"

    "+tr("SleepyHead Project Page")+": http://sourceforge.net/projects/sleepyhead
    "+ +tr("SleepyHead Wiki")+": http://sleepyhead.sourceforge.net
    "+ +tr("Authors Twitter Feed")+": http://twitter.com/jedimark64

    "+ +"
    "+ +tr("

    The author wishes to express thanks to James Marshall and Rich Freeman for their assistance with this project.

    ")+ +"

    "+tr("This software comes with absolutely no warranty, either express of implied. It comes with no guarantee of fitness for any particular purpose. No guarantees are made regarding the accuracy of any data this program displays.")+"

    " +"

    "+tr("This is NOT medical software, it is merely a research tool that provides a visual interpretation of data recorded by supported devices. This software is NOT suitable for medical diagnosis, CPAP complaince reporting and other similar purposes.")+"

    " +"

    "+tr("The author and any associates of his accept NO responsibilty for damages, issues or non-issues resulting from the use or mis-use of this software
    Use this software entirely at your own risk.")+"

    " +"

    "+tr("If you find this free software to be of use, please consider supporting the development efforts by making a paypal donation to the Author")+"

    " "
    " - ).arg(major_version).arg(minor_version).arg(revision_number).arg(release_number).arg(__DATE__).arg(__TIME__).arg(gitrev).arg(ReleaseStatus).arg(GetAppRoot()); +).arg(major_version).arg(minor_version).arg(revision_number).arg(release_number).arg(__DATE__).arg(__TIME__).arg(gitrev).arg(ReleaseStatus).arg(GetAppRoot()); //"
    " QDialog aboutbox; - aboutbox.setWindowTitle(QObject::tr("About a")); + aboutbox.setWindowTitle(QObject::tr("About SleepyHead")); QVBoxLayout layout(&aboutbox); @@ -1576,7 +1574,7 @@ void MainWindow::on_action_Reset_Graph_Layout_triggered() void MainWindow::on_action_Preferences_triggered() { if (m_inRecalculation) { - mainwin->Notify("Access to Preferences has been blocked until recalculation completes."); + mainwin->Notify(tr("Access to Preferences has been blocked until recalculation completes.")); return; } PreferencesDialog pd(this,p_profile); @@ -1605,7 +1603,7 @@ void MainWindow::on_oximetryButton_clicked() bool first=false; if (!oximetry) { if (!PROFILE.oxi->oximetryEnabled()) { - if (QMessageBox::question(this,"Question","Do you have a CMS50[x] Oximeter?\nOne is required to use this section.",QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) return; + if (QMessageBox::question(this,tr("Question"),tr("Do you have a CMS50[x] Oximeter?\nOne is required to use this section."),QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) return; PROFILE.oxi->setOximetryEnabled(true); } oximetry=new Oximetry(ui->tabWidget,daily->graphView()); @@ -1669,6 +1667,12 @@ void MainWindow::updatestatusBarMessage (const QString & text) void MainWindow::on_actionPrint_Report_triggered() { +#ifdef Q_WS_MAC + #if ((QT_VERSION <= QT_VERSION_CHECK(5, 0, 0)) && (QT_VERSION >= QT_VERSION_CHECK(4, 8, 0))) + QMessageBox::information(this,tr("Printing Broken"),tr("Sorry.. Printing is currently broken on the Mac platform :-(\n\nWe are currently waiting on a Qt Library bugfix"),QMessageBox::Ok); + return; + #endif +#endif if (ui->tabWidget->currentWidget()==overview) { PrintReport(overview->graphView(),STR_TR_Overview); } else if (ui->tabWidget->currentWidget()==daily) { @@ -1779,7 +1783,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) if (journal && journal->settings.contains(Bookmark_Start)) { book_start=journal->settings[Bookmark_Start].toList(); if (book_start.size()>0) { - if (QMessageBox::question(this,tr("Bookmarks"),tr("Would you like to show bookmarked areas in this report?"),QMessageBox::Yes,QMessageBox::No)==QMessageBox::Yes) { + if (QMessageBox::question(this,STR_TR_Bookmarks,tr("Would you like to show bookmarked areas in this report?"),QMessageBox::Yes,QMessageBox::No)==QMessageBox::Yes) { print_bookmarks=true; } } @@ -1858,12 +1862,12 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) int maxy=0; if (!PROFILE.user->firstName().isEmpty()) { - QString userinfo=tr("Name:\t %1, %2\n").arg(PROFILE.user->lastName()).arg(PROFILE.user->firstName()); - userinfo+=tr("DOB:\t%1\n").arg(PROFILE.user->DOB().toString(Qt::SystemLocaleShortDate)); - if (!PROFILE.doctor->patientID().isEmpty()) userinfo+=tr("Patient ID:\t%1\n").arg(PROFILE.doctor->patientID()); - userinfo+=tr("Phone:\t%1\n").arg(PROFILE.user->phone()); - userinfo+=tr("Email:\t%1\n").arg(PROFILE.user->email()); - if (!PROFILE.user->address().isEmpty()) userinfo+=tr("\nAddress:\n%1").arg(PROFILE.user->address()); + QString userinfo=STR_TR_Name+QString(":\t %1, %2\n").arg(PROFILE.user->lastName()).arg(PROFILE.user->firstName()); + userinfo+=STR_TR_DOB+QString(":\t%1\n").arg(PROFILE.user->DOB().toString(Qt::SystemLocaleShortDate)); + if (!PROFILE.doctor->patientID().isEmpty()) userinfo+=STR_TR_PatientID+QString(":\t%1\n").arg(PROFILE.doctor->patientID()); + userinfo+=STR_TR_Phone+QString(":\t%1\n").arg(PROFILE.user->phone()); + userinfo+=STR_TR_Email+QString(":\t%1\n").arg(PROFILE.user->email()); + if (!PROFILE.user->address().isEmpty()) userinfo+="\n"+STR_TR_Address+QString(":\n%1").arg(PROFILE.user->address()); QRectF bounds=painter.boundingRect(QRectF(0,top,virt_width,0),userinfo,QTextOption(Qt::AlignLeft | Qt::AlignTop)); painter.drawText(bounds,userinfo,QTextOption(Qt::AlignLeft | Qt::AlignTop)); @@ -1884,29 +1888,29 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) int m=(tt/60)%60; int s=tt % 60; - cpapinfo+=tr("Mask Time: ")+QString().sprintf("%2i hours, %2i minutes, %2i seconds",h,m,s)+"\n"; - cpapinfo+=tr("Bedtime: ")+QDateTime::fromTime_t(f).time().toString("HH:mm:ss")+" "; - cpapinfo+=tr("Wake-up: ")+QDateTime::fromTime_t(l).time().toString("HH:mm:ss")+"\n\n"; + cpapinfo+=STR_TR_MaskTime+tr(": %1 hours, %2 minutes, %3 seconds\n").arg(h).arg(m).arg(s); + cpapinfo+=STR_TR_BedTime+": "+QDateTime::fromTime_t(f).time().toString("HH:mm:ss")+" "; + cpapinfo+=STR_TR_WakeUp+": "+QDateTime::fromTime_t(l).time().toString("HH:mm:ss")+"\n\n"; QString submodel; - cpapinfo+=tr("Machine: "); + cpapinfo+=STR_TR_Machine+": "; if (cpap->machine->properties.find(STR_PROP_SubModel)!=cpap->machine->properties.end()) submodel="\n"+cpap->machine->properties[STR_PROP_SubModel]; cpapinfo+=cpap->machine->properties[STR_PROP_Brand]+" "+cpap->machine->properties[STR_PROP_Model]+submodel; CPAPMode mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode); - cpapinfo+=tr("\nMode: "); + cpapinfo+="\n"+STR_TR_Mode+": "; if (mode==MODE_CPAP) { EventDataType min=round(cpap->settings_wavg(CPAP_Pressure)*2)/2.0; - cpapinfo+=tr("CPAP")+" "+QString::number(min)+STR_UNIT_CMH2O; + cpapinfo+=STR_TR_CPAP+" "+QString::number(min)+STR_UNIT_CMH2O; } else if (mode==MODE_APAP) { EventDataType min=cpap->settings_min(CPAP_PressureMin); EventDataType max=cpap->settings_max(CPAP_PressureMax); - cpapinfo+=tr("APAP")+" "+QString::number(min)+"-"+QString::number(max)+STR_UNIT_CMH2O; + cpapinfo+=STR_TR_APAP+" "+QString::number(min)+"-"+QString::number(max)+STR_UNIT_CMH2O; } else if (mode==MODE_BIPAP) { EventDataType epap=cpap->settings_min(CPAP_EPAP); EventDataType ipap=cpap->settings_max(CPAP_IPAP); EventDataType ps=cpap->settings_max(CPAP_PS); - cpapinfo+=tr("Bi-Level")+QString("\nEPAP: %1 IPAP: %2 %3\nPS: %4") + cpapinfo+=STR_TR_BiLevel+QString("\n"+STR_TR_EPAP+": %1 "+STR_TR_IPAP+": %2 %3\n"+STR_TR_PS+": %4") .arg(epap,0,'f',1).arg(ipap,0,'f',1).arg(STR_UNIT_CMH2O).arg(ps,0,'f',1); } else if (mode==MODE_ASV) { @@ -1915,7 +1919,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) EventDataType high=cpap->settings_max(CPAP_IPAPHi); EventDataType psl=cpap->settings_min(CPAP_PSMin); EventDataType psh=cpap->settings_max(CPAP_PSMax); - cpapinfo+=tr("ASV")+QString("\nEPAP: %1 IPAP: %2 - %3 %4\nPS: %5 / %6") + cpapinfo+=STR_TR_ASV+QString("\n"+STR_TR_EPAP+": %1 "+STR_TR_IPAP+": %2 - %3 %4\n"+STR_TR_PS+": %5 / %6") .arg(epap,0,'f',1) .arg(low,0,'f',1) .arg(high,0,'f',1) @@ -1923,7 +1927,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) .arg(psl,0,'f',1) .arg(psh,0,'f',1); } - else cpapinfo+=tr("Unknown"); + else cpapinfo+=STR_TR_Unknown; float ahi=(cpap->count(CPAP_Obstructive)+cpap->count(CPAP_Hypopnea)+cpap->count(CPAP_ClearAirway)+cpap->count(CPAP_Apnea)); if (PROFILE.general->calculateRDI()) ahi+=cpap->count(CPAP_RERA); @@ -1994,11 +1998,11 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) if (journal) { stats=""; if (journal->settings.contains(Journal_Weight)) - stats+=tr("Weight %1 ").arg(weightString(journal->settings[Journal_Weight].toDouble())); + stats+=STR_TR_Weight+QString(" %1 ").arg(weightString(journal->settings[Journal_Weight].toDouble())); if (journal->settings.contains(Journal_BMI)) - stats+=tr("BMI %1 ").arg(journal->settings[Journal_BMI].toDouble(),0,'f',2); + stats+=STR_TR_BMI+QString(" %1 ").arg(journal->settings[Journal_BMI].toDouble(),0,'f',2); if (journal->settings.contains(Journal_ZombieMeter)) - stats+=tr("Zombie %1/10 ").arg(journal->settings[Journal_ZombieMeter].toDouble(),0,'f',0); + stats+=STR_TR_Zombie+QString(" %1/10 ").arg(journal->settings[Journal_ZombieMeter].toDouble(),0,'f',0); if (!stats.isEmpty()) { bounds=painter.boundingRect(QRectF(0,top+ttop,virt_width,0),stats,QTextOption(Qt::AlignHCenter)); @@ -2096,6 +2100,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) } } else { + const QString EntireDay=tr("Entire Day"); if (journal) { if (journal->settings.contains(Bookmark_Start)) { QVariantList st1=journal->settings[Bookmark_Start].toList(); @@ -2107,19 +2112,19 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date) if (cpap && flow && !flow->isEmpty() && flow->visible()) { - labels.push_back("Entire Day"); + labels.push_back(EntireDay); start.push_back(cpap->first()); end.push_back(cpap->last()); graphs.push_back(flow); } if (oxi && spo2 && !spo2->isEmpty() && spo2->visible()) { - labels.push_back("Entire Day"); + labels.push_back(EntireDay); start.push_back(oxi->first()); end.push_back(oxi->last()); graphs.push_back(spo2); } if (oxi && pulse && !pulse->isEmpty() && pulse->visible()) { - labels.push_back("Entire Day"); + labels.push_back(EntireDay); start.push_back(oxi->first()); end.push_back(oxi->last()); graphs.push_back(pulse); @@ -2733,11 +2738,11 @@ void MainWindow::doReprocessEvents() qprogress->setVisible(false); m_inRecalculation=false; if (m_restartRequired) { - QMessageBox::information(this,"Restart Required",QString("Recalculations are complete, the application now needs to restart to display the changes."),QMessageBox::Ok); + QMessageBox::information(this,tr("Restart Required"),tr("Recalculations are complete, the application now needs to restart to display the changes."),QMessageBox::Ok); RestartApplication(); return; } else { - Notify("Recalculations are now complete.","Task Completed"); + Notify(tr("Recalculations are now complete."),tr("Task Completed")); FreeSessions(); QDate current=daily->getDate(); @@ -2758,10 +2763,10 @@ void MainWindow::on_actionImport_ZEO_Data_triggered() if (w.exec()==QFileDialog::Accepted) { QString filename=w.selectedFiles()[0]; if (!zeo.OpenFile(filename)) { - Notify("There was a problem opening ZEO File: "+filename); + Notify(tr("There was a problem opening ZEO File: ")+filename); return; } - Notify("Zeo CSV Import complete"); + Notify(tr("Zeo CSV Import complete")); daily->LoadDate(daily->getDate()); } @@ -2770,6 +2775,7 @@ void MainWindow::on_actionImport_ZEO_Data_triggered() void MainWindow::on_actionImport_RemStar_MSeries_Data_triggered() { +#ifdef REMSTAR_M_SUPPORT QFileDialog w; w.setFileMode(QFileDialog::ExistingFiles); w.setOption(QFileDialog::ShowDirsOnly, false); @@ -2780,12 +2786,13 @@ void MainWindow::on_actionImport_RemStar_MSeries_Data_triggered() if (w.exec()==QFileDialog::Accepted) { QString filename=w.selectedFiles()[0]; if (!mseries.Open(filename,p_profile)) { - Notify("There was a problem opening MSeries block File: "+filename); + Notify(tr("There was a problem opening MSeries block File: ")+filename); return; } - Notify("MSeries Import complete"); + Notify(tr("MSeries Import complete")); daily->LoadDate(daily->getDate()); } +#endif } void MainWindow::on_actionSleep_Disorder_Terms_Glossary_triggered() diff --git a/overview.cpp b/overview.cpp index b2f23b93..78e05da3 100644 --- a/overview.cpp +++ b/overview.cpp @@ -72,7 +72,7 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : GraphView=new gGraphView(ui->graphArea,m_shared); GraphView->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding); - GraphView->setEmptyText("No Data"); + GraphView->setEmptyText(STR_TR_NoData); GraphView->setCubeImage(images["nodata"]); // Create the custom scrollbar and attach to GraphView @@ -92,13 +92,13 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : ChannelID ahicode=PROFILE.general->calculateRDI() ? CPAP_RDI : CPAP_AHI; if (ahicode==CPAP_RDI) - AHI=createGraph(tr("RDI"),"Respiratory\nDisturbance\nIndex"); + AHI=createGraph(STR_TR_RDI,tr("Respiratory\nDisturbance\nIndex")); else - AHI=createGraph(tr("AHI"),tr("Apnea\nHypopnea\nIndex")); + AHI=createGraph(STR_TR_AHI,tr("Apnea\nHypopnea\nIndex")); UC=createGraph(tr("Usage"),tr("Usage\n(hours)")); - FL=createGraph(tr("Flow Limit"),tr("Flow Limit")); + FL=createGraph(STR_TR_FlowLimit,STR_TR_FlowLimit); float percentile=PROFILE.general->prefCalcPercentile()/100.0; int mididx=PROFILE.general->prefCalcMiddle(); @@ -112,28 +112,28 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : US=createGraph(tr("Session Times"),tr("Session Times\n(hours)"),YT_Time); - PR=createGraph(STR_TR_Pressure,tr("Pressure\n(cmH2O)")); - SET=createGraph(tr("Settings"),("Settings")); - LK=createGraph(tr("Leaks"),tr("Unintentional Leaks\n(L/min)")); - TOTLK=createGraph(tr("Total Leaks"),tr("Total Leaks\n(L/min)")); + PR=createGraph(STR_TR_Pressure,STR_TR_Pressure+"\n("+STR_UNIT_CMH2O+")"); + SET=createGraph(STR_TR_Settings,STR_TR_Settings); + LK=createGraph(STR_TR_Leaks,STR_TR_UnintentionalLeaks+"\n("+STR_UNIT_LPM+")"); + TOTLK=createGraph(STR_TR_TotalLeaks,STR_TR_TotalLeaks+"\n("+STR_UNIT_LPM+")"); NPB=createGraph(tr("% in PB"),tr("Periodic\nBreathing\n(% of night)")); if (ahicode==CPAP_RDI) { AHIHR=createGraph(tr("Peak RDI"),tr("Peak RDI\nShows RDI Clusters\n(RDI/hr)")); } else { AHIHR=createGraph(tr("Peak AHI"),tr("Peak AHI\nShows AHI Clusters\n(AHI/hr)")); } - RR=createGraph(tr("Resp. Rate"),tr("Respiratory\nRate\n(breaths/min)")); - TV=createGraph(tr("Tidal Volume"),tr("Tidal\nVolume\n(ml)")); - MV=createGraph(tr("Minute Vent."),tr("Minute\nVentilation\n(L/min)")); - TGMV=createGraph(tr("Target Vent."),tr("Target\nVentilation\n(L/min)")); - PTB=createGraph(tr("Pat. Trig. Br."),tr("Patient\nTriggered\nBreaths\n(%)")); - SES=createGraph(tr("Sessions"),tr("Sessions\n(count)")); - PULSE=createGraph(tr("Pulse Rate"),tr("Pulse Rate\n(bpm)")); + RR=createGraph(STR_TR_RespRate,tr("Respiratory\nRate\n(breaths/min)")); + TV=createGraph(STR_TR_TidalVolume,tr("Tidal\nVolume\n(ml)")); + MV=createGraph(STR_TR_MinuteVent,tr("Minute\nVentilation\n(L/min)")); + TGMV=createGraph(STR_TR_TargetVent,tr("Target\nVentilation\n(L/min)")); + PTB=createGraph(STR_TR_PatTrigBreath,tr("Patient\nTriggered\nBreaths\n(%)")); + SES=createGraph(STR_TR_Sessions,STR_TR_Sessions+tr("\n(count)")); + PULSE=createGraph(STR_TR_PulseRate,STR_TR_PulseRate+"\n("+STR_UNIT_BPM+")"); SPO2=createGraph(STR_TR_SpO2,tr("Oxygen Saturation\n(%)")); WEIGHT=createGraph(STR_TR_Weight,STR_TR_Weight,YT_Weight); BMI=createGraph(STR_TR_BMI,tr("Body\nMass\nIndex")); - ZOMBIE=createGraph(tr("Zombie"),tr("How you felt\n(0-10)")); + ZOMBIE=createGraph(STR_TR_Zombie,tr("How you felt\n(0-10)")); ahihr=new SummaryChart(tr("Events/Hr"),GT_LINE); ahihr->addSlice(ahicode,QColor("blue"),ST_MAX); @@ -155,7 +155,7 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : zombie->addSlice(Journal_ZombieMeter,QColor("dark red"),ST_SETAVG); ZOMBIE->AddLayer(zombie); - pulse=new SummaryChart(tr("Pulse Rate"),GT_LINE); + pulse=new SummaryChart(STR_TR_PulseRate,GT_LINE); pulse->setMachineType(MT_OXIMETER); pulse->addSlice(OXI_Pulse,QColor("red"),ST_mid,0.5); pulse->addSlice(OXI_Pulse,QColor("pink"),ST_MIN); @@ -173,7 +173,7 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : uc->addSlice(NoChannel,QColor("green"),ST_HOURS); UC->AddLayer(uc); - fl=new SummaryChart(tr("FL"),GT_BAR); + fl=new SummaryChart(STR_TR_FL,GT_BAR); fl->addSlice(CPAP_FlowLimit,QColor("brown"),ST_CPH); FL->AddLayer(fl); @@ -182,14 +182,14 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : us->addSlice(NoChannel,QColor("blue"),ST_SESSIONS); US->AddLayer(us); - ses=new SummaryChart(tr("Sessions"),GT_LINE); + ses=new SummaryChart(STR_TR_Sessions,GT_LINE); ses->addSlice(NoChannel,QColor("blue"),ST_SESSIONS); SES->AddLayer(ses); if (ahicode==CPAP_RDI) - bc=new SummaryChart(tr("RDI"),GT_BAR); + bc=new SummaryChart(STR_TR_RDI,GT_BAR); else - bc=new SummaryChart(tr("AHI"),GT_BAR); + bc=new SummaryChart(STR_TR_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); @@ -222,7 +222,7 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : tv->addSlice(CPAP_TidalVolume,QColor("green"),ST_max,maxperc); TV->AddLayer(tv); - mv=new SummaryChart(tr("L/m"),GT_LINE); + mv=new SummaryChart(STR_UNIT_LPM,GT_LINE); mv->addSlice(CPAP_MinuteVent,QColor("light blue"),ST_MIN); mv->addSlice(CPAP_MinuteVent,QColor("blue"),ST_mid,0.5); mv->addSlice(CPAP_MinuteVent,QColor("light green"),ST_PERC,percentile); @@ -230,7 +230,7 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : MV->AddLayer(mv); // should merge... - tgmv=new SummaryChart(tr("L/m"),GT_LINE); + tgmv=new SummaryChart(STR_UNIT_LPM,GT_LINE); tgmv->addSlice(CPAP_TgMV,QColor("light blue"),ST_MIN); tgmv->addSlice(CPAP_TgMV,QColor("blue"),ST_mid,0.5); tgmv->addSlice(CPAP_TgMV,QColor("light green"),ST_PERC,percentile); @@ -248,7 +248,7 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : // Added in summarychart.. Slightly annoying.. PR->AddLayer(pr); - lk=new SummaryChart(tr("Leaks"),GT_LINE); + lk=new SummaryChart(STR_TR_Leaks,GT_LINE); lk->addSlice(CPAP_Leak,QColor("light blue"),ST_mid,0.5); lk->addSlice(CPAP_Leak,QColor("dark grey"),ST_PERC,percentile); //lk->addSlice(CPAP_Leak,QColor("dark blue"),ST_WAVG); @@ -256,7 +256,7 @@ Overview::Overview(QWidget *parent,gGraphView * shared) : //lk->addSlice(CPAP_Leak,QColor("dark yellow")); LK->AddLayer(lk); - totlk=new SummaryChart(tr("Total Leaks"),GT_LINE); + totlk=new SummaryChart(STR_TR_TotalLeaks,GT_LINE); totlk->addSlice(CPAP_LeakTotal,QColor("light blue"),ST_mid,0.5); totlk->addSlice(CPAP_LeakTotal,QColor("dark grey"),ST_PERC,percentile); totlk->addSlice(CPAP_LeakTotal,QColor("grey"),ST_max,maxperc); @@ -579,7 +579,7 @@ void Overview::updateCube() GraphView->setCubeImage(images["nographs"]); } else { - GraphView->setEmptyText("No Data"); + GraphView->setEmptyText(STR_TR_NoData); GraphView->setCubeImage(images["nodata"]); } } else { diff --git a/oximetry.cpp b/oximetry.cpp index 401c1739..dfd132ae 100644 --- a/oximetry.cpp +++ b/oximetry.cpp @@ -455,7 +455,7 @@ void CMS50Serial::import_process() qDebug() << "User pushing import too many times in a row?"; return; } - mainwin->getOximetry()->graphView()->setEmptyText("Processing..."); + mainwin->getOximetry()->graphView()->setEmptyText(tr("Processing...")); mainwin->getOximetry()->graphView()->redraw(); qDebug() << "CMS50 import complete. Processing" << data.size() << "bytes"; @@ -466,7 +466,36 @@ void CMS50Serial::import_process() EventList * pulse=(session->eventlist[OXI_Pulse][0]); EventList * spo2=(session->eventlist[OXI_SPO2][0]); - lasttime=f2time.toTime_t(); + + int d=abs(oxitime.secsTo(cpaptime)); + + QDateTime seltime=oxitime; + + if (!cpaptime.isNull()) { + if (QMessageBox::question(mainwin,tr("Question"),tr("Did you remember to start your oximeter recording at exactly the same time you started your CPAP machine?"),QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) { + if (!cms50dplus) { + // Oximeter has a clock.. Hopefully the user remembered to set their clock on the device.. + QMessageBox::information(mainwin,"Information","That's ok, I will use the time provided by your oximeter, however it will sync better next time if you start your oximeter recording at the same time your CPAP machine starts up.\n(Please note: If you haven't set your oximeter clock you will have to manually edit this time before saving this oximetry session.)",QMessageBox::Ok); + } else { + //CMS50D+, and the user didn't start at the same time.. Kludge it because they likely turned it on around about the same time anyway. + QMessageBox::information(mainwin,"Information","It looks like your oximeter doesn't provide a valid start time, I'm going to set this oximetry session starting time to the CPAP starting time anyway.\nYou may have to adjust it manually if you remember the real start time before saving this session. (Also, did you remember to import todays CPAP data first?)",QMessageBox::Ok); + seltime=cpaptime; + } + } else { + // The best solution.. the user (hopefully) started both devices at the same time, so we pick the cpap sessions start time for optimal sync. + QMessageBox::information(mainwin,tr("Information"),tr("The most recent CPAP Session time has been selected as the start of your oximetry session.\nIf you forgot to import todays CPAP data first, go and do that now, then import again from your oximeter."),QMessageBox::Ok); + seltime=cpaptime; + } + } else { + if (cms50dplus) { + // Worst case, CMS50D+ and no CPAP data.. the time is basically set to midnight the current day. + QMessageBox::information(mainwin,tr("Information"),tr("No valid start time was provided for this oximeter session. You will likely have to adjust your oximeter sessions start time before saving."),QMessageBox::Ok); + } else { + // No point nagging the user at all in this case.. they don't have any CPAP data loaded, so they are just using SleepyHead with the oximeter + } + } + + lasttime=seltime.toTime_t(); session->SetSessionID(lasttime); lasttime*=1000; @@ -690,33 +719,37 @@ void CMS50Serial::ReadyRead() started_import=true; started_reading=false; - mainwin->getOximetry()->graphView()->setEmptyText("Please Wait, Importing..."); + mainwin->getOximetry()->graphView()->setEmptyText(tr("Please Wait, Importing...")); mainwin->getOximetry()->graphView()->timedRedraw(50); - if ((hour!=0) && (minute!=0)) { // CMS50E/F's have a realtime clock - f2time=QDateTime(QDate::currentDate(),QTime(hour,minute)); - f2time=f2time.toTimeSpec(Qt::UTC); - qDebug() << "Session start (according to CMS50)" << f2time << hex << buffer.at(i+1) << buffer.at(i+2) << ":" << dec << hour << minute ; - } else { - // otherwise pick the first session of the last days data.. - Day *day=PROFILE.GetDay(PROFILE.LastDay(),MT_CPAP); - if (day) { - int ti=day->first()/1000L; + QDate oda=QDate::currentDate(); + QTime oti=QTime(hour,minute); // Only CMS50E/F's have a realtime clock. CMS50D+ will set this to midnight - f2time=QDateTime::fromTime_t(ti); - qDebug() << "Guessing session starting from CPAP data" << f2time; - } + cms50dplus=(hour==0) && (minute==0); // Either a CMS50D+ or it's really midnight, set a flag anyway for later to help choose the right sync time - if (!f2time.isValid()) { - // CMS50D+ are rather stupid. - qDebug() << "Can't guess start time, defaulting to midnight" << f2time; - f2time=QDateTime::currentDateTime(); - f2time.setTime(QTime(0,0,0)); - //f2time.addDays(-1); - } + // If the oximeter record time is more than the current time, then assume it was from the day before + // Or should I use split time preference instead??? Foggy Confusements.. + if (oti > QTime::currentTime()) + oda=oda.addDays(-1); - } - qDebug() << "Record Import:" << f2time; + oxitime=QDateTime(oda,oti); + + // Convert it to UTC + oxitime=oxitime.toTimeSpec(Qt::UTC); + + qDebug() << "Session start (according to CMS50)" << oxitime<< hex << buffer.at(i+1) << buffer.at(i+2) << ":" << dec << hour << minute ; + + // As an alternative, pick the first session of the last days data.. + Day *day=PROFILE.GetDay(PROFILE.LastDay(),MT_CPAP); + if (day) { + int ti=day->first()/1000L; + + cpaptime=QDateTime::fromTime_t(ti); + + qDebug() << "Guessing session starting from CPAP data" << cpaptime; + } else cpaptime=QDateTime(); // null + + qDebug() << "Record Import:" << oxitime << cpaptime; i+=9; } else { started_reading=true; // Sometimes errornous crap is sent after data rec header @@ -884,11 +917,11 @@ void CMS50Serial::startImportTimeout() m_mode=SO_WAIT; cms50timer->stop(); emit(importAborted()); - mainwin->getOximetry()->graphView()->setEmptyText("Import Failed"); + mainwin->getOximetry()->graphView()->setEmptyText(tr("Import Failed")); mainwin->getOximetry()->graphView()->timedRedraw(50); return; } else { - QString a="Waiting"; + QString a=tr("Make Sure Oximeter is Ready"); for (int j=0;jgetOximetry()->graphView()->setEmptyText(a); mainwin->getOximetry()->graphView()->timedRedraw(50); @@ -1001,8 +1034,8 @@ Oximetry::Oximetry(QWidget *parent,gGraphView * shared) : PLETHY->AddLayer(plethy); - PULSE->AddLayer(lo1=new gLineOverlayBar(OXI_PulseChange,QColor("light gray"),"PD",FT_Span)); - SPO2->AddLayer(lo2=new gLineOverlayBar(OXI_SPO2Drop,QColor("light blue"),"O2",FT_Span)); + PULSE->AddLayer(lo1=new gLineOverlayBar(OXI_PulseChange,QColor("light gray"),STR_TR_PC,FT_Span)); + SPO2->AddLayer(lo2=new gLineOverlayBar(OXI_SPO2Drop,QColor("light blue"),STR_TR_O2,FT_Span)); PULSE->AddLayer(pulse); SPO2->AddLayer(spo2); PULSE->setDay(day); @@ -1312,8 +1345,8 @@ void Oximetry::on_ImportButton_clicked() PLETHY->setVisible(false); day->getSessions().clear(); GraphView->setDay(day); - GraphView->setEmptyText("Make Sure Oximeter Is Ready"); - GraphView->redraw(); + //GraphView->setEmptyText(tr("Make Sure Oximeter is Ready")); + //GraphView->redraw(); if (!oximeter->startImport()) { mainwin->Notify(tr("Oximeter Error\n\nThe device did not respond.. Make sure it's switched on.")); @@ -1426,7 +1459,7 @@ void Oximetry::spo2_changed(float o2) void Oximetry::on_saveButton_clicked() { - if (QMessageBox::question(this,"Keep This Recording?","Would you like to save this oximetery session?",QMessageBox::Yes|QMessageBox::No)==QMessageBox::Yes) { + if (QMessageBox::question(this,tr("Keep This Recording?"),tr("Would you like to save this oximetery session?"),QMessageBox::Yes|QMessageBox::No)==QMessageBox::Yes) { Session *session=oximeter->getSession(); // Process??? @@ -1451,7 +1484,7 @@ void Oximetry::on_saveButton_clicked() mainwin->getDaily()->LoadDate(mainwin->getDaily()->getDate()); mainwin->getOverview()->ReloadGraphs(); - GraphView->setEmptyText("No Oximetry Data"); + GraphView->setEmptyText(tr("No Oximetry Data")); GraphView->redraw(); } } @@ -1546,7 +1579,7 @@ bool Oximetry::openSPOFile(QString filename) bool Oximetry::openSPORFile(QString filename) { - //GraphView->setEmptyText("Please Wait"); + //GraphView->setEmptyText(tr("Please Wait")); //GraphView->redraw(); QFile f(filename); if (!f.open(QFile::ReadOnly)) return false; @@ -1623,7 +1656,7 @@ bool Oximetry::openSPORFile(QString filename) void Oximetry::on_openButton_clicked() { if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { - int res=QMessageBox::question(this,"Save Session?","Opening this oximetry file will destroy the current session.\nWould you like to keep it?","Save","Destroy It","Cancel",0,2); + int res=QMessageBox::question(this,tr("Save Session?"),tr("Opening this oximetry file will destroy the current session.\nWould you like to keep it?"),tr("Save"),tr("Destroy It"),tr("Cancel"),0,2); if (res==0) { on_saveButton_clicked(); return; @@ -1633,7 +1666,7 @@ void Oximetry::on_openButton_clicked() } // else it's already saved. QString dir=""; - QFileDialog fd(this,"Select an oximetry file",dir,"Oximetry Files (*.spo *.spoR)"); + QFileDialog fd(this,tr("Select an oximetry file"),dir,tr("Oximetry Files (*.spo *.spoR)")); fd.setAcceptMode(QFileDialog::AcceptOpen); fd.setFileMode(QFileDialog::ExistingFile); if (fd.exec()!=QFileDialog::Accepted) return; @@ -1647,7 +1680,7 @@ void Oximetry::on_openButton_clicked() else if (filename.toLower().endsWith(".spor")) r=openSPORFile(filename); if (!r) { - mainwin->Notify("Couldn't open oximetry file \""+filename+"\"."); + mainwin->Notify(tr("Couldn't open oximetry file \"")+filename+"\"."); } qDebug() << "opening" << filename; } @@ -1678,7 +1711,7 @@ void Oximetry::on_dateEdit_dateTimeChanged(const QDateTime &date) void Oximetry::openSession(Session * session) { if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { - int res=QMessageBox::question(this,"Save Session?","Opening this oximetry session will destroy the unsavedsession in the oximetry tab.\nWould you like to store it first?","Save","Destroy It","Cancel",0,2); + int res=QMessageBox::question(this,tr("Save Session?"),tr("Opening this oximetry session will destroy the unsavedsession in the oximetry tab.\nWould you like to store it first?"),tr("Save"),tr("Destroy It"),tr("Cancel"),0,2); if (res==0) { on_saveButton_clicked(); return; diff --git a/oximetry.h b/oximetry.h index db338d7a..84d11307 100644 --- a/oximetry.h +++ b/oximetry.h @@ -219,6 +219,7 @@ protected: */ class CMS50Serial:public SerialOximeter { + Q_OBJECT public: explicit CMS50Serial(QObject * parent,QString portname); virtual ~CMS50Serial(); @@ -253,7 +254,9 @@ protected: QByteArray data; QByteArray buffer; - QDateTime f2time; + QDateTime oxitime,cpaptime; + + bool cms50dplus; int datasize; int received_bytes; diff --git a/preferencesdialog.cpp b/preferencesdialog.cpp index 68b1308f..af118c29 100644 --- a/preferencesdialog.cpp +++ b/preferencesdialog.cpp @@ -660,8 +660,8 @@ void PreferencesDialog::resetGraphModel() graphModel->setColumnCount(3); QStringList headers; headers.append(tr("Graph")); - headers.append(tr("Min")); - headers.append(tr("Max")); + headers.append(STR_TR_Min); + headers.append(STR_TR_Max); graphModel->setHorizontalHeaderLabels(headers); ui->graphView->setColumnWidth(0,250); ui->graphView->setColumnWidth(1,50); @@ -815,7 +815,7 @@ void PreferencesDialog::on_createSDBackups_toggled(bool checked) break; } } - if (haveS9 && QMessageBox::question(this,"This may not be a good idea","ResMed S9 machines routinely delete certain data from your SD card older than 7 and 30 days (depending on resolution). If you ever need to reimport this data again (whether in SleepyHead or ResScan) this data won't come back. If you need to conserve disk space, please remember to carry out manual backups. Are you sure you want to disable these backups?",QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) { + if (haveS9 && QMessageBox::question(this,tr("This may not be a good idea"),tr("ResMed S9 machines routinely delete certain data from your SD card older than 7 and 30 days (depending on resolution).")+" "+tr("If you ever need to reimport this data again (whether in SleepyHead or ResScan) this data won't come back.")+" "+tr("If you need to conserve disk space, please remember to carry out manual backups.")+" "+tr("Are you sure you want to disable these backups?"),QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) { ui->createSDBackups->setChecked(true); return; } diff --git a/quazip/quazipfile.cpp b/quazip/quazipfile.cpp index 1e859e49..98b4fc95 100644 --- a/quazip/quazipfile.cpp +++ b/quazip/quazipfile.cpp @@ -169,7 +169,7 @@ void QuaZipFilePrivate::setZipError(int zipError) const if(zipError==UNZ_OK) q->setErrorString(QString()); else - q->setErrorString(q->tr("ZIP/UNZIP API error %1").arg(zipError)); + q->setErrorString(QObject::tr("ZIP/UNZIP API error %1").arg(zipError)); } bool QuaZipFile::open(OpenMode mode)