2019-03-10 16:03:19 +00:00
/* OSCAR Main
2014-04-09 21:01:57 +00:00
*
2021-10-24 03:19:53 +00:00
* Copyright ( c ) 2019 - 2022 The OSCAR Team
2018-03-28 07:10:52 +00:00
* Copyright ( c ) 2011 - 2018 Mark Watkins < mark @ jedimark . net >
2014-04-09 21:01:57 +00:00
*
* This file is subject to the terms and conditions of the GNU General Public
2018-06-04 20:48:38 +00:00
* License . See the file COPYING in the main directory of the source code
* for more details . */
2011-06-26 08:30:44 +00:00
2023-06-22 12:05:21 +00:00
# define TEST_MACROS_ENABLEDoff
# include <test_macros.h>
2019-05-03 01:51:56 +00:00
# ifdef UNITTEST_MODE
# include "tests/AutoTest.h"
# endif
2013-01-20 19:39:01 +00:00
# include <QApplication>
2019-05-31 22:50:16 +00:00
# include <QGuiApplication>
2011-10-01 12:54:20 +00:00
# include <QMessageBox>
2011-07-01 10:10:44 +00:00
# include <QDebug>
2013-09-09 15:56:02 +00:00
# include <QTranslator>
2013-10-19 02:59:52 +00:00
# include <QSettings>
# include <QFileDialog>
2018-05-07 12:13:07 +00:00
# include <QFontDatabase>
2019-03-25 23:44:49 +00:00
# include <QStandardPaths>
2019-04-02 14:04:44 +00:00
# include <QProgressDialog>
2011-10-30 14:01:33 +00:00
2014-07-09 03:49:20 +00:00
# include "version.h"
2014-06-20 07:05:40 +00:00
# include "logger.h"
2011-06-26 08:30:44 +00:00
# include "mainwindow.h"
# include "SleepLib/profiles.h"
2014-04-24 09:44:15 +00:00
# include "translation.h"
2019-07-05 01:14:41 +00:00
# include "SleepLib/common.h"
2020-07-19 01:44:05 +00:00
# include "SleepLib/deviceconnection.h"
2023-09-13 01:14:54 +00:00
# include "highresolution.h"
2014-06-20 07:05:40 +00:00
2019-04-02 14:04:44 +00:00
# include <ctime>
# include <chrono>
2011-11-20 23:39:55 +00:00
// Gah! I must add the real darn plugin system one day.
2011-10-01 12:54:20 +00:00
# include "SleepLib/loader_plugins/prs1_loader.h"
# include "SleepLib/loader_plugins/cms50_loader.h"
2014-08-17 23:36:57 +00:00
# include "SleepLib/loader_plugins/cms50f37_loader.h"
2014-05-28 09:35:21 +00:00
# include "SleepLib/loader_plugins/md300w1_loader.h"
2011-10-01 12:54:20 +00:00
# include "SleepLib/loader_plugins/zeo_loader.h"
2014-04-15 13:59:24 +00:00
# include "SleepLib/loader_plugins/somnopose_loader.h"
2011-10-01 12:54:20 +00:00
# include "SleepLib/loader_plugins/resmed_loader.h"
2011-11-20 23:39:55 +00:00
# include "SleepLib/loader_plugins/intellipap_loader.h"
2012-01-22 14:39:20 +00:00
# include "SleepLib/loader_plugins/icon_loader.h"
2021-07-08 17:43:02 +00:00
# include "SleepLib/loader_plugins/sleepstyle_loader.h"
2014-08-03 13:00:13 +00:00
# include "SleepLib/loader_plugins/weinmann_loader.h"
2020-01-23 17:51:58 +00:00
# include "SleepLib/loader_plugins/viatom_loader.h"
2021-12-07 19:57:26 +00:00
# include "SleepLib/loader_plugins/prisma_loader.h"
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
# include "SleepLib/loader_plugins/resvent_loader.h"
2011-10-01 12:54:20 +00:00
2014-04-23 13:19:56 +00:00
MainWindow * mainwin = nullptr ;
2011-06-26 08:30:44 +00:00
2019-04-02 14:04:44 +00:00
int numFilesCopied = 0 ;
2019-03-25 23:44:49 +00:00
2019-04-02 14:04:44 +00:00
// Count the number of files in this directory and all subdirectories
int countRecursively ( QString sourceFolder ) {
QDir sourceDir ( sourceFolder ) ;
2019-03-25 23:44:49 +00:00
2019-04-02 14:04:44 +00:00
if ( ! sourceDir . exists ( ) )
return 0 ;
int numFiles = sourceDir . count ( ) ;
QStringList dirs = sourceDir . entryList ( QDir : : AllDirs | QDir : : NoDotAndDotDot ) ;
for ( int i = 0 ; i < dirs . count ( ) ; i + + ) {
QString srcName = sourceFolder + QDir : : separator ( ) + dirs [ i ] ;
numFiles + = countRecursively ( srcName ) ;
}
return numFiles ;
}
2019-03-25 23:44:49 +00:00
2019-04-02 14:04:44 +00:00
bool copyRecursively ( QString sourceFolder , QString destFolder , QProgressDialog & progress ) {
bool success = false ;
QDir sourceDir ( sourceFolder ) ;
if ( ! sourceDir . exists ( ) )
return false ;
QDir destDir ( destFolder ) ;
if ( ! destDir . exists ( ) )
destDir . mkdir ( destFolder ) ;
QStringList files = sourceDir . entryList ( QDir : : Files ) ;
for ( int i = 0 ; i < files . count ( ) ; i + + ) {
QString srcName = sourceFolder + QDir : : separator ( ) + files [ i ] ;
QString destName = destFolder + QDir : : separator ( ) + files [ i ] ;
success = QFile : : copy ( srcName , destName ) ;
numFilesCopied + + ;
if ( ( numFilesCopied % 20 ) = = 1 ) { // Update progress bar every 20 files
progress . setValue ( numFilesCopied ) ;
QCoreApplication : : processEvents ( ) ;
2019-03-25 23:44:49 +00:00
}
2020-08-13 00:25:15 +00:00
if ( ! success ) {
qWarning ( ) < < " copyRecursively: Unable to copy " < < srcName < < " to " < < destName ;
2019-04-02 14:04:44 +00:00
return false ;
2020-08-13 00:25:15 +00:00
}
2019-04-02 14:04:44 +00:00
}
2019-03-25 23:44:49 +00:00
2019-04-02 14:04:44 +00:00
files . clear ( ) ;
files = sourceDir . entryList ( QDir : : AllDirs | QDir : : NoDotAndDotDot ) ;
for ( int i = 0 ; i < files . count ( ) ; i + + ) {
QString srcName = sourceFolder + QDir : : separator ( ) + files [ i ] ;
QString destName = destFolder + QDir : : separator ( ) + files [ i ] ;
// qDebug() << "Copy from "+srcName+" to "+destName;
success = copyRecursively ( srcName , destName , progress ) ;
if ( ! success )
return false ;
2019-03-25 23:44:49 +00:00
}
2019-04-02 14:04:44 +00:00
return true ;
}
2019-03-25 23:44:49 +00:00
bool processPreferenceFile ( QString path ) {
bool success = true ;
QString fullpath = path + " /Preferences.xml " ;
qDebug ( ) < < " Process " + fullpath ;
QFile fl ( fullpath ) ;
QFile tmp ( fullpath + " .tmp " ) ;
QString line ;
fl . open ( QIODevice : : ReadOnly ) ;
tmp . open ( QIODevice : : WriteOnly ) ;
QTextStream instr ( & fl ) ;
QTextStream outstr ( & tmp ) ;
2020-09-10 01:10:36 +00:00
bool isSleepyHead = false ;
2019-03-25 23:44:49 +00:00
while ( instr . readLineInto ( & line ) ) {
2020-09-10 01:10:36 +00:00
if ( line . contains ( " <SleepyHead> " ) ) // Is this SleepyHead or OSCAR preferences file?
isSleepyHead = true ;
2019-03-25 23:44:49 +00:00
line . replace ( " SleepyHead " , " OSCAR " ) ;
2020-09-10 01:10:36 +00:00
if ( isSleepyHead & & line . contains ( " VersionString " ) ) {
2019-03-25 23:44:49 +00:00
int rtAngle = line . indexOf ( " > " , 0 ) ;
int lfAngle = line . indexOf ( " < " , rtAngle ) ;
line . replace ( rtAngle + 1 , lfAngle - rtAngle - 1 , " 1.0.0-beta " ) ;
}
outstr < < line ;
}
fl . remove ( ) ;
success = tmp . rename ( fullpath ) ;
return success ;
}
bool processFile ( QString fullpath ) {
bool success = true ;
qDebug ( ) < < " Process " + fullpath ;
QFile fl ( fullpath ) ;
QFile tmp ( fullpath + " .tmp " ) ;
QString line ;
fl . open ( QIODevice : : ReadOnly ) ;
tmp . open ( QIODevice : : WriteOnly ) ;
QTextStream instr ( & fl ) ;
QTextStream outstr ( & tmp ) ;
while ( instr . readLineInto ( & line ) ) {
2019-04-24 19:12:07 +00:00
if ( line . contains ( " EnableMultithreading " ) ) {
if ( line . contains ( " true " ) ) {
line . replace ( " true " , " false " ) ;
}
}
2019-03-25 23:44:49 +00:00
line . replace ( " SleepyHead " , " OSCAR " ) ;
outstr < < line ;
}
fl . remove ( ) ;
success = tmp . rename ( fullpath ) ;
return success ;
}
bool process_a_Profile ( QString path ) {
bool success = true ;
qDebug ( ) < < " Entering profile directory " + path ;
QDir dir ( path ) ;
QStringList files = dir . entryList ( QStringList ( " *.xml " ) , QDir : : Files ) ;
for ( int i = 0 ; success & & ( i < files . count ( ) ) ; i + + ) {
success = processFile ( path + " / " + files [ i ] ) ;
}
return success ;
}
bool migrateFromSH ( QString destDir ) {
QString homeDocs = QStandardPaths : : writableLocation ( QStandardPaths : : DocumentsLocation ) + " / " ;
2019-03-29 01:24:10 +00:00
QString datadir ;
bool selectingFolder = true ;
2019-03-25 23:44:49 +00:00
bool success = false ;
2019-04-02 14:04:44 +00:00
// long int startTime, countDone, allDone;
2019-03-25 23:44:49 +00:00
if ( destDir . isEmpty ( ) ) {
qDebug ( ) < < " Migration path is empty string " ;
return success ;
}
2019-03-29 01:24:10 +00:00
while ( selectingFolder ) {
datadir = QFileDialog : : getExistingDirectory ( nullptr ,
2020-09-10 01:10:36 +00:00
QObject : : tr ( " Choose the SleepyHead or OSCAR data folder to migrate " ) + " " +
2019-04-02 14:04:44 +00:00
QObject : : tr ( " or CANCEL to skip migration. " ) ,
2019-03-29 01:24:10 +00:00
homeDocs , QFileDialog : : ShowDirsOnly ) ;
qDebug ( ) < < " Migration folder selected: " + datadir ;
if ( datadir . isEmpty ( ) ) {
qDebug ( ) < < " No migration source directory selected " ;
return false ;
} else { // We have a folder, see if is a SleepyHead folder
QDir dir ( datadir ) ;
QFile file ( datadir + " /Preferences.xml " ) ;
QDir dirP ( datadir + " /Profiles " ) ;
if ( ! file . exists ( ) | | ! dirP . exists ( ) ) { // It doesn't have a Preferences.xml file or a Profiles directory in it
// Not a new directory.. nag the user.
2020-04-30 15:35:59 +00:00
QMessageBox : : warning ( nullptr , STR_MessageBox_Error ,
2020-09-10 01:10:36 +00:00
QObject : : tr ( " The folder you chose does not contain valid SleepyHead or OSCAR data. " ) +
2020-04-30 15:35:59 +00:00
" \n \n " + QObject : : tr ( " You cannot use this folder: " ) + " " + datadir ,
QMessageBox : : Ok ) ;
continue ; // Nope, don't use it, go around the loop again
2019-03-29 01:24:10 +00:00
}
qDebug ( ) < < " Migration folder is " < < datadir ;
selectingFolder = false ;
}
2019-03-25 23:44:49 +00:00
}
2019-04-02 14:04:44 +00:00
auto startTime = std : : chrono : : steady_clock : : now ( ) ;
int numFiles = countRecursively ( datadir ) ; // count number of files to be copied
auto countDone = std : : chrono : : steady_clock : : now ( ) ;
qDebug ( ) < < " Number of files to migrate: " < < numFiles ;
QProgressDialog progress ( QObject : : tr ( " Migrating " ) + QString : : number ( numFiles ) + QObject : : tr ( " files " ) + " \n " +
2023-09-12 23:30:59 +00:00
QObject : : tr ( " from " ) + QDir ( datadir ) . dirName ( ) + " \n " + QObject : : tr ( " to " ) +
QDir ( destDir ) . dirName ( ) , QString ( ) , 0 , numFiles , 0 , Qt : : WindowSystemMenuHint | Qt : : WindowTitleHint ) ;
2019-04-02 14:04:44 +00:00
progress . setValue ( 0 ) ;
progress . setMinimumWidth ( 300 ) ;
progress . show ( ) ;
success = copyRecursively ( datadir , destDir , progress ) ;
2019-03-25 23:44:49 +00:00
if ( success ) {
qDebug ( ) < < " Finished copying " + datadir ;
}
success = processPreferenceFile ( destDir ) ;
2018-06-12 12:55:44 +00:00
2019-03-25 23:44:49 +00:00
QDir profDir ( destDir + " /Profiles " ) ;
QStringList names = profDir . entryList ( QDir : : AllDirs | QDir : : NoDotAndDotDot ) ;
for ( int i = 0 ; success & & ( i < names . count ( ) ) ; i + + ) {
success = process_a_Profile ( destDir + " /Profiles/ " + names [ i ] ) ;
}
2019-04-02 14:04:44 +00:00
progress . setValue ( numFiles ) ;
auto allDone = std : : chrono : : steady_clock : : now ( ) ;
auto elapsedCount = std : : chrono : : duration_cast < std : : chrono : : microseconds > ( countDone - startTime ) ;
auto elapsedCopy = std : : chrono : : duration_cast < std : : chrono : : microseconds > ( allDone - countDone ) ;
qDebug ( ) < < " Counting files took " < < elapsedCount . count ( ) < < " microsecs " ;
qDebug ( ) < < " Migrating files took " < < elapsedCopy . count ( ) < < " microsecs " ;
2019-03-25 23:44:49 +00:00
return success ;
}
2018-06-12 12:55:44 +00:00
2019-05-03 01:51:56 +00:00
# ifdef UNITTEST_MODE
int main ( int argc , char * argv [ ] )
{
2020-01-15 21:34:28 +00:00
initializeStrings ( ) ;
Update version display throughout to use the new information and be consistent.
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.
2020-01-16 18:05:55 +00:00
qDebug ( ) < < STR_TR_OSCAR + " " + getVersion ( ) ;
2020-01-15 21:34:28 +00:00
2019-05-03 01:51:56 +00:00
AutoTest : : run ( argc , argv ) ;
}
# else
2022-01-14 16:45:54 +00:00
# ifndef Q_OS_LINUX
// Due to a bug in Qt, creating multiple QApplication instances in a process
// causes subsequent native file dialog boxes to hang on Fedora 35.
// See https://bugreports.qt.io/browse/QTBUG-90616
//
// Since Linux users can simply use the --legacy command-line argument,
// we can remove the shift-key check that requires those multiple instances.
2020-02-17 22:26:00 +00:00
bool shiftKeyPressedAtLaunch ( int argc , char * argv [ ] )
{
// Reliably detecting the shift key requires a QGuiApplication instance, but
// we need to create the real QApplication afterwards, so create a temporary
// instance here.
QGuiApplication * app = new QGuiApplication ( argc , argv ) ;
Qt : : KeyboardModifiers keymodifier = QGuiApplication : : queryKeyboardModifiers ( ) ;
delete app ;
return keymodifier = = Qt : : ShiftModifier ;
}
2022-01-14 16:45:54 +00:00
# endif
2020-02-17 22:26:00 +00:00
2013-01-17 18:26:11 +00:00
2023-09-12 23:30:59 +00:00
void optionExit ( int exitCode , QString error ) {
if ( exitCode ) {
error . prepend ( " Command option error: " ) ;
}
qCritical ( ) < < error < < ( R " (
Help Menu
Option Description
- p Pauses execution for 1 second
- - profile < name > Name of profile . if name does not exist then uses last used profile .
- - l or - - language Force language prompt
- - datadir < folderName > Use folderName as Oscar Data folder . For relatve paths : < Documents folder > / < relative path > .
If folder does not exist then prompts user .
- - help Displays this menu and exits .
2023-09-13 01:14:54 +00:00
- - hires Enables high Resolution
- - hiresoff Disables high Resolution
2023-09-25 00:32:20 +00:00
- l or - - nop internal call to main from OSCAR .
2023-09-12 23:30:59 +00:00
) " );
exit ( exitCode ) ;
}
int main ( int argc , char * argv [ ] ) {
2019-03-25 23:44:49 +00:00
QString homeDocs = QStandardPaths : : writableLocation ( QStandardPaths : : DocumentsLocation ) + " / " ;
2018-06-12 12:55:44 +00:00
QCoreApplication : : setApplicationName ( getAppName ( ) ) ;
QCoreApplication : : setOrganizationName ( getDeveloperName ( ) ) ;
2019-04-08 14:26:59 +00:00
QCoreApplication : : setOrganizationDomain ( getDeveloperDomain ( ) ) ;
2023-09-13 01:14:54 +00:00
HighResolution : : init ( ) ;
bool hiResEnabled = false ;
for ( int i = 1 ; i < argc ; i + + ) {
if ( 0 = = strcmp ( argv [ i ] , " --hires " ) ) {
HighResolution : : init ( HighResolution : : HRM_ENABLED ) ;
} else if ( 0 = = strcmp ( argv [ i ] , " --hiresoff " ) ) {
HighResolution : : init ( HighResolution : : HRM_DISABLED ) ;
} else if ( 0 = = strcmp ( argv [ i ] , " --datadir " ) ) { i + + ;
} else if ( 0 = = strcmp ( argv [ i ] , " -profile " ) ) { i + + ;
}
}
if ( HighResolution : : isEnabled ( ) ) {
hiResEnabled = true ;
QGuiApplication : : setAttribute ( Qt : : AA_EnableHighDpiScaling ) ;
}
2018-06-12 12:55:44 +00:00
QSettings settings ;
2019-05-31 22:50:16 +00:00
// If shift key was held down when OSCAR was launched, force Software graphics Engine (aka LegacyGFX)
QString forcedEngine = " " ;
2022-01-14 16:45:54 +00:00
# ifndef Q_OS_LINUX
// Shift key check is skipped on Linux due to a Qt bug, see comment at shiftKeyPressedAtLaunch().
2020-02-17 22:26:00 +00:00
if ( shiftKeyPressedAtLaunch ( argc , argv ) ) {
2019-05-31 22:50:16 +00:00
settings . setValue ( GFXEngineSetting , ( unsigned int ) GFX_Software ) ;
forcedEngine = " Software Engine forced by shift key at launch " ;
2018-06-12 12:55:44 +00:00
}
2022-01-14 16:45:54 +00:00
# endif
2020-02-17 15:59:26 +00:00
// This argument needs to be processed before creating the QApplication,
// based on sample code at https://doc.qt.io/qt-5/qapplication.html#details
for ( int i = 1 ; i < argc ; + + i ) {
if ( ! qstrcmp ( argv [ i ] , " --legacy " ) ) {
settings . setValue ( GFXEngineSetting , ( unsigned int ) GFX_Software ) ;
forcedEngine = " Software Engine forced by --legacy command line switch " ;
}
}
2020-04-25 20:30:35 +00:00
# ifdef Q_OS_WIN
bool oscarCrashed = false ;
if ( settings . value ( " OpenGLCompatibilityCheck " ) . toBool ( ) ) {
oscarCrashed = true ;
}
if ( oscarCrashed ) {
settings . setValue ( GFXEngineSetting , ( unsigned int ) GFX_Software ) ;
forcedEngine = " Software Engine forced by previous crash " ;
settings . remove ( " OpenGLCompatibilityCheck " ) ;
}
# endif
2020-02-17 15:59:26 +00:00
GFXEngine gfxEngine = ( GFXEngine ) qMin ( ( unsigned int ) settings . value ( GFXEngineSetting , ( unsigned int ) GFX_OpenGL ) . toUInt ( ) , ( unsigned int ) MaxGFXEngine ) ;
switch ( gfxEngine ) {
case 0 : // GFX_OpenGL
QCoreApplication : : setAttribute ( Qt : : AA_UseDesktopOpenGL ) ;
break ;
case 1 : // GFX_ANGLE
QCoreApplication : : setAttribute ( Qt : : AA_UseOpenGLES ) ;
break ;
case 2 : // GFX_Software
default :
QCoreApplication : : setAttribute ( Qt : : AA_UseSoftwareOpenGL ) ;
}
2023-09-12 23:30:59 +00:00
QApplication mainapp ( argc , argv ) ;
QStringList args = mainapp . arguments ( ) ;
2020-02-17 15:59:26 +00:00
2020-04-25 20:30:35 +00:00
# ifdef Q_OS_WIN
// QMessageBox must come after the application is created. The graphics engine has to be selected before.
if ( oscarCrashed ) {
QMessageBox : : warning ( nullptr , STR_MessageBox_Error ,
QObject : : tr ( " OSCAR crashed due to an incompatibility with your graphics hardware. " ) + " \n \n " +
QObject : : tr ( " To resolve this, OSCAR has reverted to a slower but more compatible method of drawing. " ) ,
QMessageBox : : Ok ) ;
}
# endif
2018-06-12 12:55:44 +00:00
2021-11-06 23:58:42 +00:00
initializeLogger ( ) ;
// After initializing the logger, any qDebug() messages will be queued but not written to console
// until MainWindow is constructed below. In spite of that, we initialize the logger here so that
// the intervening messages show up in the debug pane.
//
// The only time this is really noticeable is when initTranslations() presents its language
// selection QDialog, which waits indefinitely for user input before MainWindow is constructed.
2014-04-17 05:52:25 +00:00
bool force_data_dir = false ;
2017-09-02 12:01:01 +00:00
QString load_profile = " " ;
2014-04-17 05:52:25 +00:00
for ( int i = 1 ; i < args . size ( ) ; i + + ) {
2023-09-12 23:30:59 +00:00
if ( ( args [ i ] = = " --language " ) | | ( args [ i ] = = " --l " ) ) {
2014-10-02 17:46:08 +00:00
settings . setValue ( LangSetting , " " ) ;
2023-09-25 00:32:20 +00:00
} else if ( ( args [ i ] = = " -l " ) | | ( args [ i ] = = " --nop " ) ) {
// do nothing. internal call to change language.
2023-09-12 23:30:59 +00:00
} else if ( args [ i ] = = " -p " ) {
2018-06-12 04:00:38 +00:00
QThread : : msleep ( 1000 ) ;
2023-09-12 23:30:59 +00:00
} else if ( args [ i ] = = " --profile " ) {
2019-03-25 23:44:49 +00:00
if ( ( i + 1 ) < args . size ( ) )
2017-09-02 12:01:01 +00:00
load_profile = args [ + + i ] ;
2019-03-25 23:44:49 +00:00
else {
2023-09-12 23:30:59 +00:00
optionExit ( 1 , " Missing argument to --profile " ) ;
2017-09-02 12:01:01 +00:00
}
2018-05-03 05:08:45 +00:00
} else if ( args [ i ] = = " --datadir " ) { // mltam's idea
2021-11-05 04:01:18 +00:00
QString datadir , datadirwas ;
2018-05-03 05:08:45 +00:00
if ( ( i + 1 ) < args . size ( ) ) {
2021-11-05 04:01:18 +00:00
datadirwas = datadir = args [ + + i ] ;
bool havefullpath = false ;
if ( datadir . length ( ) > = 2 ) {
havefullpath = ( datadir . at ( 1 ) = = QLatin1Char ( ' : ' ) ) // Allow a Windows drive letter
| | ( datadir . at ( 0 ) = = ' / ' ) // or Linux full path
| | ( datadir . at ( 0 ) = = ' \\ ' ) ;
}
2023-09-12 23:30:59 +00:00
if ( ! havefullpath ) {
2019-06-24 17:05:58 +00:00
datadir = homeDocs + datadir ;
2023-09-12 23:30:59 +00:00
qDebug ( ) < < " --datadir was: " < < datadirwas < < " ; --datadir is: " < < datadir ;
}
2019-06-24 17:05:58 +00:00
settings . setValue ( " Settings/AppData " , datadir ) ;
2019-03-25 23:44:49 +00:00
// force_data_dir = true;
2018-05-03 05:08:45 +00:00
} else {
2023-09-12 23:30:59 +00:00
optionExit ( 2 , " Missing argument to --datadir \n " ) ;
2018-05-03 05:08:45 +00:00
}
2023-09-13 01:14:54 +00:00
} else if ( 0 = = strcmp ( argv [ i ] , " --hires " ) ) { // already handle in 1st scan
} else if ( 0 = = strcmp ( argv [ i ] , " --hiresoff " ) ) { // already handle in 1st scan
2023-09-12 23:30:59 +00:00
} else if ( QString ( args [ i ] ) . contains ( " help " , Qt : : CaseInsensitive ) ) {
optionExit ( 0 , QString ( " " ) ) ;
} else {
optionExit ( 3 , QString ( " Invalid Argument. %1 " ) . arg ( args [ i ] ) ) ;
2019-04-02 14:04:44 +00:00
}
2023-09-12 23:30:59 +00:00
2019-03-25 23:44:49 +00:00
} // end of for args loop
2011-11-18 09:05:22 +00:00
2020-01-16 18:58:18 +00:00
qDebug ( ) . noquote ( ) < < " OSCAR starting " < < QDateTime : : currentDateTime ( ) . toString ( ) ;
2020-03-14 17:43:24 +00:00
qDebug ( ) < < " APP-NAME: " < < QCoreApplication : : applicationName ( ) ;
2020-03-15 20:27:36 +00:00
qDebug ( ) < < " APP-PATH: " < < QCoreApplication : : applicationDirPath ( ) ;
2020-03-14 17:43:24 +00:00
qDebug ( ) < < " APP-RESOURCES: " < < appResourcePath ( ) ;
2023-09-13 01:14:54 +00:00
HighResolution : : display ( hiResEnabled ) ;
2020-01-16 18:58:18 +00:00
2018-06-12 04:00:38 +00:00
# ifdef QT_DEBUG
QString relinfo = " debug " ;
# else
QString relinfo = " " ;
2018-06-07 01:53:20 +00:00
# endif
2018-06-12 04:00:38 +00:00
relinfo = " ( " + QSysInfo : : kernelType ( ) + " " + QSysInfo : : currentCpuArchitecture ( ) + relinfo + " ) " ;
2020-01-16 18:58:18 +00:00
relinfo = STR_AppName + " " + getVersion ( ) + " " + relinfo ;
qDebug ( ) . noquote ( ) < < relinfo ;
qDebug ( ) . noquote ( ) < < " Built with Qt " < < QT_VERSION_STR < < " on " < < getBuildDateTime ( ) ;
addBuildInfo ( relinfo ) ; // immediately add it to the build info that's accessible from the UI
2014-06-20 05:25:50 +00:00
2019-08-06 17:51:14 +00:00
SetDateFormat ( ) ;
2013-09-15 04:20:26 +00:00
////////////////////////////////////////////////////////////////////////////////////////////
// Language Selection
////////////////////////////////////////////////////////////////////////////////////////////
2018-06-09 00:59:16 +00:00
initTranslations ( ) ;
2014-10-02 17:46:08 +00:00
2018-06-12 04:00:38 +00:00
initializeStrings ( ) ; // This must be called AFTER translator is installed, but before mainwindow is setup
2018-06-14 07:25:54 +00:00
// QFontDatabase::addApplicationFont("://fonts/FreeSans.ttf");
// a.setFont(QFont("FreeSans", 11, QFont::Normal, false));
2018-06-12 04:00:38 +00:00
mainwin = new MainWindow ;
2019-06-02 02:51:18 +00:00
// Moved buildInfo calls to after translation is available as makeBuildInfo includes tr() calls
2020-01-16 18:58:18 +00:00
QStringList info = makeBuildInfo ( forcedEngine ) ;
2019-06-02 02:51:18 +00:00
for ( int i = 0 ; i < info . size ( ) ; + + i )
qDebug ( ) . noquote ( ) < < info . at ( i ) ;
2014-10-02 17:46:08 +00:00
////////////////////////////////////////////////////////////////////////////////////////////
// OpenGL Detection
////////////////////////////////////////////////////////////////////////////////////////////
2018-06-12 15:57:24 +00:00
getOpenGLVersion ( ) ;
getOpenGLVersionString ( ) ;
2014-06-02 08:16:28 +00:00
2018-06-12 15:57:24 +00:00
//bool opengl2supported = glversion >= 2.0;
//bool bad_graphics = !opengl2supported;
//bool intel_graphics = false;
2018-06-12 12:55:44 +00:00
//#ifndef NO_OPENGL_BUILD
//#endif
2014-06-02 10:28:05 +00:00
2019-04-02 14:04:44 +00:00
/*************************************************************************************
# ifdef BROKEN_OPENGL_BUILD
Q_UNUSED ( bad_graphics )
Q_UNUSED ( intel_graphics )
const QString BetterBuild = " Settings/BetterBuild " ;
if ( opengl2supported ) {
if ( ! settings . value ( BetterBuild , false ) . toBool ( ) ) {
QMessageBox : : information ( nullptr , QObject : : tr ( " A faster build of OSCAR may be available " ) ,
QObject : : tr ( " This build of OSCAR is a compatability version that also works on computers lacking OpenGL 2.0 support. " ) + " <br/><br/> " +
QObject : : tr ( " However it looks like your computer has full support for OpenGL 2.0! " ) + " <br/><br/> " +
QObject : : tr ( " This version will run fine, but a \" <b>%1</b> \" tagged build of OSCAR will likely run a bit faster on your computer. " ) . arg ( " -OpenGL " ) + " <br/><br/> " +
QObject : : tr ( " You will not be bothered with this message again. " ) , QMessageBox : : Ok , QMessageBox : : Ok ) ;
settings . setValue ( BetterBuild , true ) ;
}
2014-06-02 06:40:00 +00:00
}
2019-04-02 14:04:44 +00:00
# else
if ( bad_graphics ) {
QMessageBox : : warning ( nullptr , QObject : : tr ( " Incompatible Graphics Hardware " ) ,
QObject : : tr ( " This build of OSCAR requires OpenGL 2.0 support to function correctly, and unfortunately your computer lacks this capability. " ) + " <br/><br/> " +
QObject : : tr ( " You may need to update your computers graphics drivers from the GPU makers website. %1 " ) .
arg ( intel_graphics ? QObject : : tr ( " (<a href='http://intel.com/support'>Intel's support site</a>) " ) : " " ) + " <br/><br/> " +
QObject : : tr ( " Because graphs will not render correctly, and it may cause crashes, this build will now exit. " ) + " <br/><br/> " +
QObject : : tr ( " There is another build available tagged \" <b>-BrokenGL</b> \" that should work on your computer. " ) ,
QMessageBox : : Ok , QMessageBox : : Ok ) ;
exit ( 1 ) ;
}
# endif
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2013-10-19 02:59:52 +00:00
////////////////////////////////////////////////////////////////////////////////////////////
// Datafolder location Selection
////////////////////////////////////////////////////////////////////////////////////////////
2019-03-25 23:44:49 +00:00
// bool change_data_dir = force_data_dir;
//
// bool havefolder = false;
2014-04-17 05:52:25 +00:00
2019-03-25 23:44:49 +00:00
if ( ! settings . contains ( " Settings/AppData " ) ) { // This is first time execution
if ( settings . contains ( " Settings/AppRoot " ) ) { // allow for old AppRoot here - not really first time
settings . setValue ( " Settings/AppData " , settings . value ( " Settings/AppRoot " ) ) ;
} else {
2019-03-30 17:00:03 +00:00
settings . setValue ( " Settings/AppData " , homeDocs + getModifiedAppData ( ) ) ; // set up new data directory path
2013-10-19 02:59:52 +00:00
}
2019-03-30 17:43:37 +00:00
qDebug ( ) < < " First time: Setting " + GetAppData ( ) ;
2013-10-19 02:59:52 +00:00
}
2019-03-25 23:44:49 +00:00
QDir dir ( GetAppData ( ) ) ;
if ( ! dir . exists ( ) ) { // directory doesn't exist, verify user's choice
if ( ! force_data_dir ) { // unless they explicitly selected it by --datadir param
if ( QMessageBox : : question ( nullptr , STR_MessageBox_Question ,
2019-04-02 14:04:44 +00:00
QObject : : tr ( " OSCAR will set up a folder for your data. " ) + " \n " +
2020-09-10 01:10:36 +00:00
QObject : : tr ( " If you have been using SleepyHead or an older version of OSCAR, " ) + " \n " +
QObject : : tr ( " OSCAR can copy your old data to this folder later. " ) + " \n " +
2019-04-02 14:04:44 +00:00
QObject : : tr ( " We suggest you use this folder: " ) + QDir : : toNativeSeparators ( GetAppData ( ) ) + " \n " +
QObject : : tr ( " Click Ok to accept this, or No if you want to use a different folder. " ) ,
QMessageBox : : Ok | QMessageBox : : No , QMessageBox : : Ok ) = = QMessageBox : : No ) {
// User wants a different folder for data
2019-03-25 23:44:49 +00:00
bool change_data_dir = true ;
while ( change_data_dir ) { // Create or select an acceptable folder
QString datadir = QFileDialog : : getExistingDirectory ( nullptr ,
QObject : : tr ( " Choose or create a new folder for OSCAR data " ) , homeDocs , QFileDialog : : ShowDirsOnly ) ;
if ( datadir . isEmpty ( ) ) { // User hit Cancel instead of selecting or creating a folder
QMessageBox : : information ( nullptr , QObject : : tr ( " Exiting " ) ,
2019-04-02 14:04:44 +00:00
QObject : : tr ( " As you did not select a data folder, OSCAR will exit. " ) + " \n " +
QObject : : tr ( " Next time you run OSCAR, you will be asked again. " ) ) ;
2019-03-25 23:44:49 +00:00
return 0 ;
} else { // We have a folder, see if is already an OSCAR folder
QDir dir ( datadir ) ;
QFile file ( datadir + " /Preferences.xml " ) ;
2019-03-29 01:24:10 +00:00
QDir dirP ( datadir + " /Profiles " ) ;
2019-03-25 23:44:49 +00:00
2019-03-29 01:24:10 +00:00
if ( ! file . exists ( ) | | ! dirP . exists ( ) ) { // It doesn't have a Preferences.xml file or a Profiles directory in it
2019-03-25 23:44:49 +00:00
if ( dir . count ( ) > 2 ) { // but it has more than dot and dotdot
// Not a new directory.. nag the user.
if ( QMessageBox : : question ( nullptr , STR_MessageBox_Warning ,
2019-04-02 14:04:44 +00:00
QObject : : tr ( " The folder you chose is not empty, nor does it already contain valid OSCAR data. " ) +
" \n \n " + QObject : : tr ( " Are you sure you want to use this folder? " ) + " \n \n " +
datadir , QMessageBox : : Yes , QMessageBox : : No ) = = QMessageBox : : No ) {
2019-03-25 23:44:49 +00:00
continue ; // Nope, don't use it, go around the loop again
}
}
}
2019-04-02 14:04:44 +00:00
2019-03-25 23:44:49 +00:00
settings . setValue ( " Settings/AppData " , datadir ) ;
qDebug ( ) < < " Changing data folder to " < < datadir ;
change_data_dir = false ;
2013-10-19 02:59:52 +00:00
}
2019-03-25 23:44:49 +00:00
} // the while loop
} // user wants a different folder
} // user used --datadir folder to select a folder
} // The folder doesn't exist
2019-03-30 17:43:37 +00:00
else
2019-03-31 13:43:41 +00:00
qDebug ( ) < < " AppData folder already exists, so ... " ;
2020-01-16 18:58:18 +00:00
qDebug ( ) . noquote ( ) < < " Using " + GetAppData ( ) + " as OSCAR data folder " ;
2019-03-25 23:44:49 +00:00
2019-06-07 12:00:23 +00:00
QString path = GetAppData ( ) ;
addBuildInfo ( QObject : : tr ( " Data directory: " ) + " <a href= \" file:/// " + path + " \" > " + path + " </a> " ) ;
2019-06-02 02:51:18 +00:00
2019-03-25 23:44:49 +00:00
QDir newDir ( GetAppData ( ) ) ;
2019-05-15 21:26:56 +00:00
# if QT_VERSION < QT_VERSION_CHECK(5,9,0)
2023-09-12 23:30:59 +00:00
if ( ! newDir . exists ( ) | | newDir . count ( ) = = 0 ) // directory doesn't exist yet or is empty, try to migrate old data
2019-05-15 21:26:56 +00:00
# else
2023-09-12 23:30:59 +00:00
if ( ! newDir . exists ( ) | | newDir . isEmpty ( ) ) // directory doesn't exist yet or is empty, try to migrate old data
2019-05-15 21:26:56 +00:00
# endif
2023-09-12 23:30:59 +00:00
{
2020-09-10 01:10:36 +00:00
if ( QMessageBox : : question ( nullptr , QObject : : tr ( " Migrate SleepyHead or OSCAR Data? " ) ,
QObject : : tr ( " On the next screen OSCAR will ask you to select a folder with SleepyHead or OSCAR data " ) + " \n " +
QObject : : tr ( " Click [OK] to go to the next screen or [No] if you do not wish to use any SleepyHead or OSCAR data. " ) ,
2019-04-02 14:04:44 +00:00
QMessageBox : : Ok | QMessageBox : : No , QMessageBox : : Ok ) = = QMessageBox : : Ok ) {
2019-03-31 13:43:41 +00:00
migrateFromSH ( GetAppData ( ) ) ; // doesn't matter if no migration
}
2013-10-19 02:59:52 +00:00
}
2020-07-19 20:36:22 +00:00
// Make sure the data directory exists.
if ( ! newDir . mkpath ( " . " ) ) {
QMessageBox : : warning ( nullptr , QObject : : tr ( " Exiting " ) ,
QObject : : tr ( " Unable to create the OSCAR data folder at " ) + " \n " +
GetAppData ( ) ) ;
return 0 ;
}
2020-08-16 19:46:36 +00:00
// Make sure we can write to the data directory
QFile testFile ( GetAppData ( ) + " /testfile.txt " ) ;
if ( testFile . exists ( ) )
testFile . remove ( ) ;
if ( ! testFile . open ( QFile : : ReadWrite ) ) {
2021-01-19 04:47:59 +00:00
QString errMsg = QObject : : tr ( " Unable to write to OSCAR data directory " ) + " " + GetAppData ( ) + " \n \n " +
2020-08-16 19:46:36 +00:00
QObject : : tr ( " Error code " ) + " : " + QString : : number ( testFile . error ( ) ) + " - " + testFile . errorString ( ) + " \n \n " +
QObject : : tr ( " OSCAR cannot continue and is exiting. " ) + " \n " ;
qCritical ( ) < < errMsg ;
QMessageBox : : critical ( nullptr , QObject : : tr ( " Exiting " ) , errMsg ) ;
return 0 ;
}
else
testFile . remove ( ) ;
2020-07-19 20:36:22 +00:00
// Begin logging to file now that there's a data folder.
2020-08-16 19:46:36 +00:00
if ( ! logger - > logToFile ( ) ) {
QMessageBox : : warning ( nullptr , STR_MessageBox_Warning ,
QObject : : tr ( " Unable to write to debug log. You can still use the debug pane (Help/Troubleshooting/Show Debug Pane) but the debug log will not be written to disk. " ) ) ;
}
2019-03-25 23:44:49 +00:00
2023-09-12 23:30:59 +00:00
2018-04-22 14:22:18 +00:00
///////////////////////////////////////////////////////////////////////////////////////////
2019-07-15 03:25:49 +00:00
// Initialize preferences system (Don't use p_pref before this point!)
2018-04-22 14:22:18 +00:00
///////////////////////////////////////////////////////////////////////////////////////////
p_pref = new Preferences ( " Preferences " ) ;
2019-03-25 23:44:49 +00:00
p_pref - > Open ( ) ;
2018-04-22 14:22:18 +00:00
AppSetting = new AppWideSetting ( p_pref ) ;
2018-05-07 12:13:07 +00:00
QString language = settings . value ( LangSetting , " " ) . toString ( ) ;
AppSetting - > setLanguage ( language ) ;
2019-07-05 01:14:41 +00:00
// Set fonts from preferences file
2021-01-16 01:20:36 +00:00
qDebug ( ) < < " App font before Prefs setting " < < QApplication : : font ( ) ;
2019-07-05 01:14:41 +00:00
validateAllFonts ( ) ;
setApplicationFont ( ) ;
2019-07-15 03:25:49 +00:00
// one-time translate GraphSnapshots to ShowPieChart
p_pref - > Rename ( STR_AS_GraphSnapshots , STR_AS_ShowPieChart ) ;
2018-04-22 14:22:18 +00:00
2019-03-25 23:44:49 +00:00
p_pref - > Erase ( STR_AppName ) ;
p_pref - > Erase ( STR_GEN_SkipLogin ) ;
2013-10-19 02:59:52 +00:00
2020-07-05 01:17:25 +00:00
# ifndef NO_CHECKUPDATES
2012-01-06 16:07:54 +00:00
////////////////////////////////////////////////////////////////////////////////////////////
// Check when last checked for updates..
////////////////////////////////////////////////////////////////////////////////////////////
2018-05-07 12:13:07 +00:00
QDateTime lastchecked , today = QDateTime : : currentDateTime ( ) ;
2016-03-06 02:50:22 +00:00
bool check_updates = false ;
2014-04-17 05:52:25 +00:00
2020-07-05 01:17:25 +00:00
if ( ! getVersion ( ) . IsReleaseVersion ( ) ) {
// If test build, force update autocheck, no more than 7 day interval, show test versions
AppSetting - > setUpdatesAutoCheck ( true ) ;
AppSetting - > setUpdateCheckFrequency ( min ( AppSetting - > updateCheckFrequency ( ) , 7 ) ) ;
AppSetting - > setAllowEarlyUpdates ( true ) ;
}
2018-04-22 14:22:18 +00:00
if ( AppSetting - > updatesAutoCheck ( ) ) {
int update_frequency = AppSetting - > updateCheckFrequency ( ) ;
2014-04-17 05:52:25 +00:00
int days = 1000 ;
2018-04-22 14:22:18 +00:00
lastchecked = AppSetting - > updatesLastChecked ( ) ;
2014-04-17 05:52:25 +00:00
2018-04-22 14:22:18 +00:00
if ( lastchecked . isValid ( ) ) {
2014-04-17 05:52:25 +00:00
days = lastchecked . secsTo ( today ) ;
days / = 86400 ;
2018-04-22 14:22:18 +00:00
}
2014-04-17 05:52:25 +00:00
2020-07-05 01:17:25 +00:00
if ( days > = update_frequency ) {
2016-03-06 02:50:22 +00:00
check_updates = true ;
}
2011-10-21 05:50:31 +00:00
}
2019-02-15 01:37:29 +00:00
# endif
2011-10-21 05:50:31 +00:00
2020-01-15 22:15:24 +00:00
Version settingsVersion = Version ( AppSetting - > versionString ( ) ) ;
2020-01-16 16:54:41 +00:00
Version currentVersion = getVersion ( ) ;
if ( currentVersion . IsValid ( ) = = false ) {
// The defined version MUST be valid, otherwise comparisons between versions will fail.
QMessageBox : : critical ( nullptr , STR_MessageBox_Error , QObject : : tr ( " Version \" %1 \" is invalid, cannot continue! " ) . arg ( currentVersion ) ) ;
return 0 ;
}
if ( currentVersion > settingsVersion ) {
2018-05-07 12:13:07 +00:00
AppSetting - > setShowAboutDialog ( 1 ) ;
2019-03-25 23:44:49 +00:00
// release_notes();
// check_updates = false;
2020-01-16 16:54:41 +00:00
} else if ( currentVersion < settingsVersion ) {
2018-04-22 14:22:18 +00:00
if ( QMessageBox : : warning ( nullptr , STR_MessageBox_Error ,
Update version display throughout to use the new information and be consistent.
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.
2020-01-16 18:05:55 +00:00
QObject : : tr ( " The version of OSCAR you are running (%1) is OLDER than the one used to create this data (%2). " )
. arg ( currentVersion . displayString ( ) )
. arg ( settingsVersion . displayString ( ) )
+ " \n \n " +
2019-04-02 14:04:44 +00:00
QObject : : tr ( " It is likely that doing this will cause data corruption, are you sure you want to do this? " ) ,
QMessageBox : : Yes | QMessageBox : : No , QMessageBox : : No ) = = QMessageBox : : No ) {
2014-04-17 05:52:25 +00:00
2018-04-22 14:22:18 +00:00
return 0 ;
2011-10-01 12:54:20 +00:00
}
}
2011-10-21 05:50:31 +00:00
2020-01-16 00:45:46 +00:00
AppSetting - > setVersionString ( getVersion ( ) ) ;
2014-04-17 05:52:25 +00:00
2018-06-07 01:53:20 +00:00
////////////////////////////////////////////////////////////////////////////////////////////
// Register Importer Modules for autoscanner
////////////////////////////////////////////////////////////////////////////////////////////
schema : : init ( ) ;
PRS1Loader : : Register ( ) ;
ResmedLoader : : Register ( ) ;
IntellipapLoader : : Register ( ) ;
2021-07-08 17:43:02 +00:00
SleepStyleLoader : : Register ( ) ;
2021-10-21 17:40:58 +00:00
FPIconLoader : : Register ( ) ;
2018-06-07 01:53:20 +00:00
WeinmannLoader : : Register ( ) ;
CMS50Loader : : Register ( ) ;
CMS50F37Loader : : Register ( ) ;
MD300W1Loader : : Register ( ) ;
2020-01-23 17:51:58 +00:00
ViatomLoader : : Register ( ) ;
2021-12-07 19:57:26 +00:00
PrismaLoader : : Register ( ) ;
Support for loading Hoffrichter Point 3 machine sdcard data
The data of this machine can be read by iMatrix software from Resvent iBreeze serie. Probably this code also support Resvent iBreeze serie. But was only tested with data from the machine in the title.
Events loaded: CPAP_Obstructive, CPAP_Apnea, CPAP_Hypopnea, CPAP_FlowLimit, CPAP_RERA, CPAP_PB, CPAP_Snore
Waveforms loaded: CPAP_MaskPressure, CPAP_FlowRate
Other Events loaded: CPAP_Pressure, CPAP_IPAP, CPAP_EPAP, CPAP_Leak, CPAP_TidalVolume, CPAP_MinuteVent, CPAP_RespRate, CPAP_Ti, CPAP_IE
2023-05-06 08:49:06 +00:00
ResventLoader : : Register ( ) ;
2018-06-07 01:53:20 +00:00
2020-07-19 01:44:05 +00:00
// Begin logging device connection activity.
QString connectionsLogDir = GetLogDir ( ) + " /connections " ;
rotateLogs ( connectionsLogDir ) ; // keep a limited set of previous logs
if ( ! QDir ( connectionsLogDir ) . mkpath ( " . " ) ) {
qWarning ( ) . noquote ( ) < < " Unable to create directory " < < connectionsLogDir ;
}
QFile deviceLog ( connectionsLogDir + " /devices.xml " ) ;
if ( deviceLog . open ( QFile : : ReadWrite ) ) {
qDebug ( ) . noquote ( ) < < " Logging device connections to " < < deviceLog . fileName ( ) ;
DeviceConnectionManager : : getInstance ( ) . record ( & deviceLog ) ;
} else {
qWarning ( ) . noquote ( ) < < " Unable to start device connection logging to " < < deviceLog . fileName ( ) ;
}
2018-06-07 01:53:20 +00:00
schema : : setOrders ( ) ; // could be called in init...
2018-04-22 12:06:48 +00:00
// Scan for user profiles
Profiles : : Scan ( ) ;
2020-07-05 01:17:25 +00:00
# ifndef NO_CHECKUPDATES
2019-04-02 14:04:44 +00:00
if ( check_updates ) {
2020-07-05 01:17:25 +00:00
mainwin - > CheckForUpdates ( false ) ;
2019-04-02 14:04:44 +00:00
}
2019-02-15 01:37:29 +00:00
# endif
2011-10-21 05:50:31 +00:00
2018-06-07 01:53:20 +00:00
mainwin - > SetupGUI ( ) ;
mainwin - > show ( ) ;
2011-10-01 14:08:43 +00:00
2023-09-12 23:30:59 +00:00
int result = mainapp . exec ( ) ;
2020-07-19 01:44:05 +00:00
DeviceConnectionManager : : getInstance ( ) . record ( nullptr ) ;
2023-09-12 23:30:59 +00:00
2020-07-19 01:44:05 +00:00
return result ;
2011-06-26 08:30:44 +00:00
}
2019-05-03 01:51:56 +00:00
# endif // !UNITTEST_MODE