mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
The full version now includes the build/git information embedded within it as build metadata according to the Semantic Versioning 2.0.0 spec, for example: "1.1.0-beta-1+branch-name-a1b2c3d". Now the full version string, with all detail is always displayed EXCEPT for release versions, in which case just the simple version number ("1.1.0") is displayed in the primary UI. - Main window title: simple version for release versions, full version string otherwise - Notifications: same as main window title - System tray: same as main window title - About window title: same as main window title - About window release notes: always include full version string - Reports: always include full version string - Under the logo (about dialog, profile selector, new profile window): removed, as it is largely redundant and can interfere with the window geometry. - Database upgrade alert: same as main window title - Database newer alert: same as main window title The full version string is also included within the preference and profile .xml files, but because build metadata is ignored in version comparisons, differences in builds will not cause any spurious alerts. However, changes in prerelease versions will continue to be significant, as they should be.
2627 lines
84 KiB
C++
2627 lines
84 KiB
C++
/* OSCAR MainWindow Implementation
|
|
*
|
|
* Copyright (c) 2020 The OSCAR Team
|
|
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
|
* License. See the file COPYING in the main directory of the source code
|
|
* for more details. */
|
|
|
|
#include <QHostInfo>
|
|
#include <QFileDialog>
|
|
#include <QMessageBox>
|
|
#include <QResource>
|
|
#include <QProgressBar>
|
|
#include <QTimer>
|
|
#include <QSettings>
|
|
#include <QPixmap>
|
|
#include <QDesktopWidget>
|
|
#include <QListView>
|
|
#include <QPrinter>
|
|
#include <QPrintDialog>
|
|
#include <QPainter>
|
|
#include <QProcess>
|
|
#include <QFontMetrics>
|
|
#include <QTextDocument>
|
|
#include <QTranslator>
|
|
#include <QPushButton>
|
|
#include <QCalendarWidget>
|
|
#include <QDialogButtonBox>
|
|
#include <QTextBrowser>
|
|
#include <QStandardPaths>
|
|
#include <QDesktopServices>
|
|
#include <QScreen>
|
|
#include <cmath>
|
|
|
|
#include "common_gui.h"
|
|
#include "version.h"
|
|
|
|
|
|
// Custom loaders that don't autoscan..
|
|
#include <SleepLib/loader_plugins/zeo_loader.h>
|
|
#include <SleepLib/loader_plugins/somnopose_loader.h>
|
|
|
|
#ifdef REMSTAR_M_SUPPORT
|
|
#include <SleepLib/loader_plugins/mseries_loader.h>
|
|
#endif
|
|
|
|
#include "logger.h"
|
|
#include "mainwindow.h"
|
|
#include "ui_mainwindow.h"
|
|
#include "aboutdialog.h"
|
|
#include "newprofile.h"
|
|
#include "exportcsv.h"
|
|
#include "SleepLib/schema.h"
|
|
#include "Graphs/glcommon.h"
|
|
#include "UpdaterWindow.h"
|
|
#include "SleepLib/calcs.h"
|
|
#include "SleepLib/progressdialog.h"
|
|
|
|
#include "reports.h"
|
|
#include "statistics.h"
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,4,0)
|
|
#include <QOpenGLFunctions>
|
|
#endif
|
|
|
|
MainWindow::MainWindow(QWidget *parent) :
|
|
QMainWindow(parent),
|
|
ui(new Ui::MainWindow)
|
|
{
|
|
ui->setupUi(this);
|
|
ui->logText->setPlainText("00000: Startup: OSCAR Logger initialized");
|
|
|
|
if (logger) {
|
|
connect(logger, SIGNAL(outputLog(QString)), this, SLOT(logMessage(QString)));
|
|
}
|
|
|
|
// Bring window to top (useful when language is changed)
|
|
this->activateWindow();
|
|
this->raise();
|
|
|
|
// Initialise oscar app registry stuff
|
|
QSettings settings;
|
|
|
|
// Load previous Window geometry
|
|
restoreGeometry(settings.value("MainWindow/geometry").toByteArray());
|
|
|
|
|
|
// 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-sm.png"), this);
|
|
systray->show();
|
|
// seems to need the systray menu for notifications to work
|
|
systraymenu = new QMenu(this);
|
|
systray->setContextMenu(systraymenu);
|
|
QAction *a = systraymenu->addAction(STR_TR_OSCAR + " " + getVersion().displayString());
|
|
a->setEnabled(false);
|
|
systraymenu->addSeparator();
|
|
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(tr("E&xit"), this, SLOT(close()));
|
|
|
|
// systraymenu = nullptr;
|
|
} else { // if not available, the messages will popup in the taskbar
|
|
qDebug() << "No System Tray menues";
|
|
systray = nullptr;
|
|
systraymenu = nullptr;
|
|
}
|
|
|
|
}
|
|
|
|
bool setupRunning = false;
|
|
|
|
QString MainWindow::getMainWindowTitle()
|
|
{
|
|
QString title = STR_TR_OSCAR + " " + getVersion().displayString();
|
|
#ifdef BROKEN_OPENGL_BUILD
|
|
title += " ["+CSTR_GFX_BrokenGL+"]";
|
|
#endif
|
|
return title;
|
|
}
|
|
|
|
void MainWindow::SetupGUI()
|
|
{
|
|
setupRunning = true;
|
|
setWindowTitle(getMainWindowTitle());
|
|
|
|
#ifdef Q_OS_MAC
|
|
ui->action_About->setMenuRole(QAction::ApplicationSpecificRole);
|
|
ui->action_Preferences->setMenuRole(QAction::ApplicationSpecificRole);
|
|
ui->action_Preferences->setShortcuts(QKeySequence::Preferences);
|
|
#endif
|
|
|
|
ui->actionLine_Cursor->setChecked(AppSetting->lineCursorMode());
|
|
ui->actionPie_Chart->setChecked(AppSetting->showPieChart());
|
|
ui->actionDebug->setChecked(AppSetting->showDebug());
|
|
ui->actionShow_Performance_Counters->setChecked(AppSetting->showPerformance());
|
|
|
|
overview = nullptr;
|
|
daily = nullptr;
|
|
prefdialog = nullptr;
|
|
profileSelector = nullptr;
|
|
welcome = nullptr;
|
|
|
|
ui->oximetryButton->setDisabled(true);
|
|
ui->dailyButton->setDisabled(true);
|
|
ui->overviewButton->setDisabled(true);
|
|
ui->statisticsButton->setDisabled(true);
|
|
ui->importButton->setDisabled(true);
|
|
#ifdef helpless
|
|
ui->helpButton->setVisible(false);
|
|
#endif
|
|
|
|
m_inRecalculation = false;
|
|
m_restartRequired = false;
|
|
// Initialize Status Bar objects
|
|
|
|
QTextCharFormat format = ui->statStartDate->calendarWidget()->weekdayTextFormat(Qt::Saturday);
|
|
format.setForeground(QBrush(Qt::black, Qt::SolidPattern));
|
|
Qt::DayOfWeek dow=firstDayOfWeekFromLocale();
|
|
ui->statStartDate->calendarWidget()->setWeekdayTextFormat(Qt::Saturday, format);
|
|
ui->statStartDate->calendarWidget()->setWeekdayTextFormat(Qt::Sunday, format);
|
|
ui->statStartDate->calendarWidget()->setVerticalHeaderFormat(QCalendarWidget::NoVerticalHeader);
|
|
ui->statStartDate->calendarWidget()->setFirstDayOfWeek(dow);
|
|
ui->statEndDate->calendarWidget()->setWeekdayTextFormat(Qt::Saturday, format);
|
|
ui->statEndDate->calendarWidget()->setWeekdayTextFormat(Qt::Sunday, format);
|
|
ui->statEndDate->calendarWidget()->setVerticalHeaderFormat(QCalendarWidget::NoVerticalHeader);
|
|
ui->statEndDate->calendarWidget()->setFirstDayOfWeek(dow);
|
|
|
|
ui->statEndDate->setVisible(false);
|
|
ui->statStartDate->setVisible(false);
|
|
|
|
ui->reportModeRange->setVisible(false);
|
|
int srm = 0;
|
|
if (p_profile) {
|
|
srm = p_profile->general->statReportMode();
|
|
}
|
|
|
|
switch(srm) {
|
|
case 0:
|
|
ui->reportModeStandard->setChecked(true);
|
|
break;
|
|
case 1:
|
|
ui->reportModeMonthly->setChecked(true);
|
|
break;
|
|
case 2:
|
|
ui->reportModeRange->setChecked(true);
|
|
ui->statEndDate->setVisible(true);
|
|
ui->statStartDate->setVisible(true);
|
|
break;
|
|
default:
|
|
if (p_profile) {
|
|
p_profile->general->setStatReportMode(0);
|
|
}
|
|
break;
|
|
}
|
|
if (!AppSetting->showDebug()) {
|
|
ui->logText->hide();
|
|
}
|
|
|
|
#ifdef Q_OS_MAC
|
|
//p_profile->appearance->setAntiAliasing(false);
|
|
#endif
|
|
//ui->action_Link_Graph_Groups->setChecked(p_profile->general->linkGroups());
|
|
|
|
first_load = true;
|
|
|
|
profileSelector = new ProfileSelector(ui->tabWidget);
|
|
ui->tabWidget->insertTab(0, profileSelector, STR_TR_Profile);
|
|
|
|
// Profiles haven't been loaded here...
|
|
profileSelector->updateProfileList();
|
|
|
|
// Start with the new Profile Tab
|
|
ui->tabWidget->setCurrentWidget(profileSelector); // setting this to daily shows the cube during loading..
|
|
ui->tabWidget->setTabEnabled(1, false); // this should be the Statistics tab
|
|
|
|
ui->toolBox->setCurrentIndex(0);
|
|
bool b = AppSetting->rightSidebarVisible();
|
|
ui->action_Sidebar_Toggle->setChecked(b);
|
|
ui->toolBox->setVisible(b);
|
|
|
|
ui->actionPie_Chart->setChecked(AppSetting->showPieChart());
|
|
|
|
ui->actionDaily_Calendar->setChecked(AppSetting->calendarVisible());
|
|
|
|
on_tabWidget_currentChanged(0);
|
|
|
|
#ifndef REMSTAR_M_SUPPORT
|
|
ui->actionImport_RemStar_MSeries_Data->setVisible(false);
|
|
#endif
|
|
|
|
qsrand(QDateTime::currentDateTime().toTime_t());
|
|
|
|
QList<int> a;
|
|
|
|
int panel_width = AppSetting->rightPanelWidth();
|
|
a.push_back(this->width() - panel_width);
|
|
a.push_back(panel_width);
|
|
ui->mainsplitter->setSizes(a);
|
|
ui->mainsplitter->setStretchFactor(0,1);
|
|
ui->mainsplitter->setStretchFactor(1,0);
|
|
|
|
QTimer::singleShot(50, this, SLOT(Startup()));
|
|
|
|
ui->actionChange_Data_Folder->setVisible(false);
|
|
ui->action_Frequently_Asked_Questions->setVisible(false);
|
|
|
|
#ifndef helpless
|
|
help = new Help(this);
|
|
ui->tabWidget->addTab(help, tr("Help Browser"));
|
|
#endif
|
|
setupRunning = false;
|
|
}
|
|
|
|
void MainWindow::logMessage(QString msg)
|
|
{
|
|
ui->logText->appendPlainText(msg);
|
|
}
|
|
|
|
void MainWindow::closeEvent(QCloseEvent * event)
|
|
{
|
|
Q_UNUSED(event);
|
|
static bool runonce = false;
|
|
if (!runonce) {
|
|
if (AppSetting->removeCardReminder()) {
|
|
Notify(QObject::tr("Don't forget to place your datacard back in your CPAP machine"), QObject::tr("OSCAR Reminder"));
|
|
QThread::msleep(1000);
|
|
QApplication::processEvents();
|
|
}
|
|
schema::channel.Save();
|
|
if (p_profile) {
|
|
CloseProfile();
|
|
}
|
|
|
|
// Shutdown and Save the current User profile
|
|
Profiles::Done();
|
|
|
|
// Save current window position
|
|
QSettings settings;
|
|
settings.setValue("MainWindow/geometry", saveGeometry());
|
|
|
|
// Trash anything allocated by the Graph objects
|
|
DestroyGraphGlobals();
|
|
|
|
if (systraymenu) {
|
|
delete systraymenu;
|
|
systraymenu = nullptr;
|
|
}
|
|
if (systray) {
|
|
delete systray;
|
|
systray = nullptr;
|
|
}
|
|
|
|
disconnect(logger, SIGNAL(outputLog(QString)), this, SLOT(logMessage(QString)));
|
|
shutdownLogger();
|
|
|
|
runonce = true;
|
|
} else {
|
|
qDebug() << "Qt is still calling closevent multiple times";
|
|
QApplication::processEvents();
|
|
}
|
|
}
|
|
|
|
MainWindow::~MainWindow()
|
|
{
|
|
delete ui;
|
|
QCoreApplication::quit();
|
|
}
|
|
|
|
void MainWindow::log(QString text)
|
|
{
|
|
logger->appendClean(text);
|
|
}
|
|
|
|
void MainWindow::EnableTabs(bool b)
|
|
{
|
|
ui->tabWidget->setTabEnabled(2, b);
|
|
ui->tabWidget->setTabEnabled(3, b);
|
|
ui->tabWidget->setTabEnabled(4, b);
|
|
}
|
|
|
|
void MainWindow::Notify(QString s, QString title, int ms)
|
|
{
|
|
if (title.isEmpty()) {
|
|
title = STR_TR_OSCAR + " " + getVersion().displayString();
|
|
}
|
|
if (systray) {
|
|
QString msg = s;
|
|
|
|
#ifdef Q_OS_UNIX
|
|
// GNOME3's systray hides the last line of the displayed Qt message.
|
|
// As a workaround, add an extra line to bump the message back
|
|
// into the visible area.
|
|
char *desktop = getenv("DESKTOP_SESSION");
|
|
|
|
if (desktop && !strncmp(desktop, "gnome", 5)) {
|
|
msg += "\n";
|
|
}
|
|
#endif
|
|
|
|
systray->showMessage(title, msg, QSystemTrayIcon::Information, ms);
|
|
}
|
|
}
|
|
|
|
QString getCPAPPixmap(QString mach_class)
|
|
{
|
|
QString cpapimage;
|
|
if (mach_class == STR_MACH_ResMed) cpapimage = ":/icons/rms9.png";
|
|
else if (mach_class == STR_MACH_PRS1) cpapimage = ":/icons/prs1.png";
|
|
else if (mach_class == STR_MACH_Intellipap) cpapimage = ":/icons/intellipap.png";
|
|
return cpapimage;
|
|
}
|
|
|
|
//QIcon getCPAPIcon(QString mach_class)
|
|
//{
|
|
// QString cpapimage = getCPAPPixmap(mach_class);
|
|
|
|
// return QIcon(cpapimage);
|
|
//}
|
|
|
|
void MainWindow::PopulatePurgeMenu()
|
|
{
|
|
ui->menu_Rebuild_CPAP_Data->disconnect(ui->menu_Rebuild_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionRebuildCPAP(QAction *)));
|
|
ui->menu_Rebuild_CPAP_Data->clear();
|
|
|
|
ui->menuPurge_CPAP_Data->disconnect(ui->menuPurge_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionPurgeMachine(QAction *)));
|
|
ui->menuPurge_CPAP_Data->clear();
|
|
|
|
QList<Machine *> machines = p_profile->GetMachines(MT_CPAP);
|
|
for (int i=0; i < machines.size(); ++i) {
|
|
Machine *mach = machines.at(i);
|
|
QString name = mach->brand() + " "+
|
|
mach->model() + " "+
|
|
mach->serial();
|
|
|
|
QAction * action = new QAction(name.replace("&","&&"), ui->menu_Rebuild_CPAP_Data);
|
|
action->setIconVisibleInMenu(true);
|
|
action->setIcon(mach->getPixmap());
|
|
action->setData(mach->loaderName()+":"+mach->serial());
|
|
ui->menu_Rebuild_CPAP_Data->addAction(action);
|
|
|
|
action = new QAction(name.replace("&","&&"), ui->menuPurge_CPAP_Data);
|
|
action->setIconVisibleInMenu(true);
|
|
action->setIcon(mach->getPixmap()); //getCPAPIcon(mach->loaderName()));
|
|
action->setData(mach->loaderName()+":"+mach->serial());
|
|
|
|
ui->menuPurge_CPAP_Data->addAction(action);
|
|
}
|
|
ui->menu_Rebuild_CPAP_Data->connect(ui->menu_Rebuild_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionRebuildCPAP(QAction*)));
|
|
ui->menuPurge_CPAP_Data->connect(ui->menuPurge_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionPurgeMachine(QAction*)));
|
|
}
|
|
|
|
void MainWindow::firstRunMessage()
|
|
{
|
|
if (AppSetting->showAboutDialog() > 0) {
|
|
AboutDialog * about = new AboutDialog(this);
|
|
|
|
about->exec();
|
|
|
|
AppSetting->setShowAboutDialog(-1);
|
|
|
|
about->deleteLater();
|
|
}
|
|
}
|
|
|
|
// QString GenerateWelcomeHTML();
|
|
|
|
bool MainWindow::OpenProfile(QString profileName, bool skippassword)
|
|
{
|
|
qDebug() << "Opening profile" << profileName;
|
|
|
|
auto pit = Profiles::profiles.find(profileName);
|
|
if (pit == Profiles::profiles.end()) return false;
|
|
|
|
Profile * prof = pit.value();
|
|
if (p_profile) {
|
|
if ((prof != p_profile)) {
|
|
CloseProfile();
|
|
} else {
|
|
// Already open
|
|
return false;
|
|
}
|
|
}
|
|
prof = profileSelector->SelectProfile(profileName, skippassword); // asks for the password and updates stuff in profileSelector tab
|
|
if (!prof) {
|
|
return false;
|
|
}
|
|
// TODO: Check profile password
|
|
|
|
// Check Lockfile
|
|
QString lockhost = prof->checkLock();
|
|
|
|
if (!lockhost.isEmpty()) {
|
|
if (lockhost.compare(QHostInfo::localHostName()) != 0) {
|
|
if (QMessageBox::warning(nullptr, STR_MessageBox_Warning,
|
|
QObject::tr("There is a lockfile already present for this profile '%1', claimed on '%2'.").arg(prof->user->userName()).arg(lockhost)+"\n\n"+
|
|
QObject::tr("You can only work with one instance of an individual OSCAR profile at a time.")+"\n\n"+
|
|
QObject::tr("If you are using cloud storage, make sure OSCAR is closed and syncing has completed first on the other computer before proceeding."),
|
|
QMessageBox::Cancel |QMessageBox::Ok, QMessageBox::Cancel) == QMessageBox::Cancel) {
|
|
return false;
|
|
}
|
|
} // not worried about localhost locks anymore, just silently drop it.
|
|
|
|
prof->removeLock();
|
|
}
|
|
|
|
p_profile = prof;
|
|
|
|
ProgressDialog * progress = new ProgressDialog(this);
|
|
|
|
progress->setMessage(QObject::tr("Loading profile \"%1\"...").arg(profileName));
|
|
progress->open();
|
|
|
|
QList<Machine *> machines = p_profile->GetMachines(MT_CPAP);
|
|
if (machines.isEmpty()) {
|
|
qDebug() << "No machines in profile";
|
|
} else {
|
|
#ifdef LOCK_RESMED_SESSIONS
|
|
for (QList<Machine *>::iterator it = machines.begin(); it != machines.end(); ++it) {
|
|
QString mclass=(*it)->loaderName();
|
|
if (mclass == STR_MACH_ResMed) {
|
|
qDebug() << "ResMed machine found.. locking OSCAR preferences to suit it's summary system";
|
|
|
|
// Have to sacrifice these features to get access to summary data.
|
|
p_profile->session->setCombineCloseSessions(0);
|
|
p_profile->session->setDaySplitTime(QTime(12,0,0));
|
|
p_profile->session->setIgnoreShortSessions(false);
|
|
p_profile->session->setLockSummarySessions(true);
|
|
p_profile->general->setPrefCalcPercentile(95.0); // 95%
|
|
p_profile->general->setPrefCalcMiddle(0); // Median (50%)
|
|
p_profile->general->setPrefCalcMax(1); // Dodgy max
|
|
|
|
break;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// Todo: move this to AHIWIndow check to profile Load function...
|
|
if (p_profile->cpap->AHIWindow() < 30.0) {
|
|
p_profile->cpap->setAHIWindow(60.0);
|
|
}
|
|
|
|
|
|
if (p_profile->p_preferences[STR_PREF_ReimportBackup].toBool()) {
|
|
importCPAPBackups();
|
|
p_profile->p_preferences[STR_PREF_ReimportBackup]=false;
|
|
}
|
|
|
|
QList<MachineLoader *> loaders = GetLoaders();
|
|
for (int i=0; i<loaders.size(); ++i) {
|
|
connect(loaders.at(i), SIGNAL(machineUnsupported(Machine*)), this, SLOT(MachineUnsupported(Machine*)));
|
|
}
|
|
|
|
p_profile->LoadMachineData(progress);
|
|
progress->setMessage(tr("Loading profile \"%1\"").arg(profileName));
|
|
|
|
// Show the logo?
|
|
// QPixmap logo=QPixmap(":/icons/logo-md.png").scaled(64,64);
|
|
// progress->setPixmap(logo);
|
|
|
|
QApplication::processEvents();
|
|
|
|
ui->statStartDate->setDate(p_profile->FirstDay());
|
|
ui->statEndDate->setDate(p_profile->LastDay());
|
|
|
|
// Reload everything profile related
|
|
if (daily) {
|
|
qCritical() << "OpenProfile called with active Daily object!";
|
|
qDebug() << "Abandon opening Profile";
|
|
return false;
|
|
}
|
|
welcome = new Welcome(ui->tabWidget);
|
|
ui->tabWidget->insertTab(1, welcome, tr("Welcome"));
|
|
|
|
daily = new Daily(ui->tabWidget, nullptr);
|
|
ui->tabWidget->insertTab(2, daily, STR_TR_Daily);
|
|
daily->ReloadGraphs();
|
|
|
|
if (overview) {
|
|
qCritical() << "OpenProfile called with active Overview object!";
|
|
qDebug() << "Abandon opening Profile";
|
|
return false;
|
|
}
|
|
overview = new Overview(ui->tabWidget, daily->graphView());
|
|
ui->tabWidget->insertTab(3, overview, STR_TR_Overview);
|
|
overview->ReloadGraphs();
|
|
|
|
// Should really create welcome and statistics here, but they need redoing later anyway to kill off webkit
|
|
ui->tabWidget->setCurrentIndex(AppSetting->openTabAtStart());
|
|
GenerateStatistics();
|
|
PopulatePurgeMenu();
|
|
|
|
AppSetting->setProfileName(p_profile->user->userName());
|
|
setWindowTitle(tr("%1 (Profile: %2)").arg(getMainWindowTitle()).arg(AppSetting->profileName()));
|
|
|
|
QList<Machine *> oximachines = p_profile->GetMachines(MT_OXIMETER); // Machines of any type except Journal
|
|
QList<Machine *> posmachines = p_profile->GetMachines(MT_POSITION);
|
|
QList<Machine *> stgmachines = p_profile->GetMachines(MT_SLEEPSTAGE);
|
|
bool noMachines = machines.isEmpty() && posmachines.isEmpty() && oximachines.isEmpty() && stgmachines.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); // statistics, STR_TR_Statistics);
|
|
|
|
int srm = 0;
|
|
if (p_profile) {
|
|
srm = p_profile->general->statReportMode();
|
|
}
|
|
|
|
switch (srm) {
|
|
case 0:
|
|
ui->reportModeStandard->setChecked(true);
|
|
break;
|
|
case 1:
|
|
ui->reportModeMonthly->setChecked(true);
|
|
break;
|
|
case 2:
|
|
ui->reportModeRange->setChecked(true);
|
|
ui->statEndDate->setVisible(true);
|
|
ui->statStartDate->setVisible(true);
|
|
break;
|
|
default:
|
|
if (p_profile) {
|
|
p_profile->general->setStatReportMode(0);
|
|
}
|
|
break;
|
|
}
|
|
|
|
progress->close();
|
|
delete progress;
|
|
qDebug() << "Finished opening Profile";
|
|
|
|
return true;
|
|
}
|
|
|
|
void MainWindow::CloseProfile()
|
|
{
|
|
if (daily) {
|
|
daily->Unload();
|
|
daily->clearLastDay(); // otherwise Daily will crash
|
|
delete daily;
|
|
daily = nullptr;
|
|
}
|
|
if (welcome) {
|
|
delete welcome;
|
|
welcome = nullptr;
|
|
}
|
|
if (overview) {
|
|
delete overview;
|
|
overview = nullptr;
|
|
}
|
|
|
|
if (p_profile) {
|
|
p_profile->StoreMachines();
|
|
p_profile->UnloadMachineData();
|
|
p_profile->saveChannels();
|
|
p_profile->Save();
|
|
p_profile->removeLock();
|
|
p_profile = nullptr;
|
|
}
|
|
}
|
|
|
|
|
|
void MainWindow::Startup()
|
|
{
|
|
for (auto & loader : GetLoaders()) {
|
|
loader->setParent(this);
|
|
}
|
|
QString lastProfile = AppSetting->profileName();
|
|
|
|
firstRunMessage();
|
|
|
|
if (Profiles::profiles.contains(lastProfile) && AppSetting->autoOpenLastUsed()) {
|
|
if (OpenProfile(lastProfile)) {
|
|
ui->tabWidget->setCurrentIndex(AppSetting->openTabAtStart());
|
|
|
|
if (AppSetting->autoLaunchImport()) {
|
|
on_importButton_clicked();
|
|
}
|
|
}
|
|
} else {
|
|
ui->tabWidget->setCurrentWidget(profileSelector);
|
|
}
|
|
}
|
|
|
|
int MainWindow::importCPAP(ImportPath import, const QString &message)
|
|
{
|
|
if (!import.loader) {
|
|
return 0;
|
|
}
|
|
|
|
ui->tabWidget->setCurrentWidget(welcome);
|
|
QApplication::processEvents();
|
|
ProgressDialog * progdlg = new ProgressDialog(this);
|
|
|
|
QPixmap image = import.loader->getPixmap(import.loader->PeekInfo(import.path).series);
|
|
image = image.scaled(64,64);
|
|
progdlg->setPixmap(image);
|
|
|
|
progdlg->addAbortButton();
|
|
|
|
progdlg->setWindowModality(Qt::ApplicationModal);
|
|
progdlg->open();
|
|
progdlg->setMessage(message);
|
|
|
|
|
|
connect(import.loader, SIGNAL(updateMessage(QString)), progdlg, SLOT(setMessage(QString)));
|
|
connect(import.loader, SIGNAL(setProgressMax(int)), progdlg, SLOT(setProgressMax(int)));
|
|
connect(import.loader, SIGNAL(setProgressValue(int)), progdlg, SLOT(setProgressValue(int)));
|
|
connect(progdlg, SIGNAL(abortClicked()), import.loader, SLOT(abortImport()));
|
|
|
|
int c = import.loader->Open(import.path);
|
|
|
|
if (c > 0) {
|
|
Notify(tr("Imported %1 CPAP session(s) from\n\n%2").arg(c).arg(import.path), tr("Import Success"));
|
|
} else if (c == 0) {
|
|
Notify(tr("Already up to date with CPAP data at\n\n%1").arg(import.path), tr("Up to date"));
|
|
} else {
|
|
Notify(tr("Couldn't find any valid Machine Data at\n\n%1").arg(import.path),tr("Import Problem"));
|
|
}
|
|
disconnect(progdlg, SIGNAL(abortClicked()), import.loader, SLOT(abortImport()));
|
|
disconnect(import.loader, SIGNAL(setProgressMax(int)), progdlg, SLOT(setProgressMax(int)));
|
|
disconnect(import.loader, SIGNAL(setProgressValue(int)), progdlg, SLOT(setProgressValue(int)));
|
|
disconnect(import.loader, SIGNAL(updateMessage(QString)), progdlg, SLOT(setMessage(QString)));
|
|
|
|
progdlg->close();
|
|
|
|
delete progdlg;
|
|
|
|
if (AppSetting->openTabAfterImport()>0) {
|
|
ui->tabWidget->setCurrentIndex(AppSetting->openTabAfterImport());
|
|
}
|
|
|
|
return c;
|
|
}
|
|
|
|
void MainWindow::finishCPAPImport()
|
|
{
|
|
p_profile->StoreMachines();
|
|
QList<Machine *> machines = p_profile->GetMachines(MT_CPAP);
|
|
for (Machine * mach : machines) {
|
|
mach->saveSessionInfo();
|
|
mach->SaveSummaryCache();
|
|
}
|
|
|
|
GenerateStatistics();
|
|
profileSelector->updateProfileList();
|
|
|
|
welcome->refreshPage();
|
|
|
|
if (overview) { overview->ReloadGraphs(); }
|
|
if (daily) {
|
|
// daily->populateSessionWidget();
|
|
daily->ReloadGraphs();
|
|
}
|
|
if (AppSetting->openTabAfterImport()>0) {
|
|
ui->tabWidget->setCurrentIndex(AppSetting->openTabAfterImport());
|
|
}
|
|
|
|
}
|
|
|
|
void MainWindow::importCPAPBackups()
|
|
{
|
|
// Get BackupPaths for all CPAP machines
|
|
QList<Machine *> machlist = p_profile->GetMachines(MT_CPAP);
|
|
QList<ImportPath> paths;
|
|
Q_FOREACH(Machine *m, machlist) {
|
|
paths.append(ImportPath(m->getBackupPath(), lookupLoader(m)));
|
|
}
|
|
|
|
if (paths.size() > 0) {
|
|
int c=0;
|
|
Q_FOREACH(ImportPath path, paths) {
|
|
c+=importCPAP(path, tr("Please wait, importing from backup folder(s)..."));
|
|
}
|
|
if (c>0) {
|
|
finishCPAPImport();
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef Q_OS_UNIX
|
|
# include <stdio.h>
|
|
# include <unistd.h>
|
|
|
|
# if defined(Q_OS_MAC) || defined(Q_OS_BSD4)
|
|
# include <sys/mount.h>
|
|
# elif defined(Q_OS_HAIKU)
|
|
// nothing needed
|
|
# else
|
|
# include <sys/statfs.h>
|
|
# include <mntent.h>
|
|
# endif // Q_OS_MAC/BSD
|
|
|
|
#endif // Q_OS_UNIX
|
|
|
|
//! \brief Returns a list of drive mountpoints
|
|
QStringList getDriveList()
|
|
{
|
|
QStringList drivelist;
|
|
|
|
#if defined(Q_OS_MAC) || defined(Q_OS_BSD4)
|
|
struct statfs *mounts;
|
|
int num_mounts = getmntinfo(&mounts, MNT_WAIT);
|
|
if (num_mounts >= 0) {
|
|
for (int i = 0; i < num_mounts; i++) {
|
|
QString name = mounts[i].f_mntonname;
|
|
|
|
// Only interested in drives mounted under /Volumes
|
|
if (name.toLower().startsWith("/volumes/")) {
|
|
drivelist.push_back(name);
|
|
// qDebug() << QString("Disk type '%1' mounted at: %2").arg(mounts[i].f_fstypename).arg(mounts[i].f_mntonname);
|
|
}
|
|
}
|
|
}
|
|
|
|
#elif defined(Q_OS_UNIX) && !defined(Q_OS_HAIKU)
|
|
// Unix / Linux (except BSD)
|
|
FILE *mtab = setmntent("/etc/mtab", "r");
|
|
struct mntent *m;
|
|
struct mntent mnt;
|
|
char strings[4096];
|
|
|
|
// NOTE: getmntent_r is a GNU extension, requiring glibc.
|
|
while ((m = getmntent_r(mtab, &mnt, strings, sizeof(strings)))) {
|
|
|
|
struct statfs fs;
|
|
if ((mnt.mnt_dir != NULL) && (statfs(mnt.mnt_dir, &fs) == 0)) {
|
|
QString name = mnt.mnt_dir;
|
|
quint64 size = fs.f_blocks * fs.f_bsize;
|
|
|
|
if (size > 0) { // this should theoretically ignore /dev, /proc, /sys etc..
|
|
drivelist.push_back(name);
|
|
}
|
|
// quint64 free = fs.f_bfree * fs.f_bsize;
|
|
// quint64 avail = fs.f_bavail * fs.f_bsize;
|
|
}
|
|
}
|
|
endmntent(mtab);
|
|
|
|
#elif defined(Q_OS_WIN) || defined(Q_OS_HAIKU)
|
|
QFileInfoList list = QDir::drives();
|
|
|
|
for (int i = 0; i < list.size(); ++i) {
|
|
QFileInfo fileInfo = list.at(i);
|
|
QString name = fileInfo.filePath();
|
|
if (name.at(0).toUpper() != QChar('C')) { // Ignore the C drive
|
|
drivelist.push_back(name);
|
|
}
|
|
}
|
|
|
|
#endif
|
|
return drivelist;
|
|
}
|
|
|
|
extern MainWindow * mainwin;
|
|
void ImportDialogScan::cancelbutton()
|
|
{
|
|
mainwin->importScanCancelled = true;
|
|
hide();
|
|
}
|
|
|
|
QList<ImportPath> MainWindow::detectCPAPCards()
|
|
{
|
|
const int timeout = 20000;
|
|
|
|
QList<ImportPath> detectedCards;
|
|
|
|
QList<MachineLoader *>loaders = GetLoaders(MT_CPAP);
|
|
QTime time;
|
|
time.start();
|
|
|
|
// Create dialog
|
|
ImportDialogScan popup(this) ;//, Qt::SplashScreen);
|
|
QLabel waitmsg(tr("Please insert your CPAP data card..."));
|
|
QProgressBar progress;
|
|
QVBoxLayout waitlayout;
|
|
popup.setLayout(&waitlayout);
|
|
|
|
QHBoxLayout layout2;
|
|
QIcon icon("://icons/sdcard.png");
|
|
QPushButton skipbtn(icon, tr("Choose a folder"));
|
|
QPushButton cancelbtn(STR_MessageBox_Cancel);
|
|
skipbtn.setMinimumHeight(40);
|
|
cancelbtn.setMinimumHeight(40);
|
|
waitlayout.addWidget(&waitmsg,1,Qt::AlignCenter);
|
|
waitlayout.addWidget(&progress,1);
|
|
waitlayout.addLayout(&layout2,1);
|
|
layout2.addWidget(&skipbtn);
|
|
layout2.addWidget(&cancelbtn);
|
|
popup.connect(&skipbtn, SIGNAL(clicked()), &popup, SLOT(hide()));
|
|
popup.connect(&cancelbtn, SIGNAL(clicked()), &popup, SLOT(cancelbutton()));
|
|
progress.setValue(0);
|
|
progress.setMaximum(timeout);
|
|
progress.setVisible(true);
|
|
importScanCancelled = false;
|
|
popup.show();
|
|
QApplication::processEvents();
|
|
QString lastpath = (*p_profile)[STR_PREF_LastCPAPPath].toString();
|
|
|
|
do {
|
|
// Rescan in case card inserted
|
|
QStringList AutoScannerPaths = getDriveList();
|
|
//AutoScannerPaths.push_back(lastpath);
|
|
|
|
if (!AutoScannerPaths.contains(lastpath)) {
|
|
AutoScannerPaths.append(lastpath);
|
|
}
|
|
|
|
Q_FOREACH(const QString &path, AutoScannerPaths) {
|
|
// Scan through available machine loaders and test if this folder contains valid folder structure
|
|
Q_FOREACH(MachineLoader * loader, loaders) {
|
|
if (loader->Detect(path)) {
|
|
detectedCards.append(ImportPath(path, loader));
|
|
|
|
qDebug() << "Found" << loader->loaderName() << "datacard at" << path;
|
|
}
|
|
QApplication::processEvents();
|
|
}
|
|
}
|
|
int el=time.elapsed();
|
|
progress.setValue(el);
|
|
if (el > timeout) break;
|
|
if (!popup.isVisible()) break;
|
|
// needs a slight delay here
|
|
for (int i=0; i<5; ++i) {
|
|
QThread::msleep(50);
|
|
QApplication::processEvents();
|
|
}
|
|
} while (detectedCards.size() == 0);
|
|
|
|
popup.hide();
|
|
popup.disconnect(&skipbtn, SIGNAL(clicked()), &popup, SLOT(hide()));
|
|
popup.disconnect(&cancelbtn, SIGNAL(clicked()), &popup, SLOT(hide()));
|
|
|
|
return detectedCards;
|
|
}
|
|
|
|
|
|
void MainWindow::on_action_Import_Data_triggered()
|
|
{
|
|
static bool in_import = false;
|
|
if ( p_profile == nullptr ) {
|
|
Notify(tr("No profile has been selected for Import."), STR_MessageBox_Busy);
|
|
return;
|
|
}
|
|
if (m_inRecalculation) {
|
|
Notify(tr("Access to Import has been blocked while recalculations are in progress."),STR_MessageBox_Busy);
|
|
return;
|
|
}
|
|
if (in_import) {
|
|
Notify(tr("Import is already running in the background."),STR_MessageBox_Busy);
|
|
return;
|
|
}
|
|
in_import=true;
|
|
|
|
ui->tabWidget->setCurrentWidget(welcome);
|
|
QApplication::processEvents();
|
|
|
|
QList<ImportPath> datacards = detectCPAPCards();
|
|
|
|
if (importScanCancelled) {
|
|
in_import = false;
|
|
return;
|
|
}
|
|
|
|
QList<MachineLoader *>loaders = GetLoaders(MT_CPAP);
|
|
|
|
QTime time;
|
|
time.start();
|
|
|
|
|
|
bool asknew = false;
|
|
|
|
if (datacards.size() > 0) {
|
|
MachineInfo info = datacards[0].loader->PeekInfo(datacards[0].path);
|
|
QString infostr;
|
|
if (!info.model.isEmpty()) {
|
|
QString infostr2 = info.model+" ("+info.serial+")";
|
|
infostr = tr("A %1 file structure for a %2 was located at:").arg(info.brand).arg(infostr2);
|
|
} else {
|
|
infostr = tr("A %1 file structure was located at:").arg(datacards[0].loader->loaderName());
|
|
}
|
|
|
|
if (!p_profile->cpap->autoImport()) {
|
|
QMessageBox mbox(QMessageBox::NoIcon,
|
|
tr("CPAP Data Located"), infostr+"\n\n"+QDir::toNativeSeparators(datacards[0].path)+"\n\n"+
|
|
tr("Would you like to import from this location?"),
|
|
QMessageBox::Yes | QMessageBox::No | QMessageBox::Cancel, this);
|
|
mbox.setDefaultButton(QMessageBox::Yes);
|
|
mbox.setButtonText(QMessageBox::No, tr("Specify"));
|
|
|
|
QPixmap pixmap = datacards[0].loader->getPixmap(datacards[0].loader->PeekInfo(datacards[0].path).series).scaled(64,64);
|
|
|
|
//QPixmap pixmap = QPixmap(getCPAPPixmap(datacards[0].loader->loaderName())).scaled(64,64);
|
|
mbox.setIconPixmap(pixmap);
|
|
int res = mbox.exec();
|
|
|
|
if (res == QMessageBox::Cancel) {
|
|
// Give the communal progress bar back
|
|
in_import=false;
|
|
return;
|
|
} else if (res == QMessageBox::No) {
|
|
//waitmsg->setText(tr("Please wait, launching file dialog..."));
|
|
datacards.clear();
|
|
asknew = true;
|
|
}
|
|
}
|
|
} else {
|
|
//waitmsg->setText(tr("No CPAP data card detected, launching file dialog..."));
|
|
asknew = true;
|
|
}
|
|
|
|
if (asknew) {
|
|
// popup.show();
|
|
mainwin->Notify(tr("Please remember to point the importer at the root folder or drive letter of your data-card, and not a subfolder."),
|
|
tr("Import Reminder"),8000);
|
|
|
|
QFileDialog w(this);
|
|
|
|
QString folder;
|
|
if (p_profile->contains(STR_PREF_LastCPAPPath)) {
|
|
folder = (*p_profile)[STR_PREF_LastCPAPPath].toString();
|
|
} else {
|
|
folder = QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation);
|
|
}
|
|
|
|
w.setDirectory(folder);
|
|
w.setFileMode(QFileDialog::Directory);
|
|
w.setOption(QFileDialog::ShowDirsOnly, true);
|
|
|
|
// This doesn't work on WinXP
|
|
|
|
#if defined(Q_OS_MAC)
|
|
w.setOption(QFileDialog::DontUseNativeDialog,false);
|
|
|
|
#elif defined(Q_OS_UNIX)
|
|
w.setOption(QFileDialog::DontUseNativeDialog,false);
|
|
#elif defined(Q_OS_WIN)
|
|
// check the Os version.. winxp chokes
|
|
w.setOption(QFileDialog::DontUseNativeDialog, true);
|
|
#endif
|
|
//#else
|
|
// w.setOption(QFileDialog::DontUseNativeDialog, false);
|
|
|
|
// QListView *l = w.findChild<QListView *>("listView");
|
|
// if (l) {
|
|
// l->setSelectionMode(QAbstractItemView::MultiSelection);
|
|
// }
|
|
|
|
// QTreeView *t = w.findChild<QTreeView *>();
|
|
// if (t) {
|
|
// t->setSelectionMode(QAbstractItemView::MultiSelection);
|
|
// }
|
|
|
|
//#endif
|
|
|
|
if (w.exec() != QDialog::Accepted) {
|
|
in_import=false;
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
for (int i = 0; i < w.selectedFiles().size(); i++) {
|
|
Q_FOREACH(MachineLoader * loader, loaders) {
|
|
if (loader->Detect(w.selectedFiles().at(i))) {
|
|
datacards.append(ImportPath(w.selectedFiles().at(i), loader));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
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();
|
|
}
|
|
|
|
if (c > 0) {
|
|
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();
|
|
in_import=false;
|
|
|
|
}
|
|
|
|
QMenu *MainWindow::CreateMenu(QString title)
|
|
{
|
|
QMenu *menu = new QMenu(title, ui->menubar);
|
|
ui->menubar->insertMenu(ui->menu_Help->menuAction(), menu);
|
|
return menu;
|
|
}
|
|
|
|
void MainWindow::on_action_Fullscreen_triggered()
|
|
{
|
|
if (ui->action_Fullscreen->isChecked()) {
|
|
this->showMaximized();
|
|
} else {
|
|
this->showNormal();
|
|
}
|
|
}
|
|
|
|
void MainWindow::setRecBoxHTML(QString html)
|
|
{
|
|
ui->recordsBox->setHtml(html);
|
|
}
|
|
|
|
void MainWindow::setStatsHTML(QString html)
|
|
{
|
|
ui->statisticsView->setHtml(html);
|
|
}
|
|
|
|
|
|
/***
|
|
QString MainWindow::getWelcomeHTML()
|
|
{
|
|
// This is messy, but allows it to be translated easier
|
|
return "<html>\n<head>"
|
|
" <style type='text/css'>"
|
|
" <!--h1,p,a,td,body { font-family: 'FreeSans', 'Sans Serif' } --/>"
|
|
" p,a,td,body { font-size: 14px }"
|
|
" a:link,a:visited { color: \"#000020\"; text-decoration: none; font-weight: bold;}"
|
|
" a:hover { background-color: inherit; color: red; text-decoration:none; font-weight: bold; }"
|
|
" </style>\n"
|
|
"</head>"
|
|
"<body leftmargin=0 topmargin=0 rightmargin=0>"
|
|
"<table width=\"100%\" cellspacing=0 cellpadding=4 border=0 >"
|
|
"<tr><td bgcolor=\"#d0d0d0\" colspan=2 cellpadding=0 valign=center align=center><font color=\"black\" size=+1><b>"
|
|
+ tr("Welcome to OSCAR") + "</b></font></td></tr>"
|
|
"<tr>"
|
|
"<td valign=\"top\" leftmargin=0 cellpadding=6>"
|
|
"<h3>" + tr("About OSCAR") + "</h3>"
|
|
"<p>" + tr("This software has been created to assist you in reviewing the data produced by CPAP Machines, used in the treatment of various Sleep Disorders.")
|
|
+ "</p>"
|
|
"<p>" + tr("OSCAR has been designed by a software developer with personal experience with a sleep disorder, and shaped by the feedback of many other willing testers dealing with similar conditions.")
|
|
+ "</p>"
|
|
"<p><i><b>" + tr("This is a beta release, some features may not yet behave as expected.") +
|
|
"</b></i><br/>" + tr("Please report any bugs you find to the OSCAR developer's group.") + "</p>"
|
|
|
|
"<h3>" + tr("Currenly supported machines:") + "</h3>"
|
|
"<b>" + tr("CPAP") + "</b>"
|
|
"<li>" + tr("Philips Respironics System One (CPAP Pro, Auto, BiPAP & ASV models)") + "</li>"
|
|
"<li>" + tr("ResMed S9 models (CPAP, Auto, VPAP)") + "</li>"
|
|
"<li>" + tr("DeVilbiss Intellipap (Auto)") + "</li>"
|
|
"<li>" + tr("Fisher & Paykel ICON (CPAP, Auto)") + "</li>"
|
|
"<b>" + tr("Oximetry") + "</b>"
|
|
"<li>" + tr("Contec CMS50D+, CMS50E and CMS50F (not 50FW) Oximeters") + "</li>"
|
|
"<li>" + tr("ResMed S9 Oximeter Attachment") + "</li>"
|
|
"<p><h3>" + tr("Online Help Resources") + "</h3></p>"
|
|
"<p><b>" + tr("Note:") + "</b>" +
|
|
tr("I don't recommend using this built in web browser to do any major surfing in, it will work, but it's mainly meant as a help browser.")
|
|
+
|
|
tr("(It doesn't support SSL encryption, so it's not a good idea to type your passwords or personal details anywhere.)")
|
|
+ "</p>" +
|
|
|
|
// tr("OSCAR's Online <a href=\"http://sleepyhead.sourceforge.net/wiki/index.php?title=OSCAR_Users_Guide\">Users Guide</a><br/>")
|
|
// +
|
|
// tr("<a href=\"http://sleepyhead.sourceforge.net/wiki/index.php?title=Frequently_Asked_Questions\">Frequently Asked Questions</a><br/>")
|
|
// +
|
|
// tr("<a href=\"http://sleepyhead.sourceforge.net/wiki/index.php?title=Glossary\">Glossary of Sleep Disorder Terms</a><br/>")
|
|
// +
|
|
// tr("<a href=\"http://sleepyhead.sourceforge.net/wiki/index.php?title=Main_Page\">OSCAR Wiki</a><br/>")
|
|
// +
|
|
// tr("OSCAR's <a href='http://www.sourceforge.net/projects/sleepyhead'>Project Website</a> on SourceForge<br/>")
|
|
// +
|
|
"<p><h3>" + tr("Further Information") + "</h3></p>"
|
|
"<p>" +
|
|
tr("The release notes for this version can be found in the About OSCAR menu item.") +
|
|
"<br/>" +
|
|
tr("Plus a few <a href='qrc:/docs/usage.html'>usage notes</a>, and some important information for Mac users.")
|
|
+ "<br/>" +
|
|
"<p>" + tr("About <a href='http://en.wikipedia.org/wiki/Sleep_apnea'>Sleep Apnea</a> on Wikipedia")
|
|
+ "</p>"
|
|
|
|
"<p>" + tr("Friendly forums to talk and learn about Sleep Apnea:") + "<br/>" +
|
|
tr("<a href='http://www.cpaptalk.com'>CPAPTalk Forum</a>,") +
|
|
tr("<a href='http://www.apneaboard.com/forums/'>Apnea Board</a>") + "</p>"
|
|
"</td>"
|
|
"<td><image src='qrc:/icons/logo-lg.png' width=220 height=220><br/>"
|
|
"</td>"
|
|
"</tr>"
|
|
"<tr>"
|
|
"<td colspan=2>"
|
|
"<hr/>"
|
|
"<p><b>" + tr("Copyright:") + "</b> " + "©2011-2018" +
|
|
" <a href=\"http://jedimark64.blogspot.com\">Mark Watkins</a> (jedimark) and portions ©2019 Nightowl Software</p>"
|
|
"<p><b>" + tr("License:") + "</b> " +
|
|
tr("This software is released freely under the <a href=\"qrc:/COPYING\">GNU Public License version 3</a>.") +
|
|
"</p>"
|
|
"<hr/>"
|
|
"<p><b>" + tr("DISCLAIMER:") + "</b></p>"
|
|
"<b><p>" +
|
|
tr("This is <font color='red'><u>NOT</u></font> medical software. This application is merely a data viewer, and no guarantee is made regarding accuracy or correctness of any calculations or data displayed.")
|
|
+ "</p>"
|
|
"<p>" + tr("The authors will NOT be held liable by anyone who harms themselves or others by use or misuse of this software.")
|
|
+ "</p>"
|
|
"<p>" + tr("Your doctor should always be your first and best source of guidance regarding the important matter of managing your health.")
|
|
+ "</p>"
|
|
"<p>" + tr("*** <u>Use at your own risk</u> ***") + "</p></b>"
|
|
"<hr/>"
|
|
"</td></tr>"
|
|
"</table>"
|
|
"</body>"
|
|
"</html>"
|
|
|
|
;
|
|
}
|
|
***/
|
|
void MainWindow::updateFavourites()
|
|
{
|
|
QDate date = p_profile->LastDay(MT_JOURNAL);
|
|
|
|
if (!date.isValid()) {
|
|
return;
|
|
}
|
|
|
|
QString html = "<html><head><style type='text/css'>"
|
|
"p,a,td,body { font-family: '" + QApplication::font().family() + "'; }"
|
|
"p,a,td,body { font-size: " + QString::number(QApplication::font().pointSize() + 2) + "px; }"
|
|
"a:link,a:visited { color: inherit; text-decoration: none; }" //font-weight: normal;
|
|
"a:hover { background-color: inherit; color: white; text-decoration:none; font-weight: bold; }"
|
|
"</style></head><body>"
|
|
"<table width=100% cellpadding=2 cellspacing=0>";
|
|
|
|
do {
|
|
Day *journal = p_profile->GetDay(date, MT_JOURNAL);
|
|
|
|
if (journal) {
|
|
if (journal->size() > 0) {
|
|
Session *sess = journal->firstSession(MT_JOURNAL);
|
|
QString tmp;
|
|
bool filtered = !bookmarkFilter.isEmpty();
|
|
bool found = !filtered;
|
|
|
|
if (sess->settings.contains(Bookmark_Start)) {
|
|
//QVariantList start=sess->settings[Bookmark_Start].toList();
|
|
//QVariantList end=sess->settings[Bookmark_End].toList();
|
|
QStringList notes = sess->settings[Bookmark_Notes].toStringList();
|
|
|
|
if (notes.size() > 0) {
|
|
tmp += QString("<tr><td><b><a href='daily=%1'>%2</a></b><br/>")
|
|
.arg(date.toString(Qt::ISODate))
|
|
.arg(date.toString(MedDateFormat));
|
|
|
|
tmp += "<list>";
|
|
|
|
for (int i = 0; i < notes.size(); i++) {
|
|
//QDate d=start[i].toDate();
|
|
QString note = notes[i];
|
|
|
|
if (filtered && note.contains(bookmarkFilter, Qt::CaseInsensitive)) {
|
|
found = true;
|
|
}
|
|
|
|
tmp += "<li>" + note + "</li>";
|
|
}
|
|
|
|
tmp += "</list></td>";
|
|
}
|
|
}
|
|
|
|
if (found) { html += tmp; }
|
|
}
|
|
}
|
|
|
|
date = date.addDays(-1);
|
|
} while (date >= p_profile->FirstDay(MT_JOURNAL));
|
|
|
|
html += "</table></body></html>";
|
|
ui->bookmarkView->setHtml(html);
|
|
}
|
|
|
|
void MainWindow::on_dailyButton_clicked()
|
|
{
|
|
if (daily) {
|
|
ui->tabWidget->setCurrentWidget(daily);
|
|
daily->RedrawGraphs();
|
|
}
|
|
}
|
|
void MainWindow::on_overviewButton_clicked()
|
|
{
|
|
if (overview) {
|
|
ui->tabWidget->setCurrentWidget(overview);
|
|
}
|
|
}
|
|
|
|
void MainWindow::aboutBoxLinkClicked(const QUrl &url)
|
|
{
|
|
QDesktopServices::openUrl(url);
|
|
}
|
|
|
|
void MainWindow::on_action_About_triggered()
|
|
{
|
|
AboutDialog * about = new AboutDialog(this);
|
|
|
|
about->exec();
|
|
|
|
about->deleteLater();
|
|
}
|
|
|
|
void MainWindow::on_actionDebug_toggled(bool checked)
|
|
{
|
|
AppSetting->setShowDebug(checked);
|
|
|
|
logger->strlock.lock();
|
|
if (checked) {
|
|
ui->logText->show();
|
|
} else {
|
|
ui->logText->hide();
|
|
}
|
|
// QApplication::processEvents();
|
|
logger->strlock.unlock();
|
|
}
|
|
|
|
void MainWindow::on_action_Reset_Graph_Layout_triggered()
|
|
{
|
|
if (daily && (ui->tabWidget->currentWidget() == daily)) { daily->ResetGraphLayout(); }
|
|
|
|
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()
|
|
{
|
|
if (daily && (ui->tabWidget->currentWidget() == daily)) { daily->ResetGraphOrder(1); }
|
|
|
|
if (overview && (ui->tabWidget->currentWidget() == overview)) { overview->ResetGraphOrder(1); }
|
|
}
|
|
|
|
void MainWindow::on_action_Advanced_Graph_Order_triggered()
|
|
{
|
|
if (daily && (ui->tabWidget->currentWidget() == daily)) { daily->ResetGraphOrder(2); }
|
|
|
|
if (overview && (ui->tabWidget->currentWidget() == overview)) { overview->ResetGraphOrder(2); }
|
|
}
|
|
|
|
void MainWindow::on_action_Preferences_triggered()
|
|
{
|
|
if (!p_profile) {
|
|
mainwin->Notify(tr("Please open a profile first."));
|
|
return;
|
|
}
|
|
|
|
if (m_inRecalculation) {
|
|
mainwin->Notify(tr("Access to Preferences has been blocked until recalculation completes."));
|
|
return;
|
|
}
|
|
|
|
PreferencesDialog pd(this, p_profile);
|
|
prefdialog = &pd;
|
|
|
|
if (pd.exec() == PreferencesDialog::Accepted) {
|
|
// Apply any updates from preference changes (notably fonts)
|
|
|
|
setApplicationFont();
|
|
|
|
if (daily) {
|
|
daily->RedrawGraphs();
|
|
}
|
|
|
|
if (overview) {
|
|
overview->ResetFont();
|
|
overview->RebuildGraphs(true);
|
|
}
|
|
|
|
if (welcome)
|
|
welcome->refreshPage();
|
|
|
|
if (profileSelector)
|
|
profileSelector->updateProfileList();
|
|
|
|
// These attempts to update calendar after a change to application font do NOT work, and I can find no QT documentation suggesting
|
|
// that changing the font after Calendar is created is even possible.
|
|
// qDebug() << "application font family set to" << QApplication::font().family() << "and font" << QApplication::font();
|
|
// ui->statStartDate->calendarWidget()->setFont(QApplication::font());
|
|
// ui->statStartDate->calendarWidget()->repaint();
|
|
}
|
|
|
|
prefdialog = nullptr;
|
|
}
|
|
|
|
#include "oximeterimport.h"
|
|
QDateTime datetimeDialog(QDateTime datetime, QString message);
|
|
|
|
void MainWindow::on_oximetryButton_clicked()
|
|
{
|
|
if (p_profile) {
|
|
OximeterImport oxiimp(this);
|
|
oxiimp.exec();
|
|
if (overview) overview->ReloadGraphs();
|
|
if (welcome) welcome->refreshPage();
|
|
}
|
|
}
|
|
|
|
void MainWindow::CheckForUpdates()
|
|
{
|
|
qDebug() << "procedure <CheckForUpdates> 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()
|
|
{
|
|
qDebug() << "procedure <on_actionCheck_for_Updates_triggered> called";
|
|
#ifndef NO_UPDATER
|
|
UpdaterWindow *w = new UpdaterWindow(this);
|
|
w->checkForUpdates();
|
|
#endif
|
|
}
|
|
|
|
bool toolbox_visible = false;
|
|
void MainWindow::on_action_Screenshot_triggered()
|
|
{
|
|
setUpdatesEnabled(false);
|
|
if (daily)
|
|
daily->hideSpaceHogs();
|
|
toolbox_visible = ui->toolBox->isVisible();
|
|
ui->toolBox->hide();
|
|
setUpdatesEnabled(true);
|
|
QTimer::singleShot(250, this, SLOT(DelayedScreenshot()));
|
|
}
|
|
|
|
void MainWindow::DelayedScreenshot()
|
|
{
|
|
// Make sure to scale for high resolution displays (like Retina)
|
|
// qreal pr = devicePixelRatio();
|
|
|
|
auto screenshotRect = geometry();
|
|
auto titleBarHeight = QApplication::style()->pixelMetric(QStyle::PM_TitleBarHeight);
|
|
auto pixmap = QApplication::primaryScreen()->grabWindow(QDesktopWidget().winId(),
|
|
screenshotRect.left(),
|
|
screenshotRect.top() - titleBarHeight,
|
|
screenshotRect.width(),
|
|
screenshotRect.height() + titleBarHeight);
|
|
|
|
QString a = p_pref->Get("{home}/Screenshots");
|
|
QDir dir(a);
|
|
|
|
if (!dir.exists()) {
|
|
dir.mkdir(a);
|
|
}
|
|
|
|
a += "/screenshot-" + QDateTime::currentDateTime().toString("yyyyMMdd-hhmmss") + ".png";
|
|
|
|
qDebug() << "Saving screenshot to" << a;
|
|
|
|
if (!pixmap.save(a)) {
|
|
Notify(tr("There was an error saving screenshot to file \"%1\"").arg(QDir::toNativeSeparators(a)));
|
|
} else {
|
|
Notify(tr("Screenshot saved to file \"%1\"").arg(QDir::toNativeSeparators(a)));
|
|
}
|
|
|
|
setUpdatesEnabled(false);
|
|
if (daily)
|
|
daily->showSpaceHogs();
|
|
ui->toolBox->setVisible(toolbox_visible);
|
|
setUpdatesEnabled(true);
|
|
}
|
|
|
|
void MainWindow::on_actionView_Oximetry_triggered()
|
|
{
|
|
on_oximetryButton_clicked();
|
|
}
|
|
|
|
void MainWindow::on_actionPrint_Report_triggered()
|
|
{
|
|
Report report;
|
|
|
|
if (ui->tabWidget->currentWidget() == overview) {
|
|
Report::PrintReport(overview->graphView(), STR_TR_Overview);
|
|
} else if (ui->tabWidget->currentWidget() == daily) {
|
|
Report::PrintReport(daily->graphView(), STR_TR_Daily, daily->getDate());
|
|
} else if (ui->tabWidget->currentWidget() == ui->statisticsTab) {
|
|
Statistics::printReport(this);
|
|
#ifndef helpless
|
|
} else if (ui->tabWidget->currentWidget() == help) {
|
|
help->print(&printer); // **** THIS DID NOT SURVIVE REFACTORING STATISTICS PRINT
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_action_Edit_Profile_triggered()
|
|
{
|
|
NewProfile *newprof = new NewProfile(this);
|
|
QString name = AppSetting->profileName();
|
|
newprof->edit(name);
|
|
newprof->setWindowModality(Qt::ApplicationModal);
|
|
newprof->setModal(true);
|
|
newprof->exec();
|
|
qDebug() << newprof;
|
|
delete newprof;
|
|
}
|
|
|
|
void MainWindow::on_action_CycleTabs_triggered()
|
|
{
|
|
int i;
|
|
qDebug() << "Switching Tabs";
|
|
i = ui->tabWidget->currentIndex() + 1;
|
|
|
|
if (i >= ui->tabWidget->count()) {
|
|
i = 0;
|
|
}
|
|
|
|
ui->tabWidget->setCurrentIndex(i);
|
|
}
|
|
|
|
void MainWindow::on_actionOnline_Users_Guide_triggered()
|
|
{
|
|
// 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"));
|
|
if (QMessageBox::question(nullptr, STR_MessageBox_Question, tr("The User's Guide will open in your default browser"),
|
|
QMessageBox::Ok|QMessageBox::Cancel, QMessageBox::Ok) == QMessageBox::Ok )
|
|
QDesktopServices::openUrl(QUrl("https://www.apneaboard.com/wiki/index.php?title=OSCAR_Help"));
|
|
}
|
|
|
|
void MainWindow::on_action_Frequently_Asked_Questions_triggered()
|
|
{
|
|
// QDesktopServices::openUrl(QUrl("http://sleepyhead.sourceforge.net/wiki/index.php?title=Frequently_Asked_Questions"));
|
|
QMessageBox::information(nullptr, STR_MessageBox_Information, tr("The FAQ is not yet implemented"));
|
|
}
|
|
|
|
void packEventList(EventList *el, EventDataType minval = 0)
|
|
{
|
|
if (el->count() < 2) { return; }
|
|
|
|
EventList nel(EVL_Waveform);
|
|
EventDataType t = 0, lastt = 0; //el->data(0);
|
|
qint64 ti = 0; //=el->time(0);
|
|
//nel.AddEvent(ti,lastt);
|
|
bool f = false;
|
|
qint64 lasttime = 0;
|
|
EventDataType min = 999, max = 0;
|
|
|
|
for (quint32 i = 0; i < el->count(); i++) {
|
|
t = el->data(i);
|
|
ti = el->time(i);
|
|
f = false;
|
|
|
|
if (t > minval) {
|
|
if (t != lastt) {
|
|
if (!lasttime) {
|
|
nel.setFirst(ti);
|
|
}
|
|
|
|
nel.AddEvent(ti, t);
|
|
|
|
if (t < min) { min = t; }
|
|
|
|
if (t > max) { max = t; }
|
|
|
|
lasttime = ti;
|
|
f = true;
|
|
}
|
|
} else {
|
|
if (lastt > minval) {
|
|
nel.AddEvent(ti, lastt);
|
|
lasttime = ti;
|
|
f = true;
|
|
}
|
|
}
|
|
|
|
lastt = t;
|
|
}
|
|
|
|
if (!f) {
|
|
if (t > minval) {
|
|
nel.AddEvent(ti, t);
|
|
}
|
|
}
|
|
|
|
el->setFirst(nel.first());
|
|
el->setLast(nel.last());
|
|
el->setMin(min);
|
|
el->setMax(max);
|
|
|
|
el->getData().clear();
|
|
el->getTime().clear();
|
|
el->setCount(nel.count());
|
|
|
|
el->getData() = nel.getData();
|
|
el->getTime() = nel.getTime();
|
|
}
|
|
|
|
void MainWindow::on_action_Rebuild_Oximetry_Index_triggered()
|
|
{
|
|
QVector<ChannelID> valid;
|
|
valid.push_back(OXI_Pulse);
|
|
valid.push_back(OXI_SPO2);
|
|
valid.push_back(OXI_Plethy);
|
|
//valid.push_back(OXI_PulseChange); // Delete these and recalculate..
|
|
//valid.push_back(OXI_SPO2Drop);
|
|
|
|
QVector<ChannelID> invalid;
|
|
|
|
QList<Machine *> machines = p_profile->GetMachines(MT_OXIMETER);
|
|
|
|
qint64 f = 0, l = 0;
|
|
|
|
int discard_threshold = p_profile->oxi->oxiDiscardThreshold();
|
|
Machine *m;
|
|
|
|
for (int z = 0; z < machines.size(); z++) {
|
|
m = machines.at(z);
|
|
//m->sessionlist.erase(m->sessionlist.find(0));
|
|
|
|
// For each Session
|
|
for (QHash<SessionID, Session *>::iterator s = m->sessionlist.begin(); s != m->sessionlist.end();
|
|
s++) {
|
|
Session *sess = s.value();
|
|
|
|
if (!sess) { continue; }
|
|
|
|
sess->OpenEvents();
|
|
|
|
// For each EventList contained in session
|
|
invalid.clear();
|
|
f = 0;
|
|
l = 0;
|
|
|
|
for (QHash<ChannelID, QVector<EventList *> >::iterator e = sess->eventlist.begin();
|
|
e != sess->eventlist.end(); e++) {
|
|
|
|
// Discard any non data events.
|
|
if (!valid.contains(e.key())) {
|
|
// delete and push aside for later to clean up
|
|
for (int i = 0; i < e.value().size(); i++) {
|
|
delete e.value()[i];
|
|
}
|
|
|
|
e.value().clear();
|
|
invalid.push_back(e.key());
|
|
} else {
|
|
// Valid event
|
|
|
|
|
|
// // Clean up outliers at start of eventlist chunks
|
|
// EventDataType baseline=sess->wavg(OXI_SPO2);
|
|
// if (e.key()==OXI_SPO2) {
|
|
// const int o2start_threshold=10000; // seconds since start of event
|
|
|
|
// EventDataType zz;
|
|
// int ii;
|
|
|
|
// // Trash suspect outliers in the first o2start_threshold milliseconds
|
|
// for (int j=0;j<e.value().size();j++) {
|
|
// EventList *ev=e.value()[j];
|
|
// if ((ev->count() <= (unsigned)discard_threshold))
|
|
// continue;
|
|
|
|
// qint64 ti=ev->time(0);
|
|
|
|
// zz=-1;
|
|
// // Peek o2start_threshold ms ahead and grab the value
|
|
// for (ii=0;ii<ev->count();ii++) {
|
|
// if (((ev->time(ii)-ti) > o2start_threshold)) {
|
|
// zz=ev->data(ii);
|
|
// break;
|
|
// }
|
|
// }
|
|
// if (zz<0)
|
|
// continue;
|
|
// // Trash any suspect outliers
|
|
// for (int i=0;i<ii;i++) {
|
|
// if (ev->data(i) < baseline) { //(zz-10)) {
|
|
|
|
// ev->getData()[i]=0;
|
|
// }
|
|
// }
|
|
// }
|
|
// }
|
|
QVector<EventList *> newlist;
|
|
|
|
for (int i = 0; i < e.value().size(); i++) {
|
|
if (e.value()[i]->count() > (unsigned)discard_threshold) {
|
|
newlist.push_back(e.value()[i]);
|
|
} else {
|
|
delete e.value()[i];
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < newlist.size(); i++) {
|
|
packEventList(newlist[i], 8);
|
|
|
|
EventList *el = newlist[i];
|
|
|
|
if (!f || f > el->first()) { f = el->first(); }
|
|
|
|
if (!l || l < el->last()) { l = el->last(); }
|
|
}
|
|
|
|
e.value() = newlist;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < invalid.size(); i++) {
|
|
sess->eventlist.erase(sess->eventlist.find(invalid[i]));
|
|
}
|
|
|
|
if (f) { sess->really_set_first(f); }
|
|
|
|
if (l) { sess->really_set_last(l); }
|
|
|
|
sess->m_cnt.clear();
|
|
sess->m_sum.clear();
|
|
sess->m_min.clear();
|
|
sess->m_max.clear();
|
|
sess->m_cph.clear();
|
|
sess->m_sph.clear();
|
|
sess->m_avg.clear();
|
|
sess->m_wavg.clear();
|
|
sess->m_valuesummary.clear();
|
|
sess->m_timesummary.clear();
|
|
sess->m_firstchan.clear();
|
|
sess->m_lastchan.clear();
|
|
sess->SetChanged(true);
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < machines.size(); i++) {
|
|
Machine *m = machines[i];
|
|
m->Save();
|
|
m->SaveSummaryCache();
|
|
}
|
|
|
|
daily->LoadDate(daily->getDate());
|
|
overview->ReloadGraphs();
|
|
}
|
|
void MainWindow::reloadProfile()
|
|
{
|
|
QString username = p_profile->user->userName();
|
|
int tabidx = ui->tabWidget->currentIndex();
|
|
CloseProfile();
|
|
OpenProfile(username);
|
|
ui->tabWidget->setCurrentIndex(tabidx);
|
|
}
|
|
|
|
void MainWindow::RestartApplication(bool force_login, QString cmdline)
|
|
{
|
|
CloseProfile();
|
|
p_pref->Save();
|
|
|
|
QString apppath;
|
|
#ifdef Q_OS_MAC
|
|
// In Mac OS the full path of aplication binary is:
|
|
// <base-path>/myApp.app/Contents/MacOS/myApp
|
|
|
|
// prune the extra bits to just get the app bundle path
|
|
apppath = QApplication::instance()->applicationDirPath().section("/", 0, -3);
|
|
|
|
QStringList args;
|
|
args << "-n"; // -n option is important, as it opens a new process
|
|
args << apppath;
|
|
|
|
args << "--args"; // OSCAR binary options after this
|
|
args << "-p"; // -p starts with 1 second delay, to give this process time to save..
|
|
|
|
|
|
if (force_login) { args << "-l"; }
|
|
|
|
args << cmdline;
|
|
|
|
if (QProcess::startDetached("/usr/bin/open", args)) {
|
|
QApplication::instance()->exit();
|
|
} else {
|
|
QMessageBox::warning(nullptr, tr("Gah!"),
|
|
tr("If you can read this, the restart command didn't work. You will have to do it yourself manually."), QMessageBox::Ok);
|
|
}
|
|
|
|
#else
|
|
apppath = QApplication::instance()->applicationFilePath();
|
|
|
|
// If this doesn't work on windoze, try uncommenting this method
|
|
// Technically should be the same thing..
|
|
|
|
//if (QDesktopServices::openUrl(apppath)) {
|
|
// QApplication::instance()->exit();
|
|
//} else
|
|
QStringList args;
|
|
args << "-p";
|
|
|
|
if (force_login) { args << "-l"; }
|
|
|
|
args << cmdline;
|
|
|
|
//if (change_datafolder) { args << "-d"; }
|
|
|
|
if (QProcess::startDetached(apppath, args)) {
|
|
QApplication::instance()->exit();
|
|
|
|
// ::exit(0);
|
|
} else {
|
|
QMessageBox::warning(nullptr, tr("Gah!"),
|
|
tr("If you can read this, the restart command didn't work. You will have to do it yourself manually."), QMessageBox::Ok);
|
|
}
|
|
|
|
#endif
|
|
}
|
|
|
|
void MainWindow::on_actionPurge_Current_Day_triggered()
|
|
{
|
|
if (!daily) return;
|
|
QDate date = daily->getDate();
|
|
daily->Unload(date);
|
|
Day *day = p_profile->GetDay(date, MT_CPAP);
|
|
Machine *cpap = nullptr;
|
|
if (day) cpap = day->machine(MT_CPAP);
|
|
|
|
if (cpap) {
|
|
QList<Session *>::iterator s;
|
|
|
|
QList<Session *> list;
|
|
QList<SessionID> sidlist;
|
|
for (s = day->begin(); s != day->end(); ++s) {
|
|
list.push_back(*s);
|
|
sidlist.push_back((*s)->session());
|
|
}
|
|
|
|
QHash<QString, SessionID> skipfiles;
|
|
// Read the already imported file list
|
|
|
|
QFile impfile(cpap->getDataPath()+"/imported_files.csv");
|
|
if (impfile.exists()) {
|
|
if (impfile.open(QFile::ReadOnly)) {
|
|
QTextStream impstream(&impfile);
|
|
QString serial;
|
|
impstream >> serial;
|
|
if (cpap->serial() == serial) {
|
|
QString line, file, str;
|
|
SessionID sid;
|
|
bool ok;
|
|
do {
|
|
line = impstream.readLine();
|
|
file = line.section(',',0,0);
|
|
str = line.section(',',1);
|
|
sid = str.toInt(&ok);
|
|
if (!sidlist.contains(sid)) {
|
|
skipfiles[file] = sid;
|
|
}
|
|
} while (!impstream.atEnd());
|
|
}
|
|
}
|
|
impfile.close();
|
|
// Delete the file
|
|
impfile.remove();
|
|
|
|
// Rewrite the file without the sessions being removed.
|
|
if (impfile.open(QFile::WriteOnly)) {
|
|
QTextStream out(&impfile);
|
|
out << cpap->serial();
|
|
QHash<QString, SessionID>::iterator skit;
|
|
QHash<QString, SessionID>::iterator skit_end = skipfiles.end();
|
|
for (skit = skipfiles.begin(); skit != skit_end; ++skit) {
|
|
QString a = QString("%1,%2\n").arg(skit.key()).arg(skit.value());;
|
|
out << a;
|
|
}
|
|
out.flush();
|
|
}
|
|
impfile.close();
|
|
}
|
|
|
|
QFile rxcache(p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" ));
|
|
rxcache.remove();
|
|
|
|
QFile sumfile(cpap->getDataPath()+"Summaries.xml.gz");
|
|
sumfile.remove();
|
|
|
|
// m->day.erase(m->day.find(date));
|
|
|
|
for (int i = 0; i < list.size(); i++) {
|
|
Session *sess = list.at(i);
|
|
sess->Destroy();
|
|
delete sess;
|
|
}
|
|
|
|
}
|
|
day = p_profile->GetDay(date, MT_CPAP);
|
|
Q_UNUSED(day);
|
|
|
|
daily->clearLastDay();
|
|
daily->LoadDate(date);
|
|
}
|
|
|
|
void MainWindow::on_actionRebuildCPAP(QAction *action)
|
|
{
|
|
ui->tabWidget->setCurrentWidget(welcome); // Daily view can't run during rebuild
|
|
QApplication::processEvents();
|
|
|
|
QString data = action->data().toString();
|
|
QString cls = data.section(":",0,0);
|
|
QString serial = data.section(":", 1);
|
|
QList<Machine *> machines = p_profile->GetMachines(MT_CPAP);
|
|
Machine * mach = nullptr;
|
|
for (int i=0; i < machines.size(); ++i) {
|
|
Machine * m = machines.at(i);
|
|
if ((m->loaderName() == cls) && (m->serial() == serial)) {
|
|
mach = m;
|
|
break;
|
|
}
|
|
}
|
|
if (!mach) return;
|
|
QString bpath = mach->getBackupPath();
|
|
bool backups = (dirCount(bpath) > 0) ? true : false;
|
|
|
|
if (backups) {
|
|
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 data if OSCAR's backups have been disabled."),
|
|
QMessageBox::Yes | QMessageBox::No, QMessageBox::No) == QMessageBox::No) {
|
|
return;
|
|
}
|
|
} else {
|
|
if (QMessageBox::question(this,
|
|
STR_MessageBox_Warning,
|
|
"<p><b>"+STR_MessageBox_Warning+": </b>"+tr("For some reason, OSCAR does not have any backups for the following machine:")+ "</p>" +
|
|
"<p>"+mach->brand() + " " + mach->model() + " " + mach->modelnumber() + " (" + mach->serial() + ")</p>"+
|
|
"<p>"+tr("Provided you have made <i>your <b>own</b> backups for ALL of your CPAP data</i>, you can still complete this operation, but you will have to restore from your backups manually.")+"</p>" +
|
|
"<p><b>"+tr("Are you really sure you want to do this?")+"</b></p>",
|
|
QMessageBox::Yes | QMessageBox::No, QMessageBox::No) == QMessageBox::No) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
QString path = mach->getBackupPath();
|
|
MachineLoader *loader = lookupLoader(mach);
|
|
|
|
purgeMachine(mach); // purge destroys machine record
|
|
|
|
if (backups) {
|
|
importCPAP(ImportPath(path, loader), tr("Please wait, importing from backup folder(s)..."));
|
|
} else {
|
|
if (QMessageBox::information(this, STR_MessageBox_Warning,
|
|
tr("Because there are no internal backups to rebuild from, you will have to restore from your own.")+"\n\n"+
|
|
tr("Would you like to import from your own backups now? (you will have no data visible for this machine until you do)"),
|
|
QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes) == QMessageBox::Yes) {
|
|
on_action_Import_Data_triggered();
|
|
} else {
|
|
}
|
|
}
|
|
if (overview) overview->ReloadGraphs();
|
|
if (daily) {
|
|
daily->Unload();
|
|
daily->clearLastDay(); // otherwise Daily will crash
|
|
daily->ReloadGraphs();
|
|
}
|
|
if (welcome) welcome->refreshPage();
|
|
PopulatePurgeMenu();
|
|
GenerateStatistics();
|
|
p_profile->StoreMachines();
|
|
}
|
|
|
|
void MainWindow::on_actionPurgeMachine(QAction *action)
|
|
{
|
|
QString data = action->data().toString();
|
|
QString cls = data.section(":",0,0);
|
|
QString serial = data.section(":", 1);
|
|
QList<Machine *> machines = p_profile->GetMachines(MT_CPAP);
|
|
Machine * mach = nullptr;
|
|
for (int i=0; i < machines.size(); ++i) {
|
|
Machine * m = machines.at(i);
|
|
if ((m->loaderName() == cls) && (m->serial() == serial)) {
|
|
mach = m;
|
|
break;
|
|
}
|
|
}
|
|
if (!mach) return;
|
|
|
|
if (QMessageBox::question(this, STR_MessageBox_Warning,
|
|
"<p><b>"+STR_MessageBox_Warning+":</b> " +
|
|
tr("You are about to <font size=+2>obliterate</font> OSCAR's machine database for the following machine:</p>") +
|
|
"<p>"+mach->brand() + " " + mach->model() + " " + mach->modelnumber() + " (" + mach->serial() + ")" + "</p>" +
|
|
"<p>"+tr("Note as a precaution, the backup folder will be left in place.")+"</p>"+
|
|
"<p>"+tr("Are you <b>absolutely sure</b> you want to proceed?")+"</p>",
|
|
QMessageBox::Yes | QMessageBox::No, QMessageBox::No) == QMessageBox::Yes) {
|
|
purgeMachine(mach);
|
|
}
|
|
|
|
}
|
|
|
|
void MainWindow::purgeMachine(Machine * mach)
|
|
{
|
|
// detect backups
|
|
daily->Unload(daily->getDate());
|
|
|
|
// Technicially the above won't sessions under short session limit.. Using Purge to clean up the rest.
|
|
if (mach->Purge(3478216)) {
|
|
mach->sessionlist.clear();
|
|
mach->day.clear();
|
|
QDir dir;
|
|
QString path = mach->getDataPath();
|
|
qDebug() << "path to machine" << path;
|
|
path.chop(1);
|
|
|
|
p_profile->DelMachine(mach);
|
|
delete mach;
|
|
// remove the directory unless it's got unexpected crap in it..
|
|
bool deleted = false;
|
|
if (!dir.remove(path)) {
|
|
#ifdef Q_OS_WIN
|
|
wchar_t* directoryPtr = (wchar_t*)path.utf16();
|
|
SetFileAttributes(directoryPtr, GetFileAttributes(directoryPtr) & ~FILE_ATTRIBUTE_READONLY);
|
|
if (!::RemoveDirectory(directoryPtr)) {
|
|
DWORD lastError = ::GetLastError();
|
|
qDebug() << "RemoveDirectory" << path << "GetLastError: " << lastError << "(Error 145 is expected)";
|
|
if (lastError == 145) {
|
|
qDebug() << path << "remaining directory contents are" << QDir(path).entryList();
|
|
}
|
|
|
|
} else {
|
|
qDebug() << "Success on second attempt deleting folder with windows API " << path;
|
|
deleted = true;
|
|
}
|
|
#else
|
|
qDebug() << "Couldn't remove directory" << path;
|
|
#endif
|
|
} else {
|
|
deleted = true;
|
|
}
|
|
if (!deleted) {
|
|
qDebug() << "Leaving backup folder intact";
|
|
}
|
|
|
|
PopulatePurgeMenu();
|
|
p_profile->StoreMachines();
|
|
} else {
|
|
QMessageBox::warning(this, STR_MessageBox_Error,
|
|
tr("A file permission error casued the purge process to fail; you will have to delete the following folder manually:") +
|
|
"\n\n" + QDir::toNativeSeparators(mach->getDataPath()), QMessageBox::Ok, QMessageBox::Ok);
|
|
|
|
if (overview)
|
|
overview->ReloadGraphs();
|
|
|
|
if (daily) {
|
|
daily->clearLastDay(); // otherwise Daily will crash
|
|
daily->ReloadGraphs();
|
|
}
|
|
if (welcome)
|
|
welcome->refreshPage();
|
|
|
|
//GenerateStatistics();
|
|
return;
|
|
}
|
|
|
|
|
|
if (overview)
|
|
overview->ReloadGraphs();
|
|
QFile rxcache(p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" ));
|
|
rxcache.remove();
|
|
|
|
if (daily) {
|
|
daily->clearLastDay(); // otherwise Daily will crash
|
|
daily->ReloadGraphs();
|
|
}
|
|
if (welcome)
|
|
welcome->refreshPage();
|
|
|
|
QApplication::processEvents();
|
|
}
|
|
|
|
void MainWindow::keyPressEvent(QKeyEvent *event)
|
|
{
|
|
Q_UNUSED(event)
|
|
//qDebug() << "Keypress:" << event->key();
|
|
}
|
|
|
|
void MainWindow::on_action_Sidebar_Toggle_toggled(bool visible)
|
|
{
|
|
ui->toolBox->setVisible(visible);
|
|
AppSetting->setRightSidebarVisible(visible);
|
|
}
|
|
|
|
void MainWindow::on_helpButton_clicked()
|
|
{
|
|
#ifndef helpless
|
|
ui->tabWidget->setCurrentWidget(help);
|
|
#else
|
|
QMessageBox::information(nullptr, STR_MessageBox_Error, tr("No help is available."));
|
|
#endif
|
|
}
|
|
|
|
void MainWindow::on_actionView_Statistics_triggered()
|
|
{
|
|
ui->tabWidget->setCurrentWidget(ui->statisticsTab);
|
|
}
|
|
|
|
void MainWindow::on_tabWidget_currentChanged(int index)
|
|
{
|
|
Q_UNUSED(index);
|
|
// QWidget *widget = ui->tabWidget->currentWidget();
|
|
}
|
|
|
|
|
|
void MainWindow::on_filterBookmarks_editingFinished()
|
|
{
|
|
bookmarkFilter = ui->filterBookmarks->text();
|
|
updateFavourites();
|
|
}
|
|
|
|
void MainWindow::on_filterBookmarksButton_clicked()
|
|
{
|
|
if (!bookmarkFilter.isEmpty()) {
|
|
ui->filterBookmarks->setText("");
|
|
bookmarkFilter = "";
|
|
updateFavourites();
|
|
}
|
|
}
|
|
|
|
void MainWindow::recompressEvents()
|
|
{
|
|
QTimer::singleShot(0, this, SLOT(doRecompressEvents()));
|
|
}
|
|
void MainWindow::reprocessEvents(bool restart)
|
|
{
|
|
m_restartRequired = restart;
|
|
QTimer::singleShot(0, this, SLOT(doReprocessEvents()));
|
|
}
|
|
|
|
|
|
void MainWindow::FreeSessions()
|
|
{
|
|
QDate first = p_profile->FirstDay();
|
|
QDate date = p_profile->LastDay();
|
|
Day *day;
|
|
QDate current = daily->getDate();
|
|
|
|
do {
|
|
day = p_profile->GetDay(date, MT_CPAP);
|
|
|
|
if (day) {
|
|
if (date != current) {
|
|
day->CloseEvents();
|
|
}
|
|
}
|
|
|
|
date = date.addDays(-1);
|
|
} while (date >= first);
|
|
}
|
|
|
|
void MainWindow::MachineUnsupported(Machine * m)
|
|
{
|
|
if (m == nullptr) {
|
|
qCritical() << "MainWindow::MachineUnsupported called with null machine object";
|
|
return;
|
|
}
|
|
QMessageBox::information(this, STR_MessageBox_Error, QObject::tr("Sorry, your %1 %2 machine is not currently supported.").arg(m->brand()).arg(m->model()), QMessageBox::Ok);
|
|
}
|
|
|
|
void MainWindow::doRecompressEvents()
|
|
{
|
|
if (!p_profile) return;
|
|
ProgressDialog progress(this);
|
|
progress.setMessage("Recompressing Session Files");
|
|
progress.setProgressMax(p_profile->daylist.size());
|
|
QPixmap icon = QPixmap(":/icons/logo-md.png").scaled(64,64);
|
|
progress.setPixmap(icon);
|
|
progress.open();
|
|
|
|
bool isopen;
|
|
int idx = 0;
|
|
for (Day * day : p_profile->daylist) {
|
|
for (Session * sess : day->sessions) {
|
|
isopen = sess->eventsLoaded();
|
|
// Load the events and summary if they aren't loaded already
|
|
sess->LoadSummary();
|
|
sess->OpenEvents();
|
|
sess->SetChanged(true);
|
|
sess->machine()->SaveSession(sess);
|
|
|
|
if (!isopen) {
|
|
sess->TrashEvents();
|
|
}
|
|
}
|
|
progress.setProgressValue(++idx);
|
|
QApplication::processEvents();
|
|
}
|
|
progress.close();
|
|
}
|
|
void MainWindow::doReprocessEvents()
|
|
{
|
|
if (!p_profile) return;
|
|
|
|
ProgressDialog progress(this);
|
|
progress.setMessage("Recalculating summaries");
|
|
progress.setProgressMax(p_profile->daylist.size());
|
|
QPixmap icon = QPixmap(":/icons/logo-md.png").scaled(64,64);
|
|
progress.setPixmap(icon);
|
|
progress.open();
|
|
|
|
if (daily) {
|
|
daily->Unload();
|
|
daily->clearLastDay(); // otherwise Daily will crash
|
|
delete daily;
|
|
daily = nullptr;
|
|
}
|
|
if (welcome) {
|
|
delete welcome;
|
|
welcome = nullptr;
|
|
}
|
|
if (overview) {
|
|
delete overview;
|
|
overview = nullptr;
|
|
}
|
|
|
|
for (Day * day : p_profile->daylist) {
|
|
for (Session * sess : day->sessions) {
|
|
bool isopen = sess->eventsLoaded();
|
|
|
|
// Load the events if they aren't loaded already
|
|
sess->LoadSummary();
|
|
sess->OpenEvents();
|
|
|
|
// Destroy any current user flags..
|
|
sess->destroyEvent(CPAP_UserFlag1);
|
|
sess->destroyEvent(CPAP_UserFlag2);
|
|
sess->destroyEvent(CPAP_UserFlag3);
|
|
|
|
// AHI flags
|
|
sess->destroyEvent(CPAP_AHI);
|
|
sess->destroyEvent(CPAP_RDI);
|
|
|
|
if (sess->machine()->loaderName() != STR_MACH_PRS1) {
|
|
sess->destroyEvent(CPAP_LargeLeak);
|
|
} else {
|
|
sess->destroyEvent(CPAP_Leak);
|
|
}
|
|
|
|
sess->SetChanged(true);
|
|
|
|
sess->UpdateSummaries();
|
|
sess->machine()->SaveSession(sess);
|
|
|
|
if (!isopen) {
|
|
sess->TrashEvents();
|
|
}
|
|
}
|
|
}
|
|
progress.close();
|
|
|
|
welcome = new Welcome(ui->tabWidget);
|
|
ui->tabWidget->insertTab(1, welcome, tr("Welcome"));
|
|
|
|
daily = new Daily(ui->tabWidget, nullptr);
|
|
ui->tabWidget->insertTab(2, daily, STR_TR_Daily);
|
|
daily->ReloadGraphs();
|
|
|
|
overview = new Overview(ui->tabWidget, daily->graphView());
|
|
ui->tabWidget->insertTab(3, overview, STR_TR_Overview);
|
|
overview->ReloadGraphs();
|
|
|
|
// Should really create welcome and statistics here, but they need redoing later anyway to kill off webkit
|
|
ui->tabWidget->setCurrentIndex(AppSetting->openTabAtStart());
|
|
GenerateStatistics();
|
|
PopulatePurgeMenu();
|
|
|
|
}
|
|
|
|
void MainWindow::on_actionImport_ZEO_Data_triggered()
|
|
{
|
|
QFileDialog w;
|
|
w.setFileMode(QFileDialog::ExistingFiles);
|
|
w.setWindowFlags(this->windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
w.setOption(QFileDialog::ShowDirsOnly, false);
|
|
w.setOption(QFileDialog::DontUseNativeDialog, true);
|
|
w.setNameFilters(QStringList("Zeo CSV File (*.csv)"));
|
|
|
|
ZEOLoader zeo;
|
|
|
|
if (w.exec() == QFileDialog::Accepted) {
|
|
QString filename = w.selectedFiles()[0];
|
|
|
|
if (!zeo.OpenFile(filename)) {
|
|
Notify(tr("There was a problem opening ZEO File: ") + filename);
|
|
return;
|
|
}
|
|
|
|
Notify(tr("Zeo CSV Import complete"));
|
|
daily->LoadDate(daily->getDate());
|
|
}
|
|
|
|
|
|
}
|
|
|
|
void MainWindow::on_actionImport_RemStar_MSeries_Data_triggered()
|
|
{
|
|
#ifdef REMSTAR_M_SUPPORT
|
|
QFileDialog w;
|
|
w.setFileMode(QFileDialog::ExistingFiles);
|
|
w.setOption(QFileDialog::ShowDirsOnly, false);
|
|
w.setOption(QFileDialog::DontUseNativeDialog, true);
|
|
w.setNameFilters(QStringList("M-Series data file (*.bin)"));
|
|
|
|
MSeriesLoader mseries;
|
|
|
|
if (w.exec() == QFileDialog::Accepted) {
|
|
QString filename = w.selectedFiles()[0];
|
|
|
|
if (!mseries.Open(filename, p_profile)) {
|
|
Notify(tr("There was a problem opening MSeries block File: ") + filename);
|
|
return;
|
|
}
|
|
|
|
Notify(tr("MSeries Import complete"));
|
|
daily->LoadDate(daily->getDate());
|
|
}
|
|
|
|
#endif
|
|
}
|
|
|
|
void MainWindow::on_actionSleep_Disorder_Terms_Glossary_triggered()
|
|
{
|
|
// QDesktopServices::openUrl(QUrl("http://sleepyhead.sourceforge.net/wiki/index.php?title=Glossary"));
|
|
// QMessageBox::information(nullptr, STR_MessageBox_Information, tr("The Glossary is not yet implemented"));
|
|
if (QMessageBox::question(nullptr, STR_MessageBox_Question, tr("The Glossary will open in your default browser"),
|
|
QMessageBox::Ok|QMessageBox::Cancel, QMessageBox::Ok) == QMessageBox::Ok )
|
|
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()
|
|
{
|
|
// if (QMessageBox::question(this,STR_MessageBox_Warning,tr("Changing the language will reset custom Event and Waveform names/labels/descriptions.")+"\n\n"+tr("Are you sure you want to do this?"), QMessageBox::Yes, QMessageBox::No) == QMessageBox::No) {
|
|
// return;
|
|
// }
|
|
|
|
RestartApplication(true, "--language");
|
|
}
|
|
|
|
void MainWindow::on_actionChange_Data_Folder_triggered()
|
|
{
|
|
if (p_profile) {
|
|
p_profile->Save();
|
|
p_profile->removeLock();
|
|
}
|
|
p_pref->Save();
|
|
|
|
RestartApplication(false, "-d");
|
|
}
|
|
|
|
void MainWindow::on_actionImport_Somnopose_Data_triggered()
|
|
{
|
|
QFileDialog w;
|
|
w.setFileMode(QFileDialog::ExistingFiles);
|
|
w.setWindowFlags(this->windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
w.setOption(QFileDialog::ShowDirsOnly, false);
|
|
w.setOption(QFileDialog::DontUseNativeDialog, true);
|
|
w.setNameFilters(QStringList("Somnopause CSV File (*.csv)"));
|
|
|
|
SomnoposeLoader somno;
|
|
|
|
if (w.exec() == QFileDialog::Accepted) {
|
|
QString filename = w.selectedFiles()[0];
|
|
|
|
if (!somno.OpenFile(filename)) {
|
|
Notify(tr("There was a problem opening Somnopose Data File: ") + filename);
|
|
return;
|
|
}
|
|
|
|
Notify(tr("Somnopause Data Import complete"));
|
|
daily->LoadDate(daily->getDate());
|
|
}
|
|
|
|
}
|
|
|
|
void MainWindow::GenerateStatistics()
|
|
{
|
|
QDate first = p_profile->FirstDay();
|
|
QDate last = p_profile->LastDay();
|
|
ui->statStartDate->setMinimumDate(first);
|
|
ui->statStartDate->setMaximumDate(last);
|
|
|
|
ui->statEndDate->setMinimumDate(first);
|
|
ui->statEndDate->setMaximumDate(last);
|
|
|
|
Statistics stats;
|
|
QString htmlStats = stats.GenerateHTML();
|
|
QString htmlRecords = stats.UpdateRecordsBox();
|
|
|
|
updateFavourites();
|
|
|
|
setStatsHTML(htmlStats);
|
|
setRecBoxHTML(htmlRecords);
|
|
|
|
}
|
|
|
|
|
|
void MainWindow::JumpDaily()
|
|
{
|
|
qDebug() << "Set current Widget to Daily";
|
|
// sleep(3);
|
|
ui->tabWidget->setCurrentWidget(daily);
|
|
}
|
|
void MainWindow::JumpOverview()
|
|
{
|
|
ui->tabWidget->setCurrentWidget(overview);
|
|
}
|
|
void MainWindow::JumpStatistics()
|
|
{
|
|
ui->tabWidget->setCurrentWidget(ui->statisticsTab);
|
|
}
|
|
void MainWindow::JumpImport()
|
|
{
|
|
on_importButton_clicked();
|
|
}
|
|
void MainWindow::JumpOxiWizard()
|
|
{
|
|
on_oximetryButton_clicked();
|
|
}
|
|
|
|
|
|
void MainWindow::on_statisticsButton_clicked()
|
|
{
|
|
if (p_profile) {
|
|
ui->tabWidget->setCurrentWidget(ui->statisticsTab);
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_reportModeMonthly_clicked()
|
|
{
|
|
ui->statStartDate->setVisible(false);
|
|
ui->statEndDate->setVisible(false);
|
|
if (p_profile->general->statReportMode() != 1) {
|
|
p_profile->general->setStatReportMode(1);
|
|
GenerateStatistics();
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_reportModeStandard_clicked()
|
|
{
|
|
ui->statStartDate->setVisible(false);
|
|
ui->statEndDate->setVisible(false);
|
|
if (p_profile->general->statReportMode() != 0) {
|
|
p_profile->general->setStatReportMode(0);
|
|
GenerateStatistics();
|
|
}
|
|
}
|
|
|
|
|
|
void MainWindow::on_reportModeRange_clicked()
|
|
{
|
|
ui->statStartDate->setVisible(true);
|
|
ui->statEndDate->setVisible(true);
|
|
if (p_profile->general->statReportMode() != 2) {
|
|
p_profile->general->setStatReportMode(2);
|
|
GenerateStatistics();
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_actionPurgeCurrentDaysOximetry_triggered()
|
|
{
|
|
if (!daily)
|
|
return;
|
|
QDate date = daily->getDate();
|
|
Day * day = p_profile->GetDay(date, MT_OXIMETER);
|
|
if (day) {
|
|
if (QMessageBox::question(this, STR_MessageBox_Warning,
|
|
tr("Are you sure you want to delete oximetry data for %1").
|
|
arg(daily->getDate().toString(Qt::DefaultLocaleLongDate))+"<br/><br/>"+
|
|
tr("<b>Please be aware you can not undo this operation!</b>"),
|
|
QMessageBox::Yes | QMessageBox::No, QMessageBox::No) == QMessageBox::No) {
|
|
return;
|
|
}
|
|
|
|
QList<Session *> sessionlist=day->getSessions(MT_OXIMETER);
|
|
|
|
for (auto & sess : sessionlist) {
|
|
sess->Destroy();
|
|
delete sess;
|
|
}
|
|
|
|
|
|
if (daily) {
|
|
daily->clearLastDay(); // otherwise Daily will crash
|
|
daily->ReloadGraphs();
|
|
}
|
|
if (overview) overview->ReloadGraphs();
|
|
if (welcome) welcome->refreshPage();
|
|
} else {
|
|
QMessageBox::information(this, STR_MessageBox_Information,
|
|
tr("Select the day with valid oximetry data in daily view first."),QMessageBox::Ok);
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_importButton_clicked()
|
|
{
|
|
on_action_Import_Data_triggered();
|
|
}
|
|
|
|
|
|
void MainWindow::on_actionLine_Cursor_toggled(bool b)
|
|
{
|
|
AppSetting->setLineCursorMode(b);
|
|
if (ui->tabWidget->currentWidget() == daily) {
|
|
daily->graphView()->timedRedraw(0);
|
|
} else if (ui->tabWidget->currentWidget() == overview) {
|
|
overview->graphView()->timedRedraw(0);
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_actionPie_Chart_toggled(bool visible)
|
|
{
|
|
AppSetting->setShowPieChart(visible);
|
|
if (!setupRunning && daily) {
|
|
daily->updateLeftSidebar();
|
|
// daily->ReloadGraphs();
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_actionLeft_Daily_Sidebar_toggled(bool visible)
|
|
{
|
|
if (daily) daily->setSidebarVisible(visible);
|
|
}
|
|
|
|
void MainWindow::on_actionDaily_Calendar_toggled(bool visible)
|
|
{
|
|
if (daily) daily->setCalendarVisible(visible);
|
|
}
|
|
|
|
#include "SleepLib/journal.h"
|
|
|
|
void MainWindow::on_actionExport_Journal_triggered()
|
|
{
|
|
QString folder;
|
|
|
|
folder = QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation);
|
|
|
|
folder += QDir::separator() + tr("%1's Journal").arg(p_profile->user->userName()) + ".xml";
|
|
|
|
QString filename = QFileDialog::getSaveFileName(this, tr("Choose where to save journal"), folder, tr("XML Files (*.xml)"));
|
|
|
|
BackupJournal(filename);
|
|
}
|
|
|
|
void MainWindow::on_actionShow_Performance_Counters_toggled(bool arg1)
|
|
{
|
|
AppSetting->setShowPerformance(arg1);
|
|
}
|
|
|
|
void MainWindow::on_actionExport_CSV_triggered()
|
|
{
|
|
ExportCSV ex(this);
|
|
|
|
if (ex.exec() == ExportCSV::Accepted) {
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_actionExport_Review_triggered()
|
|
{
|
|
QMessageBox::information(nullptr, STR_MessageBox_Information, tr("Export review is not yet implemented"));
|
|
}
|
|
|
|
void MainWindow::on_mainsplitter_splitterMoved(int, int)
|
|
{
|
|
AppSetting->setRightPanelWidth(ui->mainsplitter->sizes()[1]);
|
|
}
|
|
|
|
#include "translation.h"
|
|
void MainWindow::on_actionReport_a_Bug_triggered()
|
|
{
|
|
// QSettings settings;
|
|
// QString language = settings.value(LangSetting).toString();
|
|
//
|
|
// QDesktopServices::openUrl(QUrl(QString("https://sleepyhead.jedimark.net/report_bugs.php?lang=%1&version=%2&platform=%3").arg(language).arg(getVersion()).arg(PlatformString)));
|
|
QMessageBox::information(nullptr, STR_MessageBox_Error, tr("Reporting issues is not yet implemented"));
|
|
}
|
|
|
|
void MainWindow::on_actionSystem_Information_triggered()
|
|
{
|
|
QString text = ""; // tr("OSCAR version:") + "<br/>";
|
|
QStringList info = getBuildInfo();
|
|
for (int i = 0; i < info.size(); ++i)
|
|
text += info.at(i) + "<br/>";
|
|
QMessageBox::information(nullptr, tr("OSCAR Information"), text);
|
|
}
|
|
|
|
void MainWindow::on_profilesButton_clicked()
|
|
{
|
|
ui->tabWidget->setCurrentWidget(profileSelector);
|
|
}
|
|
|
|
void MainWindow::on_bookmarkView_anchorClicked(const QUrl &arg1)
|
|
{
|
|
on_recordsBox_anchorClicked(arg1);
|
|
}
|
|
|
|
void MainWindow::on_recordsBox_anchorClicked(const QUrl &linkurl)
|
|
{
|
|
QString link = linkurl.toString().section("=", 0, 0).toLower();
|
|
QString data = linkurl.toString().section("=", 1).toLower();
|
|
qDebug() << linkurl.toString() << link << data;
|
|
|
|
if (link == "daily") {
|
|
QDate date = QDate::fromString(data, Qt::ISODate);
|
|
ui->tabWidget->setCurrentWidget(daily);
|
|
QApplication::processEvents();
|
|
daily->LoadDate(date);
|
|
} else if (link == "overview") {
|
|
QString date1 = data.section(",", 0, 0);
|
|
QString date2 = data.section(",", 1);
|
|
|
|
QDate d1 = QDate::fromString(date1, Qt::ISODate);
|
|
QDate d2 = QDate::fromString(date2, Qt::ISODate);
|
|
overview->setRange(d1, d2);
|
|
ui->tabWidget->setCurrentWidget(overview);
|
|
} else if (link == "import") {
|
|
// Don't run the importer directly from here because it destroys the object that called this function..
|
|
// Schedule it instead
|
|
if (data == "cpap") QTimer::singleShot(0, mainwin, SLOT(on_importButton_clicked()));
|
|
if (data == "oximeter") QTimer::singleShot(0, mainwin, SLOT(on_oximetryButton_clicked()));
|
|
} else if (link == "statistics") {
|
|
ui->tabWidget->setCurrentWidget(ui->statisticsTab);
|
|
}
|
|
}
|
|
|
|
void MainWindow::on_statisticsView_anchorClicked(const QUrl &url)
|
|
{
|
|
on_recordsBox_anchorClicked(url);
|
|
}
|