diff --git a/README b/README index 36c088d9..562c0a39 100644 --- a/README +++ b/README @@ -1,23 +1,26 @@ OpenSource CPAP Analysis Reporter - OSCAR -OSCAR started as a maintenance fork of SleepyHead version 1.1.0 when that was abandoned by Mark Watkins. +OSCAR is a derivative of SleepyHead version 1.1.0,created when that was abandoned by Mark Watkins. -SleepyHead was cross platform, opensource sleep tracking program for reviewing CPAP and Oximetry data, -which are devices used in the treatment of Sleep Disorders like Obstructive Sleep Apnea. It was released +SleepyHead was a cross platform, opensource sleep tracking program for reviewing CPAP and Oximetry data, +which are devices used in the treatment of Sleep Disorders like Obstructive Sleep Apnea. It was released under the GPL version 3 license. See the file COPYING for those details. SleepyHead was written by Mark Watkins (aka Jedimark), an Australian software developer afflicted with sleep apnea. SleepyHead was copyright (C) 2011-2018 by Mark Watkins -Most of OSCAR is copyright (c) 2011-2018 by Mark Watkins Portions of OSCAR are copyright (c) 2019 by Nightowl Software -Requirements: +Requirements: (Linux package names are given, Windows and Mac should be similar) ------------- Qt5.9 SDK with help (Qt5.9 from http://qt.io/download-open-source recommended) -Qt5.7 SDL without help is possible (for Debian Stretch) -Linux needs libudev-dev for qserialport to compile +The help modules are libqt5help5, qttools5-dev and qttools5-dev-tools + +Qt5.7 SDK without help is possible (for Debian Stretch) + +All systems need qt5-default, libqt5serialport5, libqt5serailport5-dev and libqt5opengl5-dev +Linux also needs libudev-dev and zlib1g-dev OSX needs Xcode and commandline tools from AppStore, @@ -31,7 +34,7 @@ Recommend shadow building to not cruft up the source code folder: cd .. mkdir build cd build -qmake ../oscr-code/OSCAR_QT.pro +qmake ../oscar-code/OSCAR_QT.pro make You may need to add a -spec option to qmake to suit your platform. @@ -59,17 +62,14 @@ in redistribution of derivatives. It is built using Qt SDK (Open Source Edition), available from http://qt.io. -It also uses QuaZip, by Sergey A. Tachenov, which is a C++ wrapper over Gilles Vollant's ZIP/UNZIP package -http://sourceforge.net/projects/quazip/ - Redistribution of derivatives ( a note added by Mark Watins ) ----------------------------- -Mark Watkins created this software to help lessen the exploitation of others. Seeing his work being used to exploit others +Mark Watkins created this software to help lessen the exploitation of others. Seeing his work being used to exploit others is incredibly un-motivational, and incredibly disrespectful of all the work he put into this project. -If you plan on reselling any derivatives of SleepyHead, I specifically request that you give due credit and -link back, mentioning clearly in your advertising material, software installer and about screens that your -derivative "is based on the free and open-source software SleepyHead available from http://sleepyhead.jedimark.net, +If you plan on reselling any derivatives of SleepyHead, I specifically request that you give due credit and +link back, mentioning clearly in your advertising material, software installer and about screens that your +derivative "is based on the free and open-source software SleepyHead available from http://sleepyhead.jedimark.net, developed and copyright by Mark Watkins (C) 2011-2018." It is not enough to reference that your derivative "is based on GPL software". diff --git a/oscar/Graphs/gGraphView.cpp b/oscar/Graphs/gGraphView.cpp index c50275b0..829ed9c4 100644 --- a/oscar/Graphs/gGraphView.cpp +++ b/oscar/Graphs/gGraphView.cpp @@ -331,6 +331,7 @@ gGraphView::gGraphView(QWidget *parent, gGraphView *shared) this->setMouseTracking(true); m_emptytext = STR_Empty_NoData; + this->setEmptyImage(QPixmap(":/icons/logo.png")); InitGraphGlobals(); // FIXME: sstangl: handle error return. #ifdef ENABLE_THREADED_DRAWING m_idealthreads = QThread::idealThreadCount(); @@ -630,7 +631,7 @@ bool gGraphView::pinchTriggered(QPinchGesture * gesture) Q_UNUSED(group) - //qDebug() << gesture << gesture->scaleFactor(); +// qDebug() << gesture << gesture->scaleFactor(); if (gesture->state() == Qt::GestureStarted) { pinch_min = m_minx; pinch_max = m_maxx; @@ -1371,7 +1372,7 @@ void gGraphView::paintGL() if (height() <= 0) { return; } - // Create QPainter object, note this is only valid from paintGL events! +// Create QPainter object, note this is only valid from paintGL events! QPainter painter(this); painter.setRenderHint(QPainter::HighQualityAntialiasing, true); @@ -1387,9 +1388,18 @@ void gGraphView::paintGL() strings_drawn_this_frame = 0; strings_cached_this_frame = 0; +// qDebug() << "About to call renderGraphs from paintGL()"; +// sleep(3); + graphs_drawn = renderGraphs(painter); +// qDebug() << "Finished call renderGraphs from paintGL()"; +// sleep(3); + if (!graphs_drawn) { // No graphs drawn? show something useful :) + qDebug() << "No graphs drawn"; +// sleep(3); + QString txt; if (m_showAuthorMessage) { if (emptyText() == STR_Empty_Brick) { @@ -1399,18 +1409,38 @@ void gGraphView::paintGL() txt = QObject::tr("There is no data to graph"); } } + qDebug() << txt; +// sleep(3); + // int x2, y2; // GetTextExtent(m_emptytext, x2, y2, bigfont); // int tp2, tp1; - if (!m_emptyimage.isNull()) { - painter.drawPixmap(width() /2 - m_emptyimage.width() /2, height() /2 - m_emptyimage.height() /2 , m_emptyimage); + if (this->m_emptyimage.isNull() ) + qDebug() << "m_emptyimage is NULL"; + else + qDebug() << "m_emptyimage is not NULL"; + sleep(3); + + if (! this->m_emptyimage.isNull()) { + int x = width()/2 - this->m_emptyimage.width()/2; + int y = height()/2 - this->m_emptyimage.height()/2; +// qDebug() << "size is " + QString::number(m_emptyimage.width()) + " by " + QString::number(m_emptyimage.height()); +// qDebug() << "x-pos: " + QString::number(x) + " y-pos: " + QString::number(y); +// sleep(3); + + QRectF target(QRect(x, y, this->m_emptyimage.width(), this->m_emptyimage.height())); + QRectF source(QRect(0, 0, this->m_emptyimage.width(), this->m_emptyimage.height())); + + painter.drawPixmap(target, this->m_emptyimage, source); + // tp2 = height() /2 + m_emptyimage.height()/2 + y2; - } /*else { - tp2 = height() / 2 + y2; }*/ +// qDebug() << "Break 9"; +// sleep(3); + QColor col = Qt::black; painter.setPen(col); @@ -1418,12 +1448,27 @@ void gGraphView::paintGL() // painter.drawText((width() / 2) - x2 / 2, tp2, m_emptytext); QRectF rec(0,0,width(),0); +// if (defaultfont == nullptr) +// qDebug() << "Default font is NULL"; +// else +// qDebug() << "Default font is not NULL"; +// sleep(3); + painter.setFont(*defaultfont); rec = painter.boundingRect(rec, Qt::AlignHCenter | Qt::AlignBottom, txt); rec.moveBottom(height()-5); +// qDebug() << "Break 8"; +// sleep(3); + painter.drawText(rec, Qt::AlignHCenter | Qt::AlignBottom, txt); +// qDebug() << "Break 7"; +// sleep(3); + } +// qDebug() << "Break 1"; +// sleep(3); + if (AppSetting->lineCursorMode()) { emit updateCurrentTime(graphs_drawn ? m_currenttime : 0.0F); } else { @@ -1431,8 +1476,14 @@ void gGraphView::paintGL() } AppSetting->usePixmapCaching() ? DrawTextQueCached(painter) :DrawTextQue(painter); +// qDebug() << "Break 2"; +// sleep(3); + m_tooltip->paint(painter); +// qDebug() << "Break 3"; +// sleep(3); + #ifdef DEBUG_EFFICIENCY const int rs = 20; static double ring[rs] = {0}; @@ -1482,6 +1533,9 @@ void gGraphView::paintGL() painter.end(); +// qDebug() << "Break 4"; +// sleep(3); + #ifndef BROKEN_OPENGL_BUILD #if QT_VERSION < QT_VERSION_CHECK(5,4,0) swapBuffers(); @@ -1493,6 +1547,9 @@ void gGraphView::paintGL() redrawtimer->start(); } +// qDebug() << "Break 5"; +// sleep(3); + } QString gGraphView::getRangeString() diff --git a/oscar/Graphs/gGraphView.h b/oscar/Graphs/gGraphView.h index 0c0ecd0b..d35ce4ff 100644 --- a/oscar/Graphs/gGraphView.h +++ b/oscar/Graphs/gGraphView.h @@ -82,7 +82,7 @@ protected: enum FlagType { FT_Bar, FT_Dot, FT_Span }; -void setEmptyImage(QString text, QPixmap pixmap); +//void setEmptyImage(QString text, QPixmap pixmap); class MyLabel:public QWidget { diff --git a/oscar/SleepLib/day.h b/oscar/SleepLib/day.h index c21bae3b..bb74e48c 100644 --- a/oscar/SleepLib/day.h +++ b/oscar/SleepLib/day.h @@ -267,7 +267,7 @@ class Day return (c * 60.0) / minutes; } - //! \brief SleepyyHead Events Index, AHI combined with SleepyHead detected events.. :) + //! \brief SleepyyHead Events Index, AHI combined with OSCAR detected events.. :) EventDataType calcSHEI() { EventDataType c = count(CPAP_Hypopnea) + count(CPAP_Obstructive) + count(CPAP_Apnea) + count(CPAP_ClearAirway) + count(CPAP_UserFlag1) + count(CPAP_UserFlag2); EventDataType minutes = hours(MT_CPAP) * 60.0; diff --git a/oscar/SleepLib/machine.cpp b/oscar/SleepLib/machine.cpp index d5c7a7cc..f0c61924 100644 --- a/oscar/SleepLib/machine.cpp +++ b/oscar/SleepLib/machine.cpp @@ -561,6 +561,7 @@ const QString Machine::getEventsPath() } const QString Machine::getBackupPath() { + qDebug() << "Backup Path is " + getDataPath() + "Backup/"; return getDataPath() + "Backup/"; } diff --git a/oscar/SleepLib/machine_common.h b/oscar/SleepLib/machine_common.h index 511d73e9..f8650699 100644 --- a/oscar/SleepLib/machine_common.h +++ b/oscar/SleepLib/machine_common.h @@ -31,7 +31,7 @@ class BoundsError {}; //! \brief Exception class for to trap old database versions. class OldDBVersion {}; -const quint32 magic = 0xC73216AB; // Magic number for SleepyHead Data Files.. Don't touch! +const quint32 magic = 0xC73216AB; // Magic number for OSCAR Data Files.. Don't touch! //const int max_number_event_fields=10; // This should probably move somewhere else diff --git a/oscar/SleepLib/preferences.cpp b/oscar/SleepLib/preferences.cpp index c5016296..215f0d59 100644 --- a/oscar/SleepLib/preferences.cpp +++ b/oscar/SleepLib/preferences.cpp @@ -16,6 +16,7 @@ #include #include #include +#include #ifdef Q_OS_WIN #include "windows.h" @@ -194,6 +195,10 @@ bool Preferences::Open(QString filename) QDomElement root = doc.documentElement(); if (root.tagName() != STR_AppName) { + if (root.tagName() == "SleepyHead" ) { + QMessageBox::warning(nullptr, STR_MessageBox_Error, QObject::tr( "You must run the OSCAR Migration Tool"), QMessageBox::Ok); + exit(1); + } return false; } diff --git a/oscar/SleepLib/profiles.cpp b/oscar/SleepLib/profiles.cpp index 69cb1b46..df2d3dc1 100644 --- a/oscar/SleepLib/profiles.cpp +++ b/oscar/SleepLib/profiles.cpp @@ -164,7 +164,7 @@ bool Profile::OpenMachines() QDomElement root = doc.firstChild().toElement(); if (root.tagName().toLower() != "machines") { - //qDebug() << "No Machines Tag in Profiles.xml"; + qDebug() << "No Machines Tag in machines.xml"; return false; } diff --git a/oscar/UpdaterWindow.cpp b/oscar/UpdaterWindow.cpp index e3e5cbfa..1ddc0651 100644 --- a/oscar/UpdaterWindow.cpp +++ b/oscar/UpdaterWindow.cpp @@ -480,7 +480,7 @@ void UpdaterWindow::ParseUpdatesXML(QIODevice *dev) for (it = updatesparser.packages.begin(); it!=updatesparser.packages.end(); ++it) { const PackageUpdate & update = it.value(); - if (it.key() == UPDATE_SleepyHead) { + if (it.key() == UPDATE_OSCAR) { if (compareVersion(update.versionString)>0) { updateList.push_back(update); } @@ -501,14 +501,14 @@ void UpdaterWindow::ParseUpdatesXML(QIODevice *dev) } if (updateList.size()==0) { - mainwin->Notify(tr("No updates were found for your platform."), tr("SleepyHead Updates"), 5000); + mainwin->Notify(tr("No updates were found for your platform."), tr("OSCAR Updates"), 5000); PREF[STR_GEN_UpdatesLastChecked] = QDateTime::currentDateTime(); close(); return; } else { - if (QMessageBox::question(mainwin, tr("SleepyHead Updates"), - tr("New SleepyHead Updates are avilable:")+"\n\n"+ + if (QMessageBox::question(mainwin, tr("OSCAR Updates"), + tr("New OSCAR Updates are avilable:")+"\n\n"+ tr("Would you like to download and install them now?"), QMessageBox::Yes, QMessageBox::No) == QMessageBox::Yes) { diff --git a/oscar/UpdaterWindow.h b/oscar/UpdaterWindow.h index 408cb2f0..428a9bc5 100644 --- a/oscar/UpdaterWindow.h +++ b/oscar/UpdaterWindow.h @@ -30,9 +30,9 @@ enum RequestMode { RM_None, RM_CheckUpdates, RM_GetFile }; /*! \class UpdaterWindow - \brief Auto-Update Module for SleepyHead + \brief Auto-Update Module for OSCAR - This class handles the complete Auto-Update procedure for SleepyHead, it does the network checks, + This class handles the complete Auto-Update procedure for OSCAR, it does the network checks, parses the update.xml from SourceForge host, checks for any new updates, and provides the UI and mechanisms to download and replace the binaries according to what is specified in update.xml. */ diff --git a/oscar/aboutdialog.cpp b/oscar/aboutdialog.cpp index 653cb19f..afaee3d4 100644 --- a/oscar/aboutdialog.cpp +++ b/oscar/aboutdialog.cpp @@ -1,4 +1,4 @@ -/* SleepyHead AboutDialog Implementation +/* OSCAR AboutDialog Implementation * * Date created: 7/5/2018 * @@ -93,7 +93,7 @@ QString AboutDialog::getCredits() QString AboutDialog::getRelnotes() { QFile clfile(":/docs/release_notes.html"); - QString changeLog = tr("Sorry, could not locate changelog."); + QString changeLog = tr("Sorry, could not locate Release Notes."); if (clfile.open(QIODevice::ReadOnly)) { //Todo, write XML parser and only show the latest.. //QTextStream ts(&clfile); diff --git a/oscar/aboutdialog.h b/oscar/aboutdialog.h index a2a933aa..b3f7e6a4 100644 --- a/oscar/aboutdialog.h +++ b/oscar/aboutdialog.h @@ -1,4 +1,4 @@ -/* SleepyHead AboutDialog Header +/* OSCAR AboutDialog Header * * Copyright (c) 2018 Mark Watkins * diff --git a/oscar/daily.cpp b/oscar/daily.cpp index 1a64bb60..8963a919 100644 --- a/oscar/daily.cpp +++ b/oscar/daily.cpp @@ -69,6 +69,7 @@ void Daily::setSidebarVisible(bool visible) Daily::Daily(QWidget *parent,gGraphView * shared) :QWidget(parent), ui(new Ui::Daily) { + qDebug() << "Creating new Daily object"; ui->setupUi(this); // Remove Incomplete Extras Tab @@ -132,8 +133,12 @@ Daily::Daily(QWidget *parent,gGraphView * shared) ui->graphMainArea->setAutoFillBackground(false); GraphView=new gGraphView(ui->graphFrame,shared); +// qDebug() << "New GraphView object created in Daily"; +// sleep(3); GraphView->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding); + GraphView->setEmptyImage(QPixmap(":/icons/logo.png")); + snapGV=new gGraphView(GraphView); snapGV->setMinimumSize(172,172); snapGV->hideSplitter(); @@ -196,6 +201,14 @@ Daily::Daily(QWidget *parent,gGraphView * shared) graphlist[schema::channel[code].code()] = new gGraph(schema::channel[code].code(), GraphView, schema::channel[code].label(), channelInfo(code), default_height); } + if ( p_profile == nullptr ) { + qDebug() << "In daily, p_profile is NULL"; + return; + } + else if (p_profile->general == nullptr ) { + qDebug() << "In daily, p_profile->general is NULL"; + return; + } if (p_profile->general->calculateRDI()) { AHI=new gGraph("AHI", GraphView,STR_TR_RDI, channelInfo(CPAP_RDI), default_height); } else { @@ -406,6 +419,8 @@ Daily::Daily(QWidget *parent,gGraphView * shared) ui->weightSpinBox->setDecimals(3); ui->weightSpinBox->setSuffix(STR_UNIT_KG); } + + GraphView->setEmptyImage(QPixmap(":/icons/logo.png")); GraphView->setEmptyText(STR_Empty_NoData); previous_date=QDate(); @@ -418,8 +433,8 @@ Daily::Daily(QWidget *parent,gGraphView * shared) connect(GraphView, SIGNAL(updateCurrentTime(double)), this, SLOT(on_LineCursorUpdate(double))); connect(GraphView, SIGNAL(updateRange(double,double)), this, SLOT(on_RangeUpdate(double,double))); connect(GraphView, SIGNAL(GraphsChanged()), this, SLOT(updateGraphCombo())); - - GraphView->setEmptyImage(QPixmap(":/icons/logo.png")); +// qDebug() << "Finished making new Daily object"; +// sleep(3); } Daily::~Daily() @@ -444,7 +459,11 @@ Daily::~Daily() void Daily::showEvent(QShowEvent *) { +// qDebug() << "Start showEvent in Daily object"; +// sleep(3); RedrawGraphs(); +// qDebug() << "Finished showEvent Daily object"; +// sleep(3); } void Daily::doToggleSession(Session * sess) @@ -522,6 +541,8 @@ void Daily::Link_clicked(const QUrl &url) void Daily::ReloadGraphs() { +// qDebug() << "Start ReloadGraphs Daily object"; +// sleep(3); GraphView->setDay(nullptr); ui->splitter->setVisible(true); @@ -547,6 +568,8 @@ void Daily::ReloadGraphs() Load(d); ui->calButton->setText(ui->calendar->selectedDate().toString(Qt::TextDate)); graphView()->redraw(); +// qDebug() << "Finished ReloadGraphs in Daily object"; +// sleep(3); } void Daily::hideSpaceHogs() @@ -1578,6 +1601,8 @@ void Daily::Load(QDate date) ui->BMI->display(0); ui->BMI->setVisible(false); ui->BMIlabel->setVisible(false); + ui->toggleGraphs->setVisible(false); + ui->toggleEvents->setVisible(false); BookmarksChanged=false; Session *journal=GetJournalSession(date); @@ -2291,11 +2316,12 @@ void Daily::updateCube() { //brick.. if ((GraphView->visibleGraphs()==0)) { - ui->toggleGraphs->setArrowType(Qt::UpArrow); - ui->toggleGraphs->setToolTip(tr("Show all graphs")); - ui->toggleGraphs->blockSignals(true); - ui->toggleGraphs->setChecked(true); - ui->toggleGraphs->blockSignals(false); + ui->toggleGraphs->setVisible(false); +// ui->toggleGraphs->setArrowType(Qt::UpArrow); +// ui->toggleGraphs->setToolTip(tr("Show all graphs")); +// ui->toggleGraphs->blockSignals(true); +// ui->toggleGraphs->setChecked(true); +// ui->toggleGraphs->blockSignals(false); if (ui->graphCombo->count() > 0) { GraphView->setEmptyText(STR_Empty_NoGraphs); @@ -2312,11 +2338,12 @@ void Daily::updateCube() } } } else { - ui->toggleGraphs->setArrowType(Qt::DownArrow); - ui->toggleGraphs->setToolTip(tr("Hide all graphs")); - ui->toggleGraphs->blockSignals(true); - ui->toggleGraphs->setChecked(false); - ui->toggleGraphs->blockSignals(false); + ui->toggleGraphs->setVisible(false); +// ui->toggleGraphs->setArrowType(Qt::DownArrow); +// ui->toggleGraphs->setToolTip(tr("Hide all graphs")); +// ui->toggleGraphs->blockSignals(true); +// ui->toggleGraphs->setChecked(false); +// ui->toggleGraphs->blockSignals(false); } } @@ -2325,6 +2352,12 @@ void Daily::on_toggleGraphs_clicked(bool checked) { QString s; QIcon *icon=checked ? icon_off : icon_on; + if (ui->graphCombo == nullptr ) + qDebug() << "ToggleGraphs clicked with null graphCombo ptr"; + else + qDebug() << "ToggleGraphs clicked with non-null graphCombo ptr"; + return; +/* for (int i=0;igraphCombo->count();i++) { s=ui->graphCombo->itemText(i); ui->graphCombo->setItemIcon(i,*icon); @@ -2337,7 +2370,7 @@ void Daily::on_toggleGraphs_clicked(bool checked) updateCube(); GraphView->updateScale(); GraphView->redraw(); - + */ } void Daily::updateGraphCombo() @@ -2384,6 +2417,12 @@ void Daily::on_toggleEvents_clicked(bool checked) QString s; QIcon *icon=checked ? icon_on : icon_off; + if (ui->toggleEvents == nullptr ) + qDebug() << "ToggleEvents clicked with null toggleEvents ptr"; + else + qDebug() << "ToggleEvents clicked with non-null toggleEvents ptr"; + return; +/* ui->toggleEvents->setArrowType(checked ? Qt::DownArrow : Qt::UpArrow); ui->toggleEvents->setToolTip(checked ? tr("Hide all events") : tr("Show all events")); // ui->toggleEvents->blockSignals(true); @@ -2399,6 +2438,7 @@ void Daily::on_toggleEvents_clicked(bool checked) updateCube(); GraphView->redraw(); + */ } //void Daily::on_sessionWidget_itemSelectionChanged() @@ -2418,6 +2458,6 @@ void Daily::on_splitter_2_splitterMoved(int, int) { int size = ui->splitter_2->sizes()[0]; if (size == 0) return; - qDebug() << "Left Panel width set to " << size; +// qDebug() << "Left Panel width set to " << size; AppSetting->setDailyPanelWidth(size); } diff --git a/oscar/daily.h b/oscar/daily.h index 831f9e8c..7b0a9254 100644 --- a/oscar/daily.h +++ b/oscar/daily.h @@ -38,7 +38,7 @@ class MainWindow; /*! \class Daily - \brief SleepyHead's Daily view which displays the calendar and all the graphs relative to a selected Day + \brief OSCAR's Daily view which displays the calendar and all the graphs relative to a selected Day */ class Daily : public QWidget { diff --git a/oscar/docs/about.html b/oscar/docs/about.html index d08fb914..c1ba3408 100644 --- a/oscar/docs/about.html +++ b/oscar/docs/about.html @@ -9,7 +9,7 @@


-

Welcome to Open Source CPAP Analysis Reporter

+

Welcome to the Open Source CPAP Analysis Reporter


This software has been designed to assist you in reviewing the data produced by your CPAP machines and related equipment.


@@ -28,8 +28,8 @@

OSCAR is free (as in freedom) software, released under the GNU Public License v3, and comes with no warranty, and without ANY claims to fitness for any purpose.


-

OSCAR is copyright ©2011-2018 Mark Watkins

-

and portions ©2019 Nightowl Software

+

Portions of OSCAR are ©2019 Nightowl Software and members of the apnea community as listed in the git log

+

OSCAR is a derivative of the SleeyHead program which is copyright ©2011-2018 Mark Watkins


diff --git a/oscar/docs/release_notes.html b/oscar/docs/release_notes.html index 1725436f..bb011605 100644 --- a/oscar/docs/release_notes.html +++ b/oscar/docs/release_notes.html @@ -1,10 +1,16 @@  -

+

+OSCAR is based on SleepyHead v1.1.0-unstable-0 +Which was written and copyright 2011-2018 © Mark Watkins +

+ +

Changes and fixes in OSCAR v1.0.0-beta-1 -

  • This is a Community Release, prepared without Mark Watkin's assistance
  • +
  • Portions of OSCAR are © 2019 by Nightowl Software
  • +
  • and members of the apnea community
  • The Donation and Update features have been removed
  • The Glossary, FAQ, and Wiki features are not yet implemented
  • The Help Pages require Qt 5.9 to build; Qt 5.7 is the oldest supported
  • @@ -19,10 +25,5 @@


    -

    -OSCAR is based on SleepyHead v1.1.0-unstable-0 -Which was written and copyright 2011-2018 © Mark Watkins -

    - diff --git a/oscar/help.h b/oscar/help.h index 54dfee47..1916461e 100644 --- a/oscar/help.h +++ b/oscar/help.h @@ -1,4 +1,4 @@ -/* SleepyHead Help Header +/* OSCAR Help Header * * Copyright (c) 2018 Mark Watkins * diff --git a/oscar/logger.cpp b/oscar/logger.cpp index b5bf2cf9..8bfc1731 100644 --- a/oscar/logger.cpp +++ b/oscar/logger.cpp @@ -1,4 +1,4 @@ -/* SleepyHead Logger module implementation +/* OSCAR Logger module implementation * * Copyright (c) 2011-2018 Mark Watkins * diff --git a/oscar/main.cpp b/oscar/main.cpp index 98c11801..07d01790 100644 --- a/oscar/main.cpp +++ b/oscar/main.cpp @@ -1,4 +1,4 @@ -/* SleepyHead Main +/* OSCAR Main * * Copyright (c) 2011-2018 Mark Watkins * diff --git a/oscar/mainwindow.cpp b/oscar/mainwindow.cpp index dbf21a81..b8983eaf 100644 --- a/oscar/mainwindow.cpp +++ b/oscar/mainwindow.cpp @@ -1,4 +1,4 @@ -/* SleepyHead MainWindow Implementation +/* OSCAR MainWindow Implementation * * Copyright (c) 2011-2018 Mark Watkins * @@ -84,6 +84,7 @@ MainWindow::MainWindow(QWidget *parent) : // Nifty Notification popups in System Tray (uses Growl on Mac) if (QSystemTrayIcon::isSystemTrayAvailable() && QSystemTrayIcon::supportsMessages()) { + qDebug() << "Using System Tray for Menu"; systray = new QSystemTrayIcon(QIcon(":/icons/logo.png"), this); systray->show(); systraymenu = new QMenu(this); @@ -96,6 +97,7 @@ MainWindow::MainWindow(QWidget *parent) : systraymenu->addSeparator(); systraymenu->addAction(tr("E&xit"), this, SLOT(close())); } else { // if not available, the messages will popup in the taskbar + qDebug() << "No System Tray menues"; systray = nullptr; systraymenu = nullptr; } @@ -254,6 +256,7 @@ void MainWindow::closeEvent(QCloseEvent * event) DestroyGraphGlobals(); if (systraymenu) delete systraymenu; + if (systray) delete systray; disconnect(logger, SIGNAL(outputLog(QString)), this, SLOT(logMessage(QString))); shutdownLogger(); @@ -408,8 +411,11 @@ bool MainWindow::OpenProfile(QString profileName, bool skippassword) progress->setMessage(QObject::tr("Loading profile \"%1\"...").arg(profileName)); progress->open(); -#ifdef LOCK_RESMED_SESSIONS QList machines = p_profile->GetMachines(MT_CPAP); + if (machines.isEmpty()) { + qDebug() << "No machines in profile"; + } else { +#ifdef LOCK_RESMED_SESSIONS for (QList::iterator it = machines.begin(); it != machines.end(); ++it) { QString mclass=(*it)->loaderName(); if (mclass == STR_MACH_ResMed) { @@ -428,6 +434,7 @@ bool MainWindow::OpenProfile(QString profileName, bool skippassword) } } #endif + } // Todo: move this to AHIWIndow check to profile Load function... if (p_profile->cpap->AHIWindow() < 30.0) { @@ -460,6 +467,7 @@ bool MainWindow::OpenProfile(QString profileName, bool skippassword) // Reload everything profile related if (daily) { qCritical() << "OpenProfile called with active Daily object!"; + qDebug() << "Abandon opening Profile"; return false; } welcome = new Welcome(ui->tabWidget); @@ -471,7 +479,7 @@ bool MainWindow::OpenProfile(QString profileName, bool skippassword) if (overview) { qCritical() << "OpenProfile called with active Overview object!"; - + qDebug() << "Abandon opening Profile"; return false; } overview = new Overview(ui->tabWidget, daily->graphView()); @@ -486,14 +494,19 @@ bool MainWindow::OpenProfile(QString profileName, bool skippassword) AppSetting->setProfileName(p_profile->user->userName()); setWindowTitle(STR_TR_OSCAR + QString(" %1 (" + tr("Profile") + ": %2)").arg(getBranchVersion()).arg(AppSetting->profileName())); - ui->oximetryButton->setDisabled(false); - ui->dailyButton->setDisabled(false); - ui->overviewButton->setDisabled(false); - ui->statisticsButton->setDisabled(false); + bool noMachines = (machines.isEmpty()); ui->importButton->setDisabled(false); + ui->oximetryButton->setDisabled(false); + ui->dailyButton->setDisabled(noMachines); + ui->overviewButton->setDisabled(noMachines); + ui->statisticsButton->setDisabled(noMachines); + ui->tabWidget->setTabEnabled(2, !noMachines); // daily, STR_TR_Daily); + ui->tabWidget->setTabEnabled(3, !noMachines); // overview, STR_TR_Overview); + ui->tabWidget->setTabEnabled(4, !noMachines); // overview, STR_TR_Overview); progress->close(); delete progress; + qDebug() << "Finished opening Profile"; return true; } @@ -1051,15 +1064,15 @@ QString MainWindow::getWelcomeHTML() tr("(It doesn't support SSL encryption, so it's not a good idea to type your passwords or personal details anywhere.)") + "

    " + -// tr("SleepyHead's Online Users Guide
    ") +// tr("OSCAR's Online Users Guide
    ") // + // tr("Frequently Asked Questions
    ") // + // tr("Glossary of Sleep Disorder Terms
    ") // + -// tr("SleepyHead Wiki
    ") +// tr("OSCAR Wiki
    ") // + -// tr("SleepyHead's Project Website on SourceForge
    ") +// tr("OSCAR's Project Website on SourceForge
    ") // + "

    " + tr("Further Information") + "

    " "

    " + @@ -1259,15 +1272,17 @@ void MainWindow::on_oximetryButton_clicked() void MainWindow::CheckForUpdates() { qDebug() << "procedure called"; +#ifndef NO_UPDATER on_actionCheck_for_Updates_triggered(); +#else + QMessageBox::information(nullptr, STR_MessageBox_Information, tr("Updates are not yet implemented")); +#endif } void MainWindow::on_actionCheck_for_Updates_triggered() { -#ifdef NO_UPDATER qDebug() << "procedure called"; - QMessageBox::information(nullptr, STR_MessageBox_Information, tr("Updates are not yet implemented")); -#else +#ifndef NO_UPDATER UpdaterWindow *w = new UpdaterWindow(this); w->checkForUpdates(); #endif @@ -1436,8 +1451,8 @@ void MainWindow::on_action_CycleTabs_triggered() void MainWindow::on_actionOnline_Users_Guide_triggered() { -// QDesktopServices::openUrl(QUrl("http://sleepyhead.sourceforge.net/wiki/index.php?title=SleepyHead_Users_Guide")); - QMessageBox::information(nullptr, STR_MessageBox_Information, tr("The User's Guide is not yet implemented")); +// QDesktopServices::openUrl(QUrl("http://sleepyhead.sourceforge.net/wiki/index.php?title=OSCAR_Users_Guide")); + QMessageBox::information(nullptr, STR_MessageBox_Information, tr("The User's Guide is not yet available")); } void MainWindow::on_action_Frequently_Asked_Questions_triggered() @@ -1830,12 +1845,11 @@ void MainWindow::on_actionRebuildCPAP(QAction *action) bool backups = (dirCount(bpath) > 0) ? true : false; if (backups) { - if (QMessageBox::question(this, - STR_MessageBox_Question, + if (QMessageBox::question(this, STR_MessageBox_Question, tr("Are you sure you want to rebuild all CPAP data for the following machine:\n\n") + mach->brand() + " " + mach->model() + " " + mach->modelnumber() + " (" + mach->serial() + ")\n\n" + - tr("Please note, that this could result in loss of graph data if OSCAR's backups have been disabled or interfered with in any way."), + tr("Please note, that this could result in loss of data if OSCAR's backups have been disabled."), QMessageBox::Yes | QMessageBox::No, QMessageBox::No) == QMessageBox::No) { return; } @@ -2311,6 +2325,8 @@ void MainWindow::GenerateStatistics() void MainWindow::JumpDaily() { + qDebug() << "Set current Widget to Daily"; + sleep(3); ui->tabWidget->setCurrentWidget(daily); } void MainWindow::JumpOverview() diff --git a/oscar/mainwindow.h b/oscar/mainwindow.h index 609e1a4a..c6543b46 100644 --- a/oscar/mainwindow.h +++ b/oscar/mainwindow.h @@ -1,4 +1,4 @@ -/* SleepyHead MainWindow Headers +/* OSCAR MainWindow Headers * * Copyright (C) 2011-2018 Mark Watkins * @@ -34,17 +34,17 @@ class MainWindow; } -/*! \mainpage SleepyHead +/*! \mainpage OSCAR \section intro_sec Introduction - OpenSource CPAP Reviewer (OSCAR) is a program derived from the SleepyHead program written by Mark Watkins. + Open Source CPAP Analysis Reporter (OSCAR) is a program derived from the SleepyHead program written by Mark Watkins. - SleepyHead is Cross-Platform Open-Source software for reviewing data from %CPAP machines, which are used in the treatment of Sleep Disorders. + SleepyHead was a Cross-Platform Open-Source software for reviewing data from %CPAP machines, which are used in the treatment of Sleep Disorders. SleepyHead was created by Mark Watkins (JediMark), an Australian software developer. - This document is an attempt to provide a little technical insight into SleepyHead's program internals. + This document is an attempt to provide a little technical insight into OSCAR's program internals. \section project_info Further Information The project was hosted on sourceforge, and it's original project page can be reached at http://sourceforge.net/projects/sleepyhead. @@ -52,7 +52,7 @@ class MainWindow; There was also a SleepyHead Wiki containing further information \section structure Program Structure - SleepyHead is written in C++ using Qt Toolkit library, and comprises of 3 main components + OSCAR is written in C++ using Qt Toolkit library, and comprises of 3 main components \list \li The SleepLib Database, a specialized database for working with multiple sources of Sleep machine data. \li A custom designed, high performance and interactive OpenGL Graphing Library. @@ -77,7 +77,7 @@ class Overview; /*! \class MainWindow \author Mark Watkins - \brief The Main Application window for SleepyHead + \brief The Main Application window for OSCAR */ class MainWindow : public QMainWindow @@ -106,7 +106,7 @@ class MainWindow : public QMainWindow void CloseProfile(); bool OpenProfile(QString name, bool skippassword = false); - /*! \fn Notify(QString s,int ms=5000, QString title="SleepyHead v"+VersionString()); + /*! \fn Notify(QString s,int ms=5000, QString title="OSCAR v"+VersionString()); \brief Pops up a message box near the system tray \param QString string \param title @@ -131,7 +131,7 @@ class MainWindow : public QMainWindow Overview *getOverview() { return overview; } /*! \fn void RestartApplication(bool force_login=false); - \brief Closes down SleepyHead and restarts it + \brief Closes down OSCAR and restarts it \param bool force_login If force_login is set, it will return to the login menu even if it's set to skip diff --git a/oscar/newprofile.cpp b/oscar/newprofile.cpp index 3a805e7c..b5a5fa10 100644 --- a/oscar/newprofile.cpp +++ b/oscar/newprofile.cpp @@ -24,7 +24,7 @@ extern MainWindow *mainwin; NewProfile::NewProfile(QWidget *parent, const QString *user) : - QDialog(parent), + QDialog(parent, Qt::WindowTitleHint | Qt::WindowCloseButtonHint), ui(new Ui::NewProfile) { ui->setupUi(this); @@ -106,7 +106,7 @@ QString NewProfile::getIntroHTML() { return "" "" - "

    " + tr("Welcome to Open Source CPAP Reviewer") + "

    " + "

    " + tr("Welcome to the Open Source CPAP Analysis Reporter") + "

    " "

    " + tr("This software is being designed to assist you in reviewing the data produced by your CPAP machines and related equipment.") + "

    " diff --git a/oscar/oscar.pro b/oscar/oscar.pro index f873c96e..91865cec 100644 --- a/oscar/oscar.pro +++ b/oscar/oscar.pro @@ -4,6 +4,8 @@ # #------------------------------------------------- +message(Platform is $$QMAKESPEC ) + lessThan(QT_MAJOR_VERSION,5)|lessThan(QT_MINOR_VERSION,9) { message("You need Qt 5.9 to build OSCAR with Help Pages") DEFINES += helpless @@ -51,7 +53,7 @@ contains(DEFINES, STATIC) { } } -TARGET = Oscar +TARGET = OSCAR unix:!macx:!haiku { TARGET.path=/usr/bin } @@ -63,9 +65,11 @@ gitinfotarget.depends = FORCE win32 { system("$$_PRO_FILE_PWD_/update_gitinfo.bat"); + message("Updating gitinfo.h for Windows build") gitinfotarget.commands = "$$_PRO_FILE_PWD_/update_gitinfo.bat" } else { system("/bin/bash $$_PRO_FILE_PWD_/update_gitinfo.sh"); + message("Updating gitinfo.h for non-Windows build") gitinfotarget.commands = "/bin/bash $$_PRO_FILE_PWD_/update_gitinfo.sh" } @@ -82,12 +86,12 @@ QMAKE_EXTRA_TARGETS += gitinfotarget message("Finished generating help files"); } -QMAKE_TARGET_PRODUCT = Oscar +QMAKE_TARGET_PRODUCT = OSCAR QMAKE_TARGET_COMPANY = Nightowl Software QMAKE_TARGET_COPYRIGHT = Copyright (c)2011-2018 Mark Watkins & (c) 2019 Nightowl Software QMAKE_TARGET_DESCRIPTION = "OpenSource CPAP Analysis Reporter" VERSION = 1.0.0 -# RC_ICONS = ./icons/logo-v3.0.ico +RC_ICONS = ./icons/logo.ico macx { QMAKE_MACOSX_DEPLOYMENT_TARGET = 10.7 @@ -128,6 +132,7 @@ macx { } TRANSLATIONS = $$files($$PWD/../Translations/*.ts) + qtPrepareTool(LRELEASE, lrelease) for(file, TRANSLATIONS) { diff --git a/oscar/oximeterimport.cpp b/oscar/oximeterimport.cpp index e1b677b3..7953539d 100644 --- a/oscar/oximeterimport.cpp +++ b/oscar/oximeterimport.cpp @@ -1115,7 +1115,7 @@ void OximeterImport::setInformation() "

    " +tr("OSCAR gives you the ability to track Oximetry data alongside CPAP session data, which can give valuable insight into the effectiveness of CPAP treatment. It will also work standalone with your Pulse Oximeter, allowing you to store, track and review your recorded data.")+"

    " "

    " - +tr("OSCAR is currently compatible with Contec CMS50D+, CMS50E, CMS50F and CMS50I serial oximeters.
    (Note: Direct importing from bluetooth models is probaby not possible yet)")+"

    " + +tr("OSCAR is currently compatible with Contec CMS50D+, CMS50E, CMS50F and CMS50I serial oximeters.
    (Note: Direct importing from bluetooth models is probably not possible yet)")+"

    " "

    " +tr("You may wish to note, other companies, such as Pulox, simply rebadge Contec CMS50's under new names, such as the Pulox PO-200, PO-300, PO-400. These should also work.")+"

    " diff --git a/oscar/preferencesdialog.cpp b/oscar/preferencesdialog.cpp index f658660a..ce67fe96 100644 --- a/oscar/preferencesdialog.cpp +++ b/oscar/preferencesdialog.cpp @@ -1,4 +1,4 @@ -/* SleepyHead Preferences Dialog Implementation +/* OSCAR Preferences Dialog Implementation * * Copyright (c) 2011-2018 Mark Watkins * @@ -38,7 +38,7 @@ typedef QMessageBox::StandardButtons StandardButtons; QHash channeltype; PreferencesDialog::PreferencesDialog(QWidget *parent, Profile *_profile) : - QDialog(parent), + QDialog(parent, Qt::WindowTitleHint | Qt::WindowCloseButtonHint), ui(new Ui::PreferencesDialog), profile(_profile) { diff --git a/oscar/reports.ui b/oscar/reports.ui index a1586174..e16faa7a 100644 --- a/oscar/reports.ui +++ b/oscar/reports.ui @@ -35,7 +35,7 @@ QWebView QWidget -
    QtWebKit/QWebView
    +
    QWebView
    diff --git a/oscar/statistics.cpp b/oscar/statistics.cpp index e8dd24b8..470516a8 100644 --- a/oscar/statistics.cpp +++ b/oscar/statistics.cpp @@ -653,7 +653,7 @@ QString Statistics::htmlFooter(bool showinfo) if (showinfo) { html += "
    "; html += tr("This report was generated by OSCAR v%1").arg(ShortVersionString) + "
    " - +tr("OSCAR is free open-source CPAP review software"); + +tr("OSCAR is free open-source CPAP report software"); html += "
    "; } @@ -990,7 +990,7 @@ QString Statistics::GenerateHTML() // html += QString("") + html += QString( "


    " + tr("I can haz data?!?") + "

    "+ "

    " - "

    "+tr("Oscar has to data to report :(")+"

    "); + "

    "+tr("Oscar has no data to report :(")+"

    "); // ""; html += htmlFooter(havedata); diff --git a/oscar/welcome.cpp b/oscar/welcome.cpp index 01690787..382e9f91 100644 --- a/oscar/welcome.cpp +++ b/oscar/welcome.cpp @@ -45,11 +45,16 @@ void Welcome::refreshPage() ui->S9Warning->setVisible(showCardWarning); if (!b) { + qDebug() << "No machines in Profile"; +// sleep(3); ui->cpapIcon->setPixmap(pixmap); } ui->dailyButton->setEnabled(b); ui->overviewButton->setEnabled(b); ui->statisticsButton->setEnabled(b); +// ui->tabWidget->setTabEnabled(2, b); +// ui->tabWidget->setTabEnabled(3, b); +// ui->tabWidget->setTabEnabled(4, b); ui->cpapInfo->setHtml(GenerateCPAPHTML()); ui->oxiInfo->setHtml(GenerateOxiHTML()); diff --git a/oscar/welcome.ui b/oscar/welcome.ui index e1395b29..e41d1991 100644 --- a/oscar/welcome.ui +++ b/oscar/welcome.ui @@ -125,7 +125,7 @@ - Welcome To Open Source CPAP Reviewer + Welcome to the Open Source CPAP Analysis Reporter Qt::AlignCenter