2019-03-10 16:03:19 +00:00
/* OSCAR Main
2014-04-09 21:01:57 +00:00
*
2020-01-18 16:41:09 +00:00
* Copyright ( c ) 2019 - 2020 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
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"
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"
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"
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 ) ;
while ( instr . readLineInto ( & line ) ) {
line . replace ( " SleepyHead " , " OSCAR " ) ;
if ( line . contains ( " VersionString " ) ) {
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 ,
2019-03-29 02:31:25 +00:00
QObject : : tr ( " Choose the SleepyHead 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 ,
QObject : : tr ( " The folder you chose does not contain valid SleepyHead data. " ) +
" \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 " +
QObject : : tr ( " from " ) + QDir ( datadir ) . dirName ( ) + " \n " + QObject : : tr ( " to " ) +
QDir ( destDir ) . dirName ( ) , QString ( ) , 0 , numFiles , 0 , Qt : : WindowSystemMenuHint | Qt : : WindowTitleHint ) ;
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
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 ;
}
2019-04-02 14:04:44 +00:00
int main ( int argc , char * argv [ ] ) {
2013-01-17 18:26:11 +00:00
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 ( ) ) ;
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 = " " ;
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
}
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 ) ;
}
QApplication a ( argc , argv ) ;
QStringList args = a . arguments ( ) ;
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
QString lastlanguage = settings . value ( LangSetting , " " ) . toString ( ) ;
2019-08-19 19:32:26 +00:00
if ( lastlanguage . compare ( " is " , Qt : : CaseInsensitive ) ) // Convert code for Hebrew from 'is' to 'he'
lastlanguage = " he " ;
2018-06-12 12:55:44 +00:00
2018-04-22 12:06:48 +00:00
bool dont_load_profile = false ;
2014-04-17 05:52:25 +00:00
bool force_data_dir = false ;
2014-10-02 17:46:08 +00:00
bool changing_language = false ;
2017-09-02 12:01:01 +00:00
QString load_profile = " " ;
2013-10-19 02:59:52 +00:00
2014-10-02 17:46:08 +00:00
if ( lastlanguage . isEmpty ( ) )
changing_language = true ;
2014-04-17 05:52:25 +00:00
for ( int i = 1 ; i < args . size ( ) ; i + + ) {
2019-03-25 23:44:49 +00:00
if ( args [ i ] = = " -l " )
dont_load_profile = true ;
// else if (args[i] == "-d")
// force_data_dir = true;
2018-06-12 12:55:44 +00:00
else if ( args [ i ] = = " --language " ) {
2019-03-25 23:44:49 +00:00
changing_language = true ; // reset to force language dialog
2014-10-02 17:46:08 +00:00
settings . setValue ( LangSetting , " " ) ;
2019-05-31 22:50:16 +00:00
}
2020-02-17 15:59:26 +00:00
// "--legacy" is handle above, as it needs to be processed before creating QApplication.
2019-05-31 22:50:16 +00:00
else if ( args [ i ] = = " -p " )
2018-06-12 04:00:38 +00:00
QThread : : msleep ( 1000 ) ;
2019-03-25 23:44:49 +00:00
else if ( args [ i ] = = " --profile " ) {
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 {
2017-09-02 12:01:01 +00:00
fprintf ( stderr , " Missing argument to --profile \n " ) ;
exit ( 1 ) ;
}
2018-05-03 05:08:45 +00:00
} else if ( args [ i ] = = " --datadir " ) { // mltam's idea
QString datadir ;
if ( ( i + 1 ) < args . size ( ) ) {
2019-04-02 14:04:44 +00:00
datadir = args [ + + i ] ;
2019-07-12 19:26:05 +00:00
if ( datadir . length ( ) < 2 | | datadir . at ( 1 ) ! = QLatin1Char ( ' : ' ) ) // Allow a Windows drive letter (but not UNC)
2019-06-24 17:05:58 +00:00
datadir = homeDocs + datadir ;
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 {
2019-04-02 14:04:44 +00:00
fprintf ( stderr , " Missing argument to --datadir \n " ) ;
exit ( 1 ) ;
2018-05-03 05:08:45 +00:00
}
2019-04-02 14:04:44 +00:00
}
2019-03-25 23:44:49 +00:00
} // end of for args loop
2011-11-18 09:05:22 +00:00
2018-06-12 04:00:38 +00:00
initializeLogger ( ) ;
2020-02-17 15:59:26 +00:00
// 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
2020-08-16 19:46:36 +00:00
// the intervening messages show up in the debug pane.
2020-02-17 15:59:26 +00:00
//
// 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.
2019-06-02 02:51:18 +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 ( ) ;
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 " +
QObject : : tr ( " If you have been using SleepyHead, OSCAR can copy your old data to this folder later. " ) + " \n " +
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)
2020-07-05 01:17:25 +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
2020-07-05 01:17:25 +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
2019-03-31 13:43:41 +00:00
if ( QMessageBox : : question ( nullptr , QObject : : tr ( " Migrate SleepyHead Data? " ) ,
2019-04-02 14:04:44 +00:00
QObject : : tr ( " On the next screen OSCAR will ask you to select a folder with SleepyHead data " ) + " \n " +
QObject : : tr ( " Click [OK] to go to the next screen or [No] if you do not wish to use any SleepyHead data. " ) ,
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 ) ) {
QString errMsg = QObject : : tr ( " Unable to write to OSCAR data directory " ) + " \n " +
GetAppData ( ) + " \n " +
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
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
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 ( ) ;
FPIconLoader : : Register ( ) ;
WeinmannLoader : : Register ( ) ;
CMS50Loader : : Register ( ) ;
CMS50F37Loader : : Register ( ) ;
MD300W1Loader : : Register ( ) ;
2020-01-23 17:51:58 +00:00
ViatomLoader : : 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 ( ) ;
Q_UNUSED ( changing_language )
2018-06-12 15:57:24 +00:00
Q_UNUSED ( dont_load_profile )
2018-04-22 12:06:48 +00:00
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
2020-07-19 01:44:05 +00:00
int result = a . exec ( ) ;
DeviceConnectionManager : : getInstance ( ) . record ( nullptr ) ;
return result ;
2011-06-26 08:30:44 +00:00
}
2019-05-03 01:51:56 +00:00
# endif // !UNITTEST_MODE