From 3886543c400449ce34616d041a6dbb2c161efd33 Mon Sep 17 00:00:00 2001 From: Guy Scharf Date: Wed, 12 Feb 2020 19:45:12 -0700 Subject: [PATCH 01/12] Make Windows build process work without GIT --- Building/Windows/BuildInstall.iss | 4 ++++ oscar/update_gitinfo.bat | 3 +-- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/Building/Windows/BuildInstall.iss b/Building/Windows/BuildInstall.iss index bd562061..f19368f2 100644 --- a/Building/Windows/BuildInstall.iss +++ b/Building/Windows/BuildInstall.iss @@ -3,6 +3,10 @@ ; ; See DEPLOY.BAT for documentation on how this is used to build OSCAR installer +#define MyGitRevision "unreleased" +#define MyReleaseStatus "" +#define MyVersionNumbers "0.0.0.0" + #include "buildinfo.iss" #define MyAppPublisher "The OSCAR Team" diff --git a/oscar/update_gitinfo.bat b/oscar/update_gitinfo.bat index 6b9314f4..5e695421 100755 --- a/oscar/update_gitinfo.bat +++ b/oscar/update_gitinfo.bat @@ -25,7 +25,6 @@ if errorlevel 1 goto GitFail goto GitDone :GitFail - set GIT_REVISION="private" :GitDone @echo Update_gtinfo.bat: GIT_BRANCH=%GIT_BRANCH%, GIT_REVISION=%GIT_REVISION%, GIT_TAG=%GIT_TAG% @@ -50,4 +49,4 @@ echo Updating %DIR%git_info.h move /y %DIR%git_info.new %DIR%git_info.h :AllDone -endlocal \ No newline at end of file +endlocal From c5c6f779f7198488004dd48814d35fd311cdc38a Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Thu, 13 Feb 2020 14:31:32 -0500 Subject: [PATCH 02/12] Fix Zeo and Dreem loaders' mysteriously missing sessions. This was the same issue seen in the Viatom loader, fixed at 5e07187, and the same fix works here. --- oscar/SleepLib/loader_plugins/dreem_loader.cpp | 6 +++++- oscar/SleepLib/loader_plugins/zeo_loader.cpp | 6 +++++- 2 files changed, 10 insertions(+), 2 deletions(-) diff --git a/oscar/SleepLib/loader_plugins/dreem_loader.cpp b/oscar/SleepLib/loader_plugins/dreem_loader.cpp index 1c4d1d63..6a08dd22 100644 --- a/oscar/SleepLib/loader_plugins/dreem_loader.cpp +++ b/oscar/SleepLib/loader_plugins/dreem_loader.cpp @@ -54,7 +54,11 @@ int DreemLoader::OpenFile(const QString & filename) mach->AddSession(sess); count++; } - mach->Save(); + if (count > 0) { + mach->Save(); + mach->SaveSummaryCache(); + p_profile->StoreMachines(); + } closeCSV(); return count; } diff --git a/oscar/SleepLib/loader_plugins/zeo_loader.cpp b/oscar/SleepLib/loader_plugins/zeo_loader.cpp index 01e0dce9..c9e0c467 100644 --- a/oscar/SleepLib/loader_plugins/zeo_loader.cpp +++ b/oscar/SleepLib/loader_plugins/zeo_loader.cpp @@ -95,7 +95,11 @@ int ZEOLoader::OpenFile(const QString & filename) mach->AddSession(sess); count++; } - mach->Save(); + if (count > 0) { + mach->Save(); + mach->SaveSummaryCache(); + p_profile->StoreMachines(); + } closeCSV(); return count; } From 1a258eae34c6f5201b1845c4d8f40b2dd69cc513 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Thu, 13 Feb 2020 20:37:52 -0500 Subject: [PATCH 03/12] Update machine.info.version after upgrade re-import --- oscar/SleepLib/profiles.cpp | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/oscar/SleepLib/profiles.cpp b/oscar/SleepLib/profiles.cpp index edf94273..13bb6b27 100644 --- a/oscar/SleepLib/profiles.cpp +++ b/oscar/SleepLib/profiles.cpp @@ -565,7 +565,11 @@ void Profile::DataFormatError(Machine *m) } // Note: I deliberately haven't added a Profile help for this if (backups) { - mainwin->importCPAP(ImportPath(m->getBackupPath(), lookupLoader(m)), QObject::tr("Rebuilding from %1 Backup").arg(m->brand())); + MachineLoader * loader = lookupLoader(m); + int c = mainwin->importCPAP(ImportPath(m->getBackupPath(), loader), + QObject::tr("Rebuilding from %1 Backup").arg(m->brand())); + if ( c > 0 ) + m->info.version = loader->Version(); } else { if (!p_profile->session->backupCardData()) { // Automatic backups not available for Intellipap users yet, so don't taunt them.. From a094aee728884e7a51d5020447f054e94d9c59e8 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Fri, 14 Feb 2020 15:18:25 -0500 Subject: [PATCH 04/12] Add sanity check for STR file --- oscar/SleepLib/loader_plugins/resmed_loader.cpp | 12 ++++++++++-- oscar/SleepLib/loader_plugins/resmed_loader.h | 2 +- 2 files changed, 11 insertions(+), 3 deletions(-) diff --git a/oscar/SleepLib/loader_plugins/resmed_loader.cpp b/oscar/SleepLib/loader_plugins/resmed_loader.cpp index 4c815ba4..3efa0da0 100644 --- a/oscar/SleepLib/loader_plugins/resmed_loader.cpp +++ b/oscar/SleepLib/loader_plugins/resmed_loader.cpp @@ -483,7 +483,10 @@ int ResmedLoader::Open(const QString & dirpath) // Build a Date map of all records in STR.edf files, populating ResDayList /////////////////////////////////////////////////////////////////////////////////// - ProcessSTRfiles(mach, STRmap, firstImportDay); + if ( ! ProcessSTRfiles(mach, STRmap, firstImportDay) ) { + qCritical() << "ProcessSTR failed, abandoning this import"; + return 0; + } // We are done with the Parsed STR EDF objects, so delete them for (auto it=STRmap.begin(), end=STRmap.end(); it != end; ++it) { @@ -915,7 +918,7 @@ QString ResmedLoader::Backup(const QString & fullname, const QString & backup_pa // This function parses a list of STR files and creates a date ordered map of individual records -void ResmedLoader::ProcessSTRfiles(Machine *mach, QMap & STRmap, QDate firstImport) +bool ResmedLoader::ProcessSTRfiles(Machine *mach, QMap & STRmap, QDate firstImport) { Q_UNUSED(mach) @@ -971,6 +974,10 @@ void ResmedLoader::ProcessSTRfiles(Machine *mach, QMap & STRmap, if (!maskeventcount) { maskeventcount = str.lookupLabel("MaskEvents"); } + if ( !maskon || !maskoff || !maskeventcount ) { + qCritical() << "Corrupt or untranslated STR.edf file"; + return false; + } EDFSignal *sig = nullptr; @@ -1415,6 +1422,7 @@ void ResmedLoader::ProcessSTRfiles(Machine *mach, QMap & STRmap, #ifdef STR_DEBUG qDebug() << "Finished ProcessSTR"; #endif + return true; } /////////////////////////////////////////////////////////////////////////////////// diff --git a/oscar/SleepLib/loader_plugins/resmed_loader.h b/oscar/SleepLib/loader_plugins/resmed_loader.h index 8736ae67..35802e4b 100644 --- a/oscar/SleepLib/loader_plugins/resmed_loader.h +++ b/oscar/SleepLib/loader_plugins/resmed_loader.h @@ -138,7 +138,7 @@ class ResmedLoader : public CPAPLoader protected: //! \brief The STR.edf file is a unique edf file with many signals - void ProcessSTRfiles(Machine *, QMap &, QDate); + bool ProcessSTRfiles(Machine *, QMap &, QDate); //! \brief Scan for new files to import, group into sessions and add to task que int ScanFiles(Machine * mach, const QString & datalog_path, QDate firstImport); From b96f22bcd98299964224a7015ca36a096ed5b3dc Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Sun, 16 Feb 2020 20:12:25 -0500 Subject: [PATCH 05/12] Fix double import progress window. --- oscar/mainwindow.cpp | 25 ------------------------- 1 file changed, 25 deletions(-) diff --git a/oscar/mainwindow.cpp b/oscar/mainwindow.cpp index 2ec0729b..8195b73b 100644 --- a/oscar/mainwindow.cpp +++ b/oscar/mainwindow.cpp @@ -1064,35 +1064,17 @@ void MainWindow::importCPAPDataCards(const QList & datacards) { bool newdata = false; -// QStringList goodlocations; - - ProgressDialog * prog = new ProgressDialog(this); - prog->setMessage(tr("Processing import list...")); - prog->addAbortButton(); - prog->setWindowModality(Qt::ApplicationModal); - - prog->open(); - int c = -1; for (int i = 0; i < datacards.size(); i++) { QString dir = datacards[i].path; MachineLoader * loader = datacards[i].loader; if (!loader) continue; - connect(loader, SIGNAL(updateMessage(QString)), prog, SLOT(setMessage(QString))); - connect(loader, SIGNAL(setProgressMax(int)), prog, SLOT(setProgressMax(int))); - connect(loader, SIGNAL(setProgressValue(int)), prog, SLOT(setProgressValue(int))); - connect(prog, SIGNAL(abortClicked()), loader, SLOT(abortImport())); - - QPixmap image = loader->getPixmap(loader->PeekInfo(dir).series); - image = image.scaled(64,64); - prog->setPixmap(image); if (!dir.isEmpty()) { c = importCPAP(datacards[i], tr("Importing Data")); qDebug() << "Finished Importing data" << c; if (c >= 0) { - // goodlocations.push_back(dir); QDir d(dir.section("/",0,-1)); (*p_profile)[STR_PREF_LastCPAPPath] = d.absolutePath(); } @@ -1101,19 +1083,12 @@ void MainWindow::importCPAPDataCards(const QList & datacards) newdata = true; } } - disconnect(prog, SIGNAL(abortClicked()), loader, SLOT(abortImport())); - disconnect(loader, SIGNAL(setProgressMax(int)), prog, SLOT(setProgressMax(int))); - disconnect(loader, SIGNAL(setProgressValue(int)), prog, SLOT(setProgressValue(int))); - disconnect(loader, SIGNAL(updateMessage(QString)), prog, SLOT(setMessage(QString))); } if (newdata) { finishCPAPImport(); PopulatePurgeMenu(); } - - prog->close(); - prog->deleteLater(); } From 498451df4a9c907577b385adf1237270aec740e8 Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Sun, 16 Feb 2020 21:02:30 -0500 Subject: [PATCH 06/12] Fix Windows file dialog for loading Viatom files. --- oscar/mainwindow.cpp | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/oscar/mainwindow.cpp b/oscar/mainwindow.cpp index 8195b73b..2a4621a2 100644 --- a/oscar/mainwindow.cpp +++ b/oscar/mainwindow.cpp @@ -2460,6 +2460,10 @@ void MainWindow::on_actionImport_Viatom_Data_triggered() w.setWindowFlags(this->windowFlags() & ~Qt::WindowContextHelpButtonHint); w.setOption(QFileDialog::ShowDirsOnly, false); w.setNameFilters(viatom.getNameFilter()); +#if defined(Q_OS_WIN) + // Windows can't handle this name filter. + w.setOption(QFileDialog::DontUseNativeDialog, true); +#endif if (w.exec() == QFileDialog::Accepted) { QString filename = w.selectedFiles()[0]; From 6fed69e71cc6ecda9afdbfecd29c2992811a3d09 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Sun, 16 Feb 2020 21:12:54 -0500 Subject: [PATCH 07/12] Update database version in loader, not profiles --- oscar/SleepLib/loader_plugins/resmed_loader.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/oscar/SleepLib/loader_plugins/resmed_loader.cpp b/oscar/SleepLib/loader_plugins/resmed_loader.cpp index 4c815ba4..09974798 100644 --- a/oscar/SleepLib/loader_plugins/resmed_loader.cpp +++ b/oscar/SleepLib/loader_plugins/resmed_loader.cpp @@ -368,6 +368,7 @@ int ResmedLoader::Open(const QString & dirpath) Machine *mach = p_profile->lookupMachine(info.serial, info.loadername); if ( mach ) { // we have seen this machine qDebug() << "We have seen this machime"; + mach->setInfo( info ); // update info // QDate lastDate = p_profile->LastDay(MT_CPAP); // firstImportDay = lastDate.addDays(-1); } else { // Starting from new beginnings - new or purged From bd4918dc302ab7a0b2d03cbb2d935c4f2b995fce Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Sun, 16 Feb 2020 21:33:01 -0500 Subject: [PATCH 08/12] Remove version update from importCPAP --- oscar/SleepLib/profiles.cpp | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/oscar/SleepLib/profiles.cpp b/oscar/SleepLib/profiles.cpp index 13bb6b27..4c079452 100644 --- a/oscar/SleepLib/profiles.cpp +++ b/oscar/SleepLib/profiles.cpp @@ -566,10 +566,11 @@ void Profile::DataFormatError(Machine *m) // Note: I deliberately haven't added a Profile help for this if (backups) { MachineLoader * loader = lookupLoader(m); - int c = mainwin->importCPAP(ImportPath(m->getBackupPath(), loader), - QObject::tr("Rebuilding from %1 Backup").arg(m->brand())); - if ( c > 0 ) - m->info.version = loader->Version(); + /* int c = */ + mainwin->importCPAP(ImportPath(m->getBackupPath(), loader), + QObject::tr("Rebuilding from %1 Backup").arg(m->brand())); +// if ( c > 0 ) +// m->info.version = loader->Version(); } else { if (!p_profile->session->backupCardData()) { // Automatic backups not available for Intellipap users yet, so don't taunt them.. From c671e873a32545864990b7d2d043170badfc4ef0 Mon Sep 17 00:00:00 2001 From: Phil Olynyk Date: Sun, 16 Feb 2020 21:41:56 -0500 Subject: [PATCH 09/12] Make Open return error when sanity fails --- oscar/SleepLib/loader_plugins/resmed_loader.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/oscar/SleepLib/loader_plugins/resmed_loader.cpp b/oscar/SleepLib/loader_plugins/resmed_loader.cpp index 3efa0da0..3c529f7c 100644 --- a/oscar/SleepLib/loader_plugins/resmed_loader.cpp +++ b/oscar/SleepLib/loader_plugins/resmed_loader.cpp @@ -485,7 +485,7 @@ int ResmedLoader::Open(const QString & dirpath) if ( ! ProcessSTRfiles(mach, STRmap, firstImportDay) ) { qCritical() << "ProcessSTR failed, abandoning this import"; - return 0; + return -1; } // We are done with the Parsed STR EDF objects, so delete them From bba5f937958fc98b8682339ef999571cbe1bcf72 Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Mon, 17 Feb 2020 09:39:38 -0500 Subject: [PATCH 10/12] Comment out unused slots in MainWindow. --- oscar/mainwindow.cpp | 8 ++++++-- oscar/mainwindow.h | 6 ++++-- 2 files changed, 10 insertions(+), 4 deletions(-) diff --git a/oscar/mainwindow.cpp b/oscar/mainwindow.cpp index 2a4621a2..26c1a889 100644 --- a/oscar/mainwindow.cpp +++ b/oscar/mainwindow.cpp @@ -1327,12 +1327,14 @@ void MainWindow::on_action_Reset_Graph_Layout_triggered() if (overview && (ui->tabWidget->currentWidget() == overview)) { overview->ResetGraphLayout(); } } +/* void MainWindow::on_action_Reset_Graph_Order_triggered() { if (daily && (ui->tabWidget->currentWidget() == daily)) { daily->ResetGraphOrder(0); } if (overview && (ui->tabWidget->currentWidget() == overview)) { overview->ResetGraphOrder(0); } } +*/ void MainWindow::on_action_Standard_Graph_Order_triggered() { @@ -1417,14 +1419,14 @@ void MainWindow::CheckForUpdates() #endif } +#ifndef NO_UPDATER void MainWindow::on_actionCheck_for_Updates_triggered() { qDebug() << "procedure called"; -#ifndef NO_UPDATER UpdaterWindow *w = new UpdaterWindow(this); w->checkForUpdates(); -#endif } +#endif bool toolbox_visible = false; void MainWindow::on_action_Screenshot_triggered() @@ -2399,11 +2401,13 @@ void MainWindow::on_actionSleep_Disorder_Terms_Glossary_triggered() QDesktopServices::openUrl(QUrl("https://www.apneaboard.com/wiki/index.php?title=Definitions")); } +/* void MainWindow::on_actionHelp_Support_OSCAR_Development_triggered() { // QDesktopServices().openUrl(QUrl("https://sleepyhead.jedimark.net/donate.php")); QMessageBox::information(nullptr, STR_MessageBox_Information, tr("Donations are not implemented")); } +*/ void MainWindow::on_actionChange_Language_triggered() { diff --git a/oscar/mainwindow.h b/oscar/mainwindow.h index 17e6d800..9d91ed3a 100644 --- a/oscar/mainwindow.h +++ b/oscar/mainwindow.h @@ -214,7 +214,7 @@ class MainWindow : public QMainWindow void on_action_Reset_Graph_Layout_triggered(); //! \brief passes the ResetGraphOrder menu click to the Daily & Overview views - void on_action_Reset_Graph_Order_triggered(); + //void on_action_Reset_Graph_Order_triggered(); //! \brief passes the ResetGraphOrder menu click to the Daily & Overview views void on_action_Standard_Graph_Order_triggered(); @@ -229,7 +229,9 @@ class MainWindow : public QMainWindow void on_oximetryButton_clicked(); //! \brief Creates the UpdaterWindow object that actually does the real check for updates +#ifndef NO_UPDATER void on_actionCheck_for_Updates_triggered(); +#endif //! \brief Attempts to do a screenshot of the application window void on_action_Screenshot_triggered(); @@ -287,7 +289,7 @@ class MainWindow : public QMainWindow void on_actionSleep_Disorder_Terms_Glossary_triggered(); - void on_actionHelp_Support_OSCAR_Development_triggered(); + //void on_actionHelp_Support_OSCAR_Development_triggered(); void aboutBoxLinkClicked(const QUrl &url); From e698879e177e4149427d23f67b4417e4c8723bd7 Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Mon, 17 Feb 2020 10:59:26 -0500 Subject: [PATCH 11/12] Move graphic engine configuration before creation of QApplication. Qt was emitting console warnings about attempts to change Qt::AA_UseDesktopOpenGL after creating the QApplication. --- oscar/main.cpp | 55 +++++++++++++++++++++++++++++--------------------- 1 file changed, 32 insertions(+), 23 deletions(-) diff --git a/oscar/main.cpp b/oscar/main.cpp index 6418d33a..f5254ddf 100644 --- a/oscar/main.cpp +++ b/oscar/main.cpp @@ -264,16 +264,37 @@ int main(int argc, char *argv[]) { QSettings settings; - QApplication a(argc, argv); - QStringList args = a.arguments(); - // If shift key was held down when OSCAR was launched, force Software graphics Engine (aka LegacyGFX) - Qt::KeyboardModifiers keymodifier = QApplication::queryKeyboardModifiers(); + Qt::KeyboardModifiers keymodifier = QApplication::keyboardModifiers(); QString forcedEngine = ""; if (keymodifier == Qt::ShiftModifier){ settings.setValue(GFXEngineSetting, (unsigned int)GFX_Software); forcedEngine = "Software Engine forced by shift key at launch"; } + // This argument needs to be processed before creating the QApplication, + // based on sample code at https://doc.qt.io/qt-5/qapplication.html#details + for (int i = 1; i < argc; ++i) { + if (!qstrcmp(argv[i], "--legacy")) { + settings.setValue(GFXEngineSetting, (unsigned int)GFX_Software); + forcedEngine = "Software Engine forced by --legacy command line switch"; + } + } + GFXEngine gfxEngine = (GFXEngine)qMin((unsigned int)settings.value(GFXEngineSetting, (unsigned int)GFX_OpenGL).toUInt(), (unsigned int)MaxGFXEngine); + switch (gfxEngine) { + case 0: // GFX_OpenGL + QCoreApplication::setAttribute(Qt::AA_UseDesktopOpenGL); + break; + case 1: // GFX_ANGLE + QCoreApplication::setAttribute(Qt::AA_UseOpenGLES); + break; + case 2: // GFX_Software + default: + QCoreApplication::setAttribute(Qt::AA_UseSoftwareOpenGL); + } + + QApplication a(argc, argv); + QStringList args = a.arguments(); + QString lastlanguage = settings.value(LangSetting, "").toString(); if (lastlanguage.compare("is", Qt::CaseInsensitive)) // Convert code for Hebrew from 'is' to 'he' @@ -296,10 +317,7 @@ int main(int argc, char *argv[]) { changing_language = true; // reset to force language dialog settings.setValue(LangSetting,""); } - else if (args[i] == "--legacy") { - settings.setValue(GFXEngineSetting, (unsigned int)GFX_Software); - forcedEngine = "Software Engine forced by --legacy command line switch"; - } + // "--legacy" is handle above, as it needs to be processed before creating QApplication. else if (args[i] == "-p") QThread::msleep(1000); else if (args[i] == "--profile") { @@ -324,22 +342,13 @@ int main(int argc, char *argv[]) { } } // end of for args loop - - GFXEngine gfxEngine = (GFXEngine)qMin((unsigned int)settings.value(GFXEngineSetting, (unsigned int)GFX_OpenGL).toUInt(), (unsigned int)MaxGFXEngine); - - switch (gfxEngine) { - case 0: - QCoreApplication::setAttribute(Qt::AA_UseDesktopOpenGL); - break; - case 1: - QCoreApplication::setAttribute(Qt::AA_UseOpenGLES); - break; - case 2: - default: - QCoreApplication::setAttribute(Qt::AA_UseSoftwareOpenGL); - } - initializeLogger(); + // After initializing the logger, any qDebug() messages will be queued but not written to console + // until MainWindow is constructed below. In spite of that, we initialize the logger here so that + // the intervening messages to show up in the debug pane. + // + // The only time this is really noticeable is when initTranslations() presents its language + // selection QDialog, which waits indefinitely for user input before MainWindow is constructed. qDebug().noquote() << "OSCAR starting" << QDateTime::currentDateTime().toString(); From d75a2e9fca1809146a50131c143fa8d636420cfb Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Mon, 17 Feb 2020 11:04:48 -0500 Subject: [PATCH 12/12] Fix debug message when skipping short sessions. The session ID is not a date on all machines. --- oscar/SleepLib/machine.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/oscar/SleepLib/machine.cpp b/oscar/SleepLib/machine.cpp index e765d9f5..9c128c27 100644 --- a/oscar/SleepLib/machine.cpp +++ b/oscar/SleepLib/machine.cpp @@ -387,7 +387,7 @@ bool Machine::AddSession(Session *s) if (session_length < ignore_sessions) { // keep the session to save importing it again, but don't add it to the day record this time qDebug() << s->session() << "Ignoring short session <" << ignore_sessions - << "["+QDateTime::fromTime_t(s->session()).toString("MMM dd, yyyy hh:mm:ss")+"]"; + << "["+QDateTime::fromMSecsSinceEpoch(s->first()).toString("MMM dd, yyyy hh:mm:ss")+"]"; return true; }