2019-05-03 20:17:32 +00:00
/* SleepLib Profiles Implementation
2014-04-09 21:01:57 +00:00
*
2021-11-02 20:34:12 +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-01-26 12:48:25 +00:00
# define TEST_MACROS_ENABLEDoff
# include <test_macros.h>
2011-06-26 08:30:44 +00:00
# include <QString>
# include <QDateTime>
# include <QDir>
# include <QMessageBox>
2011-07-01 10:10:44 +00:00
# include <QDebug>
2014-07-02 03:22:09 +00:00
# include <QProcess>
2014-07-12 10:52:14 +00:00
# include <QByteArray>
# include <QHostInfo>
2018-04-22 12:06:48 +00:00
# include <QApplication>
# include <QSettings>
2012-01-02 15:34:17 +00:00
# include <algorithm>
2012-01-05 04:37:22 +00:00
# include <cmath>
2011-06-26 08:30:44 +00:00
# include "preferences.h"
# include "profiles.h"
# include "machine.h"
2014-07-28 13:56:29 +00:00
# include "machine_common.h"
2011-06-26 08:30:44 +00:00
# include "machine_loader.h"
2011-12-11 09:45:28 +00:00
# include "mainwindow.h"
2018-04-22 12:06:48 +00:00
# include "translation.h"
2020-01-15 21:34:28 +00:00
# include "version.h"
2011-12-11 09:45:28 +00:00
2014-04-17 05:58:57 +00:00
extern MainWindow * mainwin ;
2011-06-26 08:30:44 +00:00
Preferences * p_pref ;
Preferences * p_layout ;
2014-04-17 05:58:57 +00:00
Profile * p_profile ;
2011-06-26 08:30:44 +00:00
2020-01-29 14:10:29 +00:00
Profile : : Profile ( QString path , bool open )
2014-04-25 05:28:10 +00:00
: is_first_day ( true ) ,
2019-08-15 20:56:44 +00:00
m_opened ( false )
2014-04-17 05:58:57 +00:00
{
p_name = STR_GEN_Profile ;
if ( path . isEmpty ( ) ) {
2019-03-25 23:36:41 +00:00
p_path = GetAppData ( ) ;
2014-04-20 05:25:09 +00:00
} else {
p_path = path ;
}
2014-04-17 05:58:57 +00:00
( * this ) [ STR_GEN_DataFolder ] = p_path ;
path = path . replace ( " \\ " , " / " ) ;
if ( ! p_path . endsWith ( " / " ) ) {
p_path + = " / " ;
}
p_filename = p_path + p_name + STR_ext_XML ;
2018-04-22 12:06:48 +00:00
m_machlist . clear ( ) ;
2020-01-29 14:10:29 +00:00
if ( open ) {
Open ( p_filename ) ;
}
2011-12-21 11:09:50 +00:00
2018-04-22 12:06:48 +00:00
Set ( STR_GEN_DataFolder , QString ( " {home}/Profiles/{UserName} " ) ) ;
2019-12-30 20:25:37 +00:00
// Reset import warnings when running a new version of OSCAR
2020-01-16 00:45:46 +00:00
init ( STR_PREF_VersionString , getVersion ( ) . toString ( ) ) ;
Version prefVersion = Version ( ( * this ) [ STR_PREF_VersionString ] . toString ( ) ) ;
if ( prefVersion ! = getVersion ( ) ) {
qDebug ( ) < < " Resetting import warnings: version " < < prefVersion < < " to " < < getVersion ( ) ;
Set ( STR_PREF_VersionString , getVersion ( ) . toString ( ) ) ;
2019-12-30 20:25:37 +00:00
this - > Erase ( STR_IS_WarnOnUntestedMachine ) ;
this - > Erase ( STR_IS_WarnOnUnexpectedData ) ;
}
2018-04-22 12:06:48 +00:00
doctor = new DoctorInfo ( this ) ;
user = new UserInfo ( this ) ;
cpap = new CPAPSettings ( this ) ;
oxi = new OxiSettings ( this ) ;
appearance = new AppearanceSettings ( this ) ;
session = new SessionSettings ( this ) ;
general = new UserSettings ( this ) ;
2020-01-29 14:10:29 +00:00
if ( open ) {
OpenMachines ( ) ;
m_opened = true ;
}
2011-06-26 08:30:44 +00:00
}
2011-12-11 09:45:28 +00:00
2011-12-11 10:59:30 +00:00
Profile : : ~ Profile ( )
{
2020-01-29 14:10:29 +00:00
if ( m_opened ) {
removeLock ( ) ;
}
2014-04-25 05:28:10 +00:00
2022-02-27 16:01:46 +00:00
// delete device objects...
2018-05-05 21:58:11 +00:00
for ( auto & mach : m_machlist ) {
delete mach ;
2011-06-26 08:30:44 +00:00
}
2014-08-20 17:17:13 +00:00
2018-05-05 21:58:11 +00:00
for ( auto & day : daylist ) {
delete day ;
2014-08-20 17:17:13 +00:00
}
2020-02-03 16:43:13 +00:00
delete user ;
delete doctor ;
delete cpap ;
delete oxi ;
delete appearance ;
delete session ;
delete general ;
2011-06-26 08:30:44 +00:00
}
2012-01-12 22:46:06 +00:00
2014-04-20 05:25:09 +00:00
bool Profile : : Save ( QString filename )
{
2014-04-25 05:28:10 +00:00
if ( m_opened ) {
2018-04-22 12:06:48 +00:00
return Preferences : : Save ( filename ) & & StoreMachines ( ) ;
2014-04-25 05:28:10 +00:00
} else return false ;
2014-04-20 05:25:09 +00:00
}
2014-07-12 10:52:14 +00:00
bool Profile : : removeLock ( )
{
QString filename = p_path + " /lockfile " ;
QFile file ( filename ) ;
return file . remove ( ) ;
}
QString Profile : : checkLock ( )
{
QString filename = p_path + " /lockfile " ;
QFile file ( filename ) ;
if ( ! file . exists ( ) )
return QString ( ) ;
file . open ( QFile : : ReadOnly ) ;
QString lockhost = file . readLine ( 1024 ) . trimmed ( ) ;
return lockhost ;
}
2020-08-23 22:30:38 +00:00
// Properties for machines.xml:
2014-09-14 15:29:07 +00:00
const QString STR_PROP_Brand = " brand " ;
const QString STR_PROP_Model = " model " ;
const QString STR_PROP_Series = " series " ;
const QString STR_PROP_ModelNumber = " modelnumber " ;
const QString STR_PROP_SubModel = " submodel " ;
const QString STR_PROP_Serial = " serial " ;
const QString STR_PROP_DataVersion = " dataversion " ;
const QString STR_PROP_LastImported = " lastimported " ;
2020-08-23 22:30:38 +00:00
const QString STR_PROP_PurgeDate = " purgedate " ;
2014-09-14 15:29:07 +00:00
2018-04-22 12:06:48 +00:00
void Profile : : addLock ( )
2014-09-14 15:29:07 +00:00
{
QFile lockfile ( p_path + " lockfile " ) ;
lockfile . open ( QFile : : WriteOnly ) ;
QByteArray ba ;
ba . append ( QHostInfo : : localHostName ( ) ) ;
lockfile . write ( ba ) ;
lockfile . close ( ) ;
2018-04-22 12:06:48 +00:00
}
bool Profile : : OpenMachines ( )
{
2019-08-15 20:56:44 +00:00
if ( m_machlist . size ( ) > 0 ) {
qCritical ( ) < < " Skipping redundant call to Profile::OpenMachines " ;
2018-04-22 12:06:48 +00:00
return true ;
2019-08-15 20:56:44 +00:00
}
2014-09-14 15:29:07 +00:00
QString filename = p_path + " machines.xml " ;
QFile file ( filename ) ;
if ( ! file . open ( QFile : : ReadOnly ) ) {
2020-08-09 17:33:04 +00:00
qWarning ( ) < < " Could not open " < < filename . toLocal8Bit ( ) . data ( ) < < " for reading, error code " < < file . error ( ) < < file . errorString ( ) ;
2014-09-14 15:29:07 +00:00
return false ;
}
2020-08-23 22:30:38 +00:00
// qDebug() << "OpenMachines opened" << filename.toLocal8Bit().data();
2014-09-14 15:29:07 +00:00
QDomDocument doc ( " machines.xml " ) ;
if ( ! doc . setContent ( & file ) ) {
2018-06-07 01:53:20 +00:00
qWarning ( ) < < " Invalid XML Content in " < < filename . toLocal8Bit ( ) . data ( ) ;
2014-09-14 15:29:07 +00:00
return false ;
}
file . close ( ) ;
QDomElement root = doc . firstChild ( ) . toElement ( ) ;
if ( root . tagName ( ) . toLower ( ) ! = " machines " ) {
2020-08-23 22:30:38 +00:00
qWarning ( ) < < " No Machines Tag in machines.xml " ;
2014-09-14 15:29:07 +00:00
return false ;
}
QDomElement elem = root . firstChildElement ( ) ;
while ( ! elem . isNull ( ) ) {
QString pKey = elem . tagName ( ) ;
if ( pKey . toLower ( ) ! = " machine " ) {
2018-04-22 12:06:48 +00:00
qWarning ( ) < < " Profile::OpenMachines() pKey!= \" machine \" " ;
2014-09-14 15:29:07 +00:00
elem = elem . nextSiblingElement ( ) ;
continue ;
}
int m_id ;
bool ok ;
m_id = elem . attribute ( " id " , " " ) . toInt ( & ok ) ;
int mt ;
mt = elem . attribute ( " type " , " " ) . toInt ( & ok ) ;
MachineType m_type = ( MachineType ) mt ;
QString m_class = elem . attribute ( " class " , " " ) ;
MachineInfo info ;
info . type = m_type ;
info . loadername = m_class ;
QHash < QString , QString > prop ;
QDomElement e = elem . firstChildElement ( ) ;
for ( ; ! e . isNull ( ) ; e = e . nextSiblingElement ( ) ) {
QString pKey = e . tagName ( ) ;
QString key = pKey . toLower ( ) ;
if ( key = = STR_PROP_Brand ) {
info . brand = e . text ( ) ;
} else if ( key = = STR_PROP_Model ) {
info . model = e . text ( ) ;
} else if ( key = = STR_PROP_ModelNumber ) {
info . modelnumber = e . text ( ) ;
} else if ( key = = STR_PROP_Serial ) {
info . serial = e . text ( ) ;
} else if ( key = = STR_PROP_Series ) {
info . series = e . text ( ) ;
} else if ( key = = STR_PROP_DataVersion ) {
info . version = e . text ( ) . toInt ( ) ;
} else if ( key = = STR_PROP_LastImported ) {
info . lastimported = QDateTime : : fromString ( e . text ( ) , Qt : : ISODate ) ;
2020-08-23 22:30:38 +00:00
} else if ( key = = STR_PROP_PurgeDate ) {
info . purgeDate = QDate : : fromString ( e . text ( ) , Qt : : ISODate ) ;
2014-09-14 15:29:07 +00:00
} else if ( key = = " properties " ) {
QDomElement pe = e . firstChildElement ( ) ;
for ( ; ! pe . isNull ( ) ; pe = pe . nextSiblingElement ( ) ) {
prop [ pe . tagName ( ) ] = pe . text ( ) ;
}
} else {
// skip any old rubbish
if ( ( key = = " backuppath " ) | | ( key = = " path " ) | | ( key = = " submodel " ) ) continue ;
prop [ pKey ] = e . text ( ) ;
}
}
Machine * m = nullptr ;
2022-02-27 16:01:46 +00:00
// Create device needs a profile passed to it..
2018-04-22 12:06:48 +00:00
m = CreateMachine ( info , m_id ) ;
2021-10-26 18:59:41 +00:00
if ( m ) m - > info . properties = prop ;
2014-09-14 15:29:07 +00:00
elem = elem . nextSiblingElement ( ) ;
}
return true ;
}
bool Profile : : StoreMachines ( )
{
QDomDocument doc ( " Machines " ) ;
QDomElement mach = doc . createElement ( " machines " ) ;
2018-04-22 12:06:48 +00:00
for ( int i = 0 ; i < m_machlist . size ( ) ; + + i ) {
Machine * m = m_machlist [ i ] ;
2014-09-14 15:29:07 +00:00
QDomElement me = doc . createElement ( " machine " ) ;
me . setAttribute ( " id " , ( int ) m - > id ( ) ) ;
me . setAttribute ( " type " , ( int ) m - > type ( ) ) ;
me . setAttribute ( " class " , m - > loaderName ( ) ) ;
QDomElement pe = doc . createElement ( " properties " ) ;
me . appendChild ( pe ) ;
2021-10-26 18:59:41 +00:00
for ( QHash < QString , QString > : : iterator j = m - > info . properties . begin ( ) ; j ! = m - > info . properties . end ( ) ; j + + ) {
2014-09-14 15:29:07 +00:00
QDomElement pp = doc . createElement ( j . key ( ) ) ;
pp . appendChild ( doc . createTextNode ( j . value ( ) ) ) ;
pe . appendChild ( pp ) ;
}
QDomElement mp = doc . createElement ( STR_PROP_Brand ) ;
mp . appendChild ( doc . createTextNode ( m - > brand ( ) ) ) ;
me . appendChild ( mp ) ;
mp = doc . createElement ( STR_PROP_Model ) ;
mp . appendChild ( doc . createTextNode ( m - > model ( ) ) ) ;
me . appendChild ( mp ) ;
mp = doc . createElement ( STR_PROP_ModelNumber ) ;
mp . appendChild ( doc . createTextNode ( m - > modelnumber ( ) ) ) ;
me . appendChild ( mp ) ;
mp = doc . createElement ( STR_PROP_Serial ) ;
mp . appendChild ( doc . createTextNode ( m - > serial ( ) ) ) ;
me . appendChild ( mp ) ;
mp = doc . createElement ( STR_PROP_Series ) ;
mp . appendChild ( doc . createTextNode ( m - > series ( ) ) ) ;
me . appendChild ( mp ) ;
mp = doc . createElement ( STR_PROP_DataVersion ) ;
mp . appendChild ( doc . createTextNode ( QString : : number ( m - > version ( ) ) ) ) ;
me . appendChild ( mp ) ;
mp = doc . createElement ( STR_PROP_LastImported ) ;
mp . appendChild ( doc . createTextNode ( m - > lastImported ( ) . toString ( Qt : : ISODate ) ) ) ;
me . appendChild ( mp ) ;
2020-08-23 22:30:38 +00:00
mp = doc . createElement ( STR_PROP_PurgeDate ) ;
mp . appendChild ( doc . createTextNode ( m - > purgeDate ( ) . toString ( Qt : : ISODate ) ) ) ;
me . appendChild ( mp ) ;
2014-09-14 15:29:07 +00:00
mach . appendChild ( me ) ;
}
doc . appendChild ( mach ) ;
QString filename = p_path + " machines.xml " ;
QFile file ( filename ) ;
if ( ! file . open ( QFile : : WriteOnly ) ) {
2020-08-09 17:33:04 +00:00
qWarning ( ) < < " Could not open " < < filename < < " for writing, error code " < < file . error ( ) < < file . errorString ( ) ;
2014-09-14 15:29:07 +00:00
return false ;
}
2020-08-23 22:30:38 +00:00
2014-09-14 15:29:07 +00:00
file . write ( doc . toByteArray ( ) ) ;
return true ;
}
2018-05-06 21:45:56 +00:00
qint64 Profile : : diskSpaceSummaries ( )
{
qint64 size = 0 ;
for ( auto & mach : m_machlist ) {
size + = mach - > diskSpaceSummaries ( ) ;
}
return size ;
}
qint64 Profile : : diskSpaceEvents ( )
{
qint64 size = 0 ;
for ( auto & mach : m_machlist ) {
size + = mach - > diskSpaceEvents ( ) ;
}
return size ;
}
qint64 Profile : : diskSpaceBackups ( )
{
qint64 size = 0 ;
for ( auto & mach : m_machlist ) {
size + = mach - > diskSpaceBackups ( ) ;
}
return size ;
}
qint64 Profile : : diskSpace ( )
{
return ( diskSpaceSummaries ( ) + diskSpaceEvents ( ) + diskSpaceBackups ( ) ) ;
}
2020-02-07 02:34:04 +00:00
void Profile : : forceResmedPrefs ( )
{
session - > setBackupCardData ( true ) ;
session - > setDaySplitTime ( QTime ( 12 , 0 , 0 ) ) ;
session - > setIgnoreShortSessions ( 0 ) ;
session - > setCombineCloseSessions ( 0 ) ;
session - > setLockSummarySessions ( true ) ;
general - > setPrefCalcPercentile ( 95.0 ) ; // 95%
general - > setPrefCalcMiddle ( 0 ) ; // Median (50%)
general - > setPrefCalcMax ( 1 ) ; // 99.9th percentile max
}
2014-09-14 15:29:07 +00:00
2014-07-03 02:31:50 +00:00
# if defined(Q_OS_WIN)
2014-07-03 02:45:38 +00:00
class Environment
2014-07-03 02:31:50 +00:00
{
2014-07-03 02:45:38 +00:00
public :
Environment ( ) ;
QStringList path ( ) ;
QString searchInDirectory ( const QStringList & execs , QString directory ) ;
QString searchInPath ( const QString & executable , const QStringList & additionalDirs = QStringList ( ) ) ;
QProcessEnvironment env ;
} ;
Environment : : Environment ( )
{
env = QProcessEnvironment : : systemEnvironment ( ) ;
}
QStringList Environment : : path ( )
{
return env . value ( QLatin1String ( " PATH " ) , " " ) . split ( ' ; ' ) ;
2014-07-03 02:31:50 +00:00
}
2014-07-03 02:45:38 +00:00
QString Environment : : searchInDirectory ( const QStringList & execs , QString directory )
2014-07-03 02:31:50 +00:00
{
const QChar slash = QLatin1Char ( ' / ' ) ;
if ( directory . isEmpty ( ) )
return QString ( ) ;
if ( ! directory . endsWith ( slash ) )
directory + = slash ;
2018-05-05 21:58:11 +00:00
for ( auto & exec : execs ) {
2014-07-03 02:31:50 +00:00
QFileInfo fi ( directory + exec ) ;
if ( fi . exists ( ) & & fi . isFile ( ) & & fi . isExecutable ( ) )
return fi . absoluteFilePath ( ) ;
}
return QString ( ) ;
}
2014-07-03 02:45:38 +00:00
QString Environment : : searchInPath ( const QString & executable , const QStringList & additionalDirs )
2014-07-03 02:31:50 +00:00
{
if ( executable . isEmpty ( ) ) return QString ( ) ;
QString exec = QDir : : cleanPath ( executable ) ;
QFileInfo fi ( exec ) ;
QStringList execs ( exec ) ;
if ( fi . suffix ( ) . isEmpty ( ) ) {
2014-07-03 02:45:38 +00:00
QStringList extensions = env . value ( QLatin1String ( " PATHEXT " ) ) . split ( QLatin1Char ( ' ; ' ) ) ;
2014-07-03 02:31:50 +00:00
foreach ( const QString & ext , extensions ) {
QString tmp = executable + ext . toLower ( ) ;
if ( fi . isAbsolute ( ) ) {
if ( QFile : : exists ( tmp ) )
return tmp ;
} else {
execs < < tmp ;
}
}
}
if ( fi . isAbsolute ( ) )
return exec ;
QSet < QString > alreadyChecked ;
foreach ( const QString & dir , additionalDirs ) {
if ( alreadyChecked . contains ( dir ) )
continue ;
alreadyChecked . insert ( dir ) ;
QString tmp = searchInDirectory ( execs , dir ) ;
if ( ! tmp . isEmpty ( ) )
return tmp ;
}
if ( executable . indexOf ( QLatin1Char ( ' / ' ) ) ! = - 1 )
return QString ( ) ;
2018-05-05 21:58:11 +00:00
for ( auto & p : path ( ) ) {
2014-07-03 02:31:50 +00:00
if ( alreadyChecked . contains ( p ) )
continue ;
alreadyChecked . insert ( p ) ;
QString tmp = searchInDirectory ( execs , QDir : : fromNativeSeparators ( p ) ) ;
if ( ! tmp . isEmpty ( ) )
return tmp ;
}
return QString ( ) ;
}
# endif
2014-07-02 03:22:09 +00:00
// Borrowed from QtCreator (http://stackoverflow.com/questions/3490336/how-to-reveal-in-finder-or-show-in-explorer-with-qt)
2014-10-08 16:59:24 +00:00
void showInGraphicalShell ( const QString & pathIn )
2014-07-02 03:22:09 +00:00
{
// Mac, Windows support folder or file.
# if defined(Q_OS_WIN)
2015-08-27 04:00:58 +00:00
QWidget * parent = NULL ;
2014-07-03 02:45:38 +00:00
Environment env ;
const QString explorer = env . searchInPath ( QLatin1String ( " explorer.exe " ) ) ;
2014-07-02 03:22:09 +00:00
if ( explorer . isEmpty ( ) ) {
QMessageBox : : warning ( parent ,
2014-07-03 02:45:38 +00:00
QObject : : tr ( " Launching Windows Explorer failed " ) ,
QObject : : tr ( " Could not find explorer.exe in path to launch Windows Explorer. " ) ) ;
2014-07-02 03:22:09 +00:00
return ;
}
QString param ;
2014-07-03 02:45:38 +00:00
//if (!QFileInfo(pathIn).isDir())
2014-07-02 03:22:09 +00:00
param = QLatin1String ( " /select, " ) ;
param + = QDir : : toNativeSeparators ( pathIn ) ;
QProcess : : startDetached ( explorer , QStringList ( param ) ) ;
# elif defined(Q_OS_MAC)
2014-10-08 16:51:09 +00:00
// Q_UNUSED(parent)
2014-07-02 03:22:09 +00:00
QStringList scriptArgs ;
scriptArgs < < QLatin1String ( " -e " )
< < QString : : fromLatin1 ( " tell application \" Finder \" to reveal POSIX file \" %1 \" " )
. arg ( pathIn ) ;
QProcess : : execute ( QLatin1String ( " /usr/bin/osascript " ) , scriptArgs ) ;
scriptArgs . clear ( ) ;
scriptArgs < < QLatin1String ( " -e " )
< < QLatin1String ( " tell application \" Finder \" to activate " ) ;
QProcess : : execute ( " /usr/bin/osascript " , scriptArgs ) ;
# else
2014-10-08 16:59:24 +00:00
Q_UNUSED ( pathIn ) ;
2014-07-02 03:22:09 +00:00
// we cannot select a file here, because no file browser really supports it...
2014-07-04 03:45:29 +00:00
/*
2014-07-02 03:22:09 +00:00
const QFileInfo fileInfo ( pathIn ) ;
const QString folder = fileInfo . absoluteFilePath ( ) ;
const QString app = Utils : : UnixUtils : : fileBrowser ( Core : : ICore : : instance ( ) - > settings ( ) ) ;
QProcess browserProc ;
const QString browserArgs = Utils : : UnixUtils : : substituteFileBrowserParameters ( app , folder ) ;
if ( debug )
qDebug ( ) < < browserArgs ;
bool success = browserProc . startDetached ( browserArgs ) ;
const QString error = QString : : fromLocal8Bit ( browserProc . readAllStandardError ( ) ) ;
success = success & & error . isEmpty ( ) ;
if ( ! success ) {
QMessageBox : : warning ( NULL , STR_MessageBox_Error , " Could not find the file browser for your system, you will have to find your profile directory yourself. " + " \n \n " + error , QMessageBox : : Ok ) ;
// showGraphicalShellError(parent, app, error);
2014-07-04 03:45:29 +00:00
} */
2014-07-02 03:22:09 +00:00
# endif
}
2014-07-03 01:59:50 +00:00
int dirCount ( QString path )
{
QDir dir ( path ) ;
QStringList list = dir . entryList ( QDir : : NoDotAndDotDot | QDir : : Files | QDir : : Dirs ) ;
return list . size ( ) ;
}
2011-07-29 22:55:24 +00:00
void Profile : : DataFormatError ( Machine * m )
{
2014-07-02 03:22:09 +00:00
QString msg ;
2014-04-17 05:58:57 +00:00
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
msg = " <font size=+1> " + QObject : : tr ( " OSCAR %1 needs to upgrade its database for %2 %3 %4 " ) .
arg ( getVersion ( ) . displayString ( ) ) .
2014-07-28 13:56:29 +00:00
arg ( m - > brand ( ) ) . arg ( m - > model ( ) ) . arg ( m - > serial ( ) )
2014-07-02 03:22:09 +00:00
+ " </font><br/><br/> " ;
2014-04-17 05:58:57 +00:00
2014-07-02 03:22:09 +00:00
bool backups = false ;
2014-07-28 13:56:29 +00:00
if ( p_profile - > session - > backupCardData ( ) ) {
QString bpath = m - > getBackupPath ( ) ;
2014-07-03 01:59:50 +00:00
int cnt = dirCount ( bpath ) ;
if ( cnt > 0 ) backups = true ;
2014-07-02 03:22:09 +00:00
}
2014-05-13 01:28:41 +00:00
2014-07-02 03:22:09 +00:00
if ( backups ) {
2019-02-24 00:58:25 +00:00
msg = msg + QObject : : tr ( " <b>OSCAR maintains a backup of your devices data card that it uses for this purpose.</b> " ) + " <br/><br/> " ;
2022-02-27 16:01:46 +00:00
msg = msg + QObject : : tr ( " <i>Your old device data should be regenerated provided this backup feature has not been disabled in preferences during a previous data import.</i> " ) + " <br/><br/> " ;
2014-07-02 03:22:09 +00:00
backups = true ;
} else {
2019-02-24 00:58:25 +00:00
msg = msg + " <font size=+1> " + STR_MessageBox_Warning + " :</font> " + QObject : : tr ( " OSCAR does not yet have any automatic card backups stored for this device. " ) + " <br/><br/> " ;
2022-02-27 16:01:46 +00:00
msg = msg + QObject : : tr ( " This means you will need to import this device data again afterwards from your own backups or data card. " ) + " <br/><br/> " ;
2014-07-02 03:22:09 +00:00
}
2014-04-17 05:58:57 +00:00
2020-03-26 13:54:55 +00:00
msg + = " <font size=+1> " + QObject : : tr ( " Important: " ) + " </font> " + QObject : : tr ( " Once you upgrade, you <font size=+1>cannot</font> use this profile with the previous version anymore. " ) + " <br/><br/> " +
2019-02-24 00:58:25 +00:00
QObject : : tr ( " If you are concerned, click No to exit, and backup your profile manually, before starting OSCAR again. " ) + " <br/><br/> " ;
msg = msg + " <font size=+1> " + QObject : : tr ( " Are you ready to upgrade, so you can run the new version of OSCAR? " ) + " </font> " ;
2014-07-02 03:22:09 +00:00
2022-02-27 16:01:46 +00:00
QMessageBox * question = new QMessageBox ( QMessageBox : : Warning , QObject : : tr ( " Device Database Changes " ) , msg , QMessageBox : : Yes | QMessageBox : : No ) ;
2014-07-02 03:22:09 +00:00
question - > setDefaultButton ( QMessageBox : : Yes ) ;
QFont font ( " Sans Serif " , 11 , QFont : : Normal ) ;
question - > setFont ( font ) ;
if ( question - > exec ( ) = = QMessageBox : : Yes ) {
if ( ! m - > Purge ( 3478216 ) ) {
// Purge failed.. probably a permissions error.. let the user deal with it.
2014-05-17 05:04:40 +00:00
QMessageBox : : critical ( nullptr , STR_MessageBox_Error ,
2019-02-24 00:58:25 +00:00
QObject : : tr ( " Sorry, the purge operation failed, which means this version of OSCAR can't start. " ) + " \n \n " +
2022-02-27 16:01:46 +00:00
QObject : : tr ( " The device data folder needs to be removed manually. " ) + " \n \n " +
2014-05-17 08:34:16 +00:00
QObject : : tr ( " This folder currently resides at the following location: " ) + " \n \n " +
2014-07-02 03:22:09 +00:00
QDir : : toNativeSeparators ( Get ( p_preferences [ STR_GEN_DataFolder ] . toString ( ) ) ) , QMessageBox : : Ok ) ;
2012-01-06 16:07:54 +00:00
QApplication : : exit ( - 1 ) ;
2011-07-29 22:55:24 +00:00
}
2014-05-13 01:28:41 +00:00
// Note: I deliberately haven't added a Profile help for this
2014-07-02 03:22:09 +00:00
if ( backups ) {
2020-02-14 01:37:52 +00:00
MachineLoader * loader = lookupLoader ( m ) ;
2020-05-06 17:40:59 +00:00
int c = mainwin - > importCPAP ( ImportPath ( m - > getBackupPath ( ) , loader ) ,
QObject : : tr ( " Rebuilding from %1 Backup " ) . arg ( m - > brand ( ) ) ) ;
if ( c > = 0 ) {
// Make sure the updated version gets saved, even if there were no sessions to import.
mainwin - > finishCPAPImport ( ) ;
}
2014-07-02 03:22:09 +00:00
} else {
2014-07-03 01:59:50 +00:00
if ( ! p_profile - > session - > backupCardData ( ) ) {
// Automatic backups not available for Intellipap users yet, so don't taunt them..
2014-07-28 13:56:29 +00:00
if ( m - > loaderName ( ) ! = STR_MACH_Intellipap ) {
2019-02-24 00:58:25 +00:00
if ( QMessageBox : : question ( nullptr , STR_MessageBox_Question , QObject : : tr ( " Would you like to switch on automatic backups, so next time a new version of OSCAR needs to do so, it can rebuild from these? " ) ,
2014-07-03 01:59:50 +00:00
QMessageBox : : Yes | QMessageBox : : No , QMessageBox : : Yes ) ) {
p_profile - > session - > setBackupCardData ( true ) ;
}
}
}
2014-07-02 03:22:09 +00:00
QMessageBox : : information ( nullptr , STR_MessageBox_Information ,
2019-02-24 00:58:25 +00:00
QObject : : tr ( " OSCAR will now start the import wizard so you can reinstall your %1 data. " ) . arg ( m - > brand ( ) )
2014-07-02 03:22:09 +00:00
, QMessageBox : : Ok , QMessageBox : : Ok ) ;
mainwin - > startImportDialog ( ) ;
}
2014-07-11 12:09:38 +00:00
p_profile - > Save ( ) ;
2014-07-02 03:22:09 +00:00
delete question ;
2011-07-29 22:55:24 +00:00
} else {
2014-07-02 03:22:09 +00:00
delete question ;
QMessageBox : : information ( nullptr , STR_MessageBox_Information ,
2019-02-24 00:58:25 +00:00
QObject : : tr ( " OSCAR will now exit, then (attempt to) launch your computers file manager so you can manually back your profile up: " ) + " \n \n " +
2014-07-02 03:22:09 +00:00
QDir : : toNativeSeparators ( Get ( p_preferences [ STR_GEN_DataFolder ] . toString ( ) ) ) + " \n \n " +
2019-02-24 00:58:25 +00:00
QObject : : tr ( " Use your file manager to make a copy of your profile directory, then afterwards, restart OSCAR and complete the upgrade process. " )
2014-07-02 03:22:09 +00:00
, QMessageBox : : Ok , QMessageBox : : Ok ) ;
showInGraphicalShell ( Get ( p_preferences [ STR_GEN_DataFolder ] . toString ( ) ) ) ;
2012-01-06 16:07:54 +00:00
QApplication : : exit ( - 1 ) ;
2011-07-29 22:55:24 +00:00
}
2014-04-17 05:58:57 +00:00
2014-07-02 03:22:09 +00:00
2011-07-29 22:55:24 +00:00
return ;
}
2018-04-22 12:06:48 +00:00
void Profile : : UnloadMachineData ( )
{
2018-05-05 21:58:11 +00:00
for ( auto & mach : m_machlist ) {
2019-05-24 03:27:02 +00:00
mach - > saveSessionInfo ( ) ;
2018-05-05 21:58:11 +00:00
mach - > sessionlist . clear ( ) ;
mach - > day . clear ( ) ;
2018-04-22 12:06:48 +00:00
}
2019-05-26 15:18:42 +00:00
for ( auto & day : daylist ) {
delete day ;
}
daylist . clear ( ) ;
2018-04-22 12:06:48 +00:00
removeLock ( ) ;
}
2018-06-05 01:16:03 +00:00
void Profile : : LoadMachineData ( ProgressDialog * progress )
2011-06-26 08:30:44 +00:00
{
2018-04-22 12:06:48 +00:00
addLock ( ) ;
2011-12-11 09:45:28 +00:00
2018-05-05 21:58:11 +00:00
for ( auto & mach : m_machlist ) {
MachineLoader * loader = lookupLoader ( mach ) ;
2011-06-26 08:30:44 +00:00
if ( loader ) {
2018-05-05 21:58:11 +00:00
if ( mach - > version ( ) < loader - > Version ( ) ) {
2020-08-03 18:01:04 +00:00
qDebug ( ) < < " LoadMachineData " < < mach - > loaderName ( ) < < " data format error, machine version " < < mach - > version ( ) < < " loader version " < < loader - > Version ( ) ;
2019-12-28 04:29:18 +00:00
progress - > hide ( ) ;
2018-05-05 21:58:11 +00:00
DataFormatError ( mach ) ;
2019-12-28 04:29:18 +00:00
progress - > show ( ) ;
2011-07-29 22:55:24 +00:00
} else {
try {
2018-06-05 01:16:03 +00:00
mach - > Load ( progress ) ;
2016-01-04 06:32:37 +00:00
} catch ( OldDBVersion & e ) {
2020-08-03 18:01:04 +00:00
qDebug ( ) < < " LoadMachineData " < < mach - > loaderName ( ) < < " load failure, machine version " < < mach - > version ( ) < < " loader version " < < loader - > Version ( ) ;
2014-05-08 04:46:23 +00:00
Q_UNUSED ( e )
2019-12-28 04:29:18 +00:00
progress - > hide ( ) ;
2018-05-05 21:58:11 +00:00
DataFormatError ( mach ) ;
2019-12-28 04:29:18 +00:00
progress - > show ( ) ;
2011-06-26 08:30:44 +00:00
}
2011-07-29 22:55:24 +00:00
}
2011-06-26 08:30:44 +00:00
} else {
2018-06-05 01:16:03 +00:00
mach - > Load ( progress ) ;
2011-06-26 08:30:44 +00:00
}
}
2018-06-05 01:16:03 +00:00
progress - > setMessage ( " Loading Channel Information " ) ;
2018-04-22 12:06:48 +00:00
loadChannels ( ) ;
2011-06-26 08:30:44 +00:00
}
2018-05-05 21:58:11 +00:00
void Profile : : removeMachine ( Machine * mach )
2018-04-22 12:06:48 +00:00
{
2018-05-07 16:54:08 +00:00
if ( m_machlist . removeAll ( mach ) ) {
2014-07-28 13:56:29 +00:00
2018-05-07 16:54:08 +00:00
QHash < QString , QHash < QString , Machine * > > : : iterator mlit = MachineList . find ( mach - > loaderName ( ) ) ;
if ( mlit ! = MachineList . end ( ) ) {
QHash < QString , Machine * > : : iterator mit = mlit . value ( ) . find ( mach - > serial ( ) ) ;
if ( mit ! = mlit . value ( ) . end ( ) ) {
mlit . value ( ) . erase ( mit ) ;
}
2018-04-22 12:06:48 +00:00
}
}
}
Machine * Profile : : lookupMachine ( QString serial , QString loadername )
2011-06-26 08:30:44 +00:00
{
2018-05-05 21:58:11 +00:00
auto mlit = MachineList . find ( loadername ) ;
2018-04-22 12:06:48 +00:00
if ( mlit ! = MachineList . end ( ) ) {
2018-05-05 21:58:11 +00:00
auto mit = mlit . value ( ) . find ( serial ) ;
2018-04-22 12:06:48 +00:00
if ( mit ! = mlit . value ( ) . end ( ) ) {
return mit . value ( ) ;
}
2011-06-26 08:30:44 +00:00
}
2018-04-22 12:06:48 +00:00
return nullptr ;
}
2014-04-17 05:58:57 +00:00
2014-07-28 13:56:29 +00:00
2018-04-22 12:06:48 +00:00
Machine * Profile : : CreateMachine ( MachineInfo info , MachineID id )
{
Machine * m = nullptr ;
2014-07-28 13:56:29 +00:00
2018-05-05 21:58:11 +00:00
auto mlit = MachineList . find ( info . loadername ) ;
2014-10-01 04:16:00 +00:00
2018-04-22 12:06:48 +00:00
if ( mlit ! = MachineList . end ( ) ) {
2018-05-05 21:58:11 +00:00
auto mit = mlit . value ( ) . find ( info . serial ) ;
2018-04-22 12:06:48 +00:00
if ( mit ! = mlit . value ( ) . end ( ) ) {
mit . value ( ) - > setInfo ( info ) ; // update info
return mit . value ( ) ;
}
}
2014-07-28 13:56:29 +00:00
2018-04-22 12:06:48 +00:00
// Before we create, find any lost folder to get the old ID
if ( ( id = = 0 ) & & ( ( info . type = = MT_OXIMETER ) | | ( info . type = = MT_JOURNAL ) | | ( info . type = = MT_POSITION ) | | ( info . type = = MT_SLEEPSTAGE ) ) ) {
QString dataPath = Get ( " { " + STR_GEN_DataFolder + " }/ " ) ;
QDir dir ( dataPath ) ;
QStringList namefilter ( QString ( info . loadername + " _* " ) ) ;
QStringList files = dir . entryList ( namefilter , QDir : : Dirs ) ;
if ( files . size ( ) > 0 ) {
QString idstr = files [ 0 ] . section ( " _ " , - 1 ) ;
bool ok ;
id = idstr . toInt ( & ok , 16 ) ;
}
}
2014-07-28 13:56:29 +00:00
2018-04-22 12:06:48 +00:00
switch ( info . type ) {
case MT_CPAP :
2018-06-04 23:26:46 +00:00
m = new CPAP ( this , id ) ;
2018-04-22 12:06:48 +00:00
break ;
case MT_SLEEPSTAGE :
2018-06-04 23:26:46 +00:00
m = new SleepStage ( this , id ) ;
2018-04-22 12:06:48 +00:00
break ;
case MT_OXIMETER :
2018-06-04 23:26:46 +00:00
m = new Oximeter ( this , id ) ;
2018-04-22 12:06:48 +00:00
break ;
case MT_POSITION :
2018-06-04 23:26:46 +00:00
m = new PositionSensor ( this , id ) ;
2018-04-22 12:06:48 +00:00
break ;
case MT_JOURNAL :
2018-06-04 23:26:46 +00:00
m = new Machine ( this , id ) ;
2018-04-22 12:06:48 +00:00
m - > setType ( MT_JOURNAL ) ;
break ;
default :
2018-06-04 23:26:46 +00:00
m = new Machine ( this , id ) ;
2018-04-22 12:06:48 +00:00
break ;
}
m - > setInfo ( info ) ;
2018-06-07 01:53:20 +00:00
// qDebug() << "Reading" << info.loadername << "Machine Record" << (info.serial.isEmpty() ? m->hexid() : info.serial);
2018-04-22 12:06:48 +00:00
MachineList [ info . loadername ] [ info . serial ] = m ;
AddMachine ( m ) ;
return m ;
2011-06-26 08:30:44 +00:00
}
2018-04-22 12:06:48 +00:00
2014-04-17 05:58:57 +00:00
void Profile : : AddMachine ( Machine * m )
{
2011-07-24 16:34:53 +00:00
if ( ! m ) {
qWarning ( ) < < " Empty Machine in Profile::AddMachine() " ;
return ;
}
2018-04-22 12:06:48 +00:00
m_machlist . append ( m ) ;
2014-08-04 19:57:48 +00:00
}
2014-04-17 05:58:57 +00:00
void Profile : : DelMachine ( Machine * m )
{
2011-07-24 16:34:53 +00:00
if ( ! m ) {
qWarning ( ) < < " Empty Machine in Profile::AddMachine() " ;
return ;
}
2014-04-17 05:58:57 +00:00
2018-04-22 12:06:48 +00:00
removeMachine ( m ) ;
2014-08-04 19:57:48 +00:00
}
2011-06-26 08:30:44 +00:00
2014-08-20 17:17:13 +00:00
Day * Profile : : addDay ( QDate date )
2014-04-17 05:58:57 +00:00
{
2018-05-05 21:58:11 +00:00
auto dit = daylist . find ( date ) ;
2014-08-20 17:17:13 +00:00
if ( dit = = daylist . end ( ) ) {
dit = daylist . insert ( date , new Day ( ) ) ;
2012-01-08 03:29:22 +00:00
}
2014-08-20 17:17:13 +00:00
Day * day = dit . value ( ) ;
2014-09-17 02:34:50 +00:00
day - > setDate ( date ) ;
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
if ( is_first_day ) {
2014-04-17 05:58:57 +00:00
m_first = m_last = date ;
is_first_day = false ;
}
if ( m_first > date ) {
m_first = date ;
}
if ( m_last < date ) {
m_last = date ;
2011-06-26 08:30:44 +00:00
}
2014-08-20 17:17:13 +00:00
return day ;
2011-06-26 08:30:44 +00:00
}
2022-02-27 16:01:46 +00:00
// Get Day record if data available for date and device type,
2014-04-23 13:19:56 +00:00
// and has enabled session data, else return nullptr
2014-04-17 05:58:57 +00:00
Day * Profile : : GetGoodDay ( QDate date , MachineType type )
2011-12-28 12:36:40 +00:00
{
2014-04-11 22:22:23 +00:00
Day * day = GetDay ( date , type ) ;
if ( ! day )
2014-04-23 13:19:56 +00:00
return nullptr ;
2014-04-11 22:22:23 +00:00
2022-02-27 16:01:46 +00:00
// For a device match, find at least one enabled Session.
2018-05-05 21:58:11 +00:00
for ( auto & sess : day - > sessions ) {
2014-09-17 06:12:38 +00:00
if ( ( ( type = = MT_UNKNOWN ) | | ( sess - > type ( ) = = type ) ) & & sess - > enabled ( ) ) {
2014-09-11 14:23:08 +00:00
day - > OpenSummary ( ) ;
return day ;
}
}
// No enabled Sessions were found.
return nullptr ;
}
Day * Profile : : FindGoodDay ( QDate date , MachineType type )
{
Day * day = FindDay ( date , type ) ;
if ( ! day )
return nullptr ;
2022-02-27 16:01:46 +00:00
// For a device match, find at least one enabled Session.
2018-05-05 21:58:11 +00:00
for ( auto & sess : day - > sessions ) {
2014-09-17 06:12:38 +00:00
if ( ( ( type = = MT_UNKNOWN ) | | ( sess - > type ( ) = = type ) ) & & sess - > enabled ( ) ) {
2014-04-11 22:22:23 +00:00
return day ;
2014-09-11 14:23:08 +00:00
}
2011-12-28 12:36:40 +00:00
}
2014-04-17 05:58:57 +00:00
2014-04-11 22:22:23 +00:00
// No enabled Sessions were found.
2014-04-23 13:19:56 +00:00
return nullptr ;
2011-12-28 12:36:40 +00:00
}
2014-09-11 14:23:08 +00:00
2014-04-17 05:58:57 +00:00
Day * Profile : : GetDay ( QDate date , MachineType type )
2011-06-26 08:30:44 +00:00
{
2018-05-05 21:58:11 +00:00
auto di = daylist . find ( date ) ;
2014-08-20 17:17:13 +00:00
if ( di = = daylist . end ( ) ) return nullptr ;
2014-04-17 05:58:57 +00:00
2014-08-20 17:17:13 +00:00
Day * day = di . value ( ) ;
2014-04-22 07:16:24 +00:00
2014-09-11 14:23:08 +00:00
if ( type = = MT_UNKNOWN ) {
day - > OpenSummary ( ) ;
return day ; // just want the day record
}
if ( day - > machines . contains ( type ) ) {
day - > OpenSummary ( ) ;
return day ;
}
return nullptr ;
}
Day * Profile : : FindDay ( QDate date , MachineType type )
{
2018-05-05 21:58:11 +00:00
auto di = daylist . find ( date ) ;
2014-09-11 14:23:08 +00:00
if ( di = = daylist . end ( ) ) return nullptr ;
Day * day = di . value ( ) ;
if ( type = = MT_UNKNOWN ) {
return day ; // just want the day record
}
2014-04-22 07:16:24 +00:00
2014-09-11 14:23:08 +00:00
if ( day - > machines . contains ( type ) ) {
return day ;
}
2014-04-17 05:58:57 +00:00
2014-04-23 13:19:56 +00:00
return nullptr ;
2011-06-26 08:30:44 +00:00
}
2014-04-17 05:58:57 +00:00
MachineLoader * GetLoader ( QString name )
2011-06-26 08:30:44 +00:00
{
2018-05-05 21:58:11 +00:00
QList < MachineLoader * > loaders = GetLoaders ( ) ;
2014-04-17 05:58:57 +00:00
2018-05-05 21:58:11 +00:00
for ( auto & loader : loaders ) {
2014-07-28 13:56:29 +00:00
if ( loader - > loaderName ( ) = = name ) {
2014-04-22 07:16:24 +00:00
return loader ;
2011-06-26 08:30:44 +00:00
}
}
2014-04-17 05:58:57 +00:00
2014-04-23 13:19:56 +00:00
return nullptr ;
2011-06-26 08:30:44 +00:00
}
2022-02-27 16:01:46 +00:00
// Returns a QVector containing all device objects regisered of type t
2013-09-14 23:32:14 +00:00
QList < Machine * > Profile : : GetMachines ( MachineType t )
2011-06-26 08:30:44 +00:00
{
2011-12-08 04:10:35 +00:00
QList < Machine * > vec ;
2011-06-26 08:30:44 +00:00
2018-05-05 21:58:11 +00:00
for ( auto & mach : m_machlist ) {
if ( ! mach ) {
2018-04-22 12:06:48 +00:00
qWarning ( ) < < " Profile::GetMachines() m == nullptr " ;
2011-07-19 15:31:51 +00:00
continue ;
}
2014-04-17 05:58:57 +00:00
2018-05-05 21:58:11 +00:00
MachineType mt = mach - > type ( ) ;
2014-04-17 05:58:57 +00:00
if ( ( t = = MT_UNKNOWN ) | | ( mt = = t ) ) {
2018-05-05 21:58:11 +00:00
vec . push_back ( mach ) ;
2011-06-26 08:30:44 +00:00
}
}
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
return vec ;
}
2014-04-17 05:58:57 +00:00
Machine * Profile : : GetMachine ( MachineType t )
2011-06-26 08:30:44 +00:00
{
2014-04-17 05:58:57 +00:00
QList < Machine * > vec = GetMachines ( t ) ;
if ( vec . size ( ) = = 0 ) {
2014-04-23 13:19:56 +00:00
return nullptr ;
2014-04-17 05:58:57 +00:00
}
2022-02-27 16:01:46 +00:00
// Find most recently imported device
2020-09-12 16:53:36 +00:00
int idx = 0 ;
for ( int i = 1 ; i < vec . size ( ) ; i + + ) {
if ( vec [ i ] - > lastImported ( ) > vec [ idx ] - > lastImported ( ) )
idx = i ;
}
return vec [ idx ] ;
2011-06-26 08:30:44 +00:00
}
2014-07-25 07:53:48 +00:00
//bool Profile::trashMachine(Machine * mach)
//{
// QMap<QDate, QList<Day *> >::iterator it_end = daylist.end();
// QMap<QDate, QList<Day *> >::iterator it;
2019-03-25 23:36:41 +00:00
//
2014-07-25 07:53:48 +00:00
// QList<QDate> datelist;
// QList<Day *> days;
2019-03-25 23:36:41 +00:00
//
2014-07-25 07:53:48 +00:00
// for (it = daylist.begin(); it != it_end; ++it) {
// for (int i = 0; i< it.value().size(); ++i) {
// Day * day = it.value().at(i);
// if (day->machine() == mach) {
// days.push_back(day);
// datelist.push_back(it.key());
// }
// }
// }
2019-03-25 23:36:41 +00:00
//
2014-07-25 07:53:48 +00:00
// for (int i=0; i < datelist.size(); ++i) {
// Day * day = days.at(i);
// it = daylist.find(datelist.at(i));
// if (it != daylist.end()) {
// it.value().removeAll(day);
// if (it.value().size() == 0) {
// daylist.erase(it);
// }
// }
// mach->unlinkDay(days.at(i));
// }
2019-03-25 23:36:41 +00:00
//
2014-07-25 07:53:48 +00:00
//}
bool Profile : : unlinkDay ( Day * day )
2011-12-06 14:39:14 +00:00
{
2014-08-20 17:17:13 +00:00
// Find the key...
2018-05-05 21:58:11 +00:00
for ( auto it = daylist . begin ( ) , it_end = daylist . end ( ) ; it ! = it_end ; + + it ) {
2014-08-20 17:17:13 +00:00
if ( it . value ( ) = = day ) {
daylist . erase ( it ) ;
return true ;
2014-07-25 07:53:48 +00:00
}
}
2014-08-20 17:17:13 +00:00
return false ;
2011-12-06 14:39:14 +00:00
}
2014-04-23 13:19:56 +00:00
//Profile *profile=nullptr;
2011-06-26 08:30:44 +00:00
QString SHA1 ( QString pass )
{
return pass ;
}
2014-04-17 05:58:57 +00:00
namespace Profiles {
2011-06-26 08:30:44 +00:00
2014-04-17 05:58:57 +00:00
QMap < QString , Profile * > profiles ;
2011-06-26 08:30:44 +00:00
void Done ( )
{
2019-03-25 23:36:41 +00:00
p_pref - > Save ( ) ;
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
profiles . clear ( ) ;
delete p_pref ;
2018-04-22 12:06:48 +00:00
delete AppSetting ;
2013-01-18 20:18:25 +00:00
DestroyLoaders ( ) ;
2011-06-26 08:30:44 +00:00
}
Profile * Get ( QString name )
{
2018-06-05 22:08:12 +00:00
auto it = profiles . find ( name ) ;
if ( it ! = profiles . end ( ) ) {
return it . value ( ) ;
2014-04-17 05:58:57 +00:00
}
2011-06-26 08:30:44 +00:00
2014-04-23 13:19:56 +00:00
return nullptr ;
2011-06-26 08:30:44 +00:00
}
2014-07-28 13:56:29 +00:00
2019-05-03 20:17:32 +00:00
Profile * Create ( QString name , const QString * in_path )
2011-06-26 08:30:44 +00:00
{
2019-05-03 20:17:32 +00:00
QString path ;
if ( in_path = = nullptr ) {
path = p_pref - > Get ( " {home}/Profiles/ " ) + name ;
} else {
path = * in_path ;
}
2011-06-26 08:30:44 +00:00
QDir dir ( path ) ;
2014-04-17 05:58:57 +00:00
if ( ! dir . exists ( path ) ) {
2013-09-11 12:57:06 +00:00
dir . mkpath ( path ) ;
2014-04-17 05:58:57 +00:00
}
2011-06-26 08:30:44 +00:00
//path+="/"+name;
2014-07-12 11:05:17 +00:00
p_profile = new Profile ( path ) ;
profiles [ name ] = p_profile ;
p_profile - > user - > setUserName ( name ) ;
//p_profile->Set("Realname",realname);
//if (!password.isEmpty()) p_profile.user->setPassword(password);
2014-07-28 13:56:29 +00:00
p_profile - > Set ( STR_GEN_DataFolder , QString ( " {home}/Profiles/{ " ) + QString ( STR_UI_UserName ) + QString ( " } " ) ) ;
2011-06-26 08:30:44 +00:00
2018-06-05 22:08:12 +00:00
Machine * m = new Machine ( p_profile , 0 ) ;
2014-08-20 17:17:13 +00:00
m - > setType ( MT_JOURNAL ) ;
2019-02-24 00:58:25 +00:00
MachineInfo info ( MT_JOURNAL , 0 , STR_MACH_Journal , " OSCAR " , STR_MACH_Journal , QString ( ) , m - > hexid ( ) , QString ( ) , QDateTime : : currentDateTime ( ) , 0 ) ;
2014-07-28 13:56:29 +00:00
m - > setInfo ( info ) ;
2014-07-12 11:05:17 +00:00
p_profile - > AddMachine ( m ) ;
2011-06-26 08:30:44 +00:00
2014-07-12 11:05:17 +00:00
p_profile - > Save ( ) ;
2011-06-26 08:30:44 +00:00
2014-07-12 11:05:17 +00:00
return p_profile ;
2011-06-26 08:30:44 +00:00
}
Profile * Get ( )
{
2011-06-27 06:26:29 +00:00
// username lookup
//getUserName()
return profiles [ getUserName ( ) ] ; ;
2011-06-26 08:30:44 +00:00
}
2014-09-14 15:29:07 +00:00
void saveProfileList ( )
{
2019-03-25 23:36:41 +00:00
QString filename = p_pref - > Get ( " {home}/profiles.xml " ) ;
2014-09-14 15:29:07 +00:00
QDomDocument doc ( " profiles " ) ;
QDomElement root = doc . createElement ( " profiles " ) ;
doc . appendChild ( root ) ;
2019-02-24 00:58:25 +00:00
root . appendChild ( doc . createComment ( " This file is created during Profile Scan for cloud access convenience, it's not used by Desktop version of OSCAR. " ) ) ;
2018-04-22 12:06:48 +00:00
2018-05-05 21:58:11 +00:00
for ( auto it = profiles . begin ( ) ; it ! = profiles . end ( ) ; + + it ) {
2014-09-14 15:29:07 +00:00
QDomElement elem = doc . createElement ( " profile " ) ;
elem . setAttribute ( " name " , it . key ( ) ) ;
// Not technically nessesary..
elem . setAttribute ( " path " , QString ( " {home}/Profiles/%1/Profile.xml " ) . arg ( it . key ( ) ) ) ;
root . appendChild ( elem ) ;
}
QFile file ( filename ) ;
2020-08-09 17:33:04 +00:00
if ( ! file . open ( QFile : : WriteOnly ) ) {
qWarning ( ) < < " Could not open " < < filename < < " for writing, error code " < < file . error ( ) < < file . errorString ( ) ;
return ;
}
2014-09-14 15:29:07 +00:00
file . write ( doc . toByteArray ( ) ) ;
file . close ( ) ;
}
2011-06-26 08:30:44 +00:00
2018-04-22 12:06:48 +00:00
int CleanupProfile ( Profile * prof )
{
// Migrate old per Profile settings that should have been put in program main preferences.
QStringList migrateList ;
migrateList < < STR_IS_Multithreading < < STR_US_ShowPerformance < < STR_US_ShowDebug
< < STR_US_ScrollDampening < < STR_AS_CalendarVisible < < STR_IS_CacheSessions
< < STR_AS_LineCursorMode < < STR_AS_RightSidebarVisible < < STR_AS_DailyPanelWidth
< < STR_US_ShowPerformance < < STR_AS_GraphHeight < < STR_AS_GraphSnapshots
< < STR_AS_AntiAliasing < < STR_AS_LineThickness < < STR_AS_UsePixmapCaching
< < STR_AS_SquareWave < < STR_AS_RightPanelWidth < < STR_US_TooltipTimeout
< < STR_AS_Animations < < STR_AS_AllowYAxisScaling < < STR_AS_GraphTooltips
2022-11-17 01:46:30 +00:00
< < STR_CS_UserEventPieChart < < STR_AS_OverlayType
# ifndef REMOVE_FITNESS
< < STR_AS_OverviewLinechartMode
# endif
;
2018-04-22 12:06:48 +00:00
int cnt = 0 ;
2018-05-05 21:58:11 +00:00
for ( auto & prf : migrateList ) {
2018-04-22 12:06:48 +00:00
if ( prof - > contains ( prf ) ) {
qDebug ( ) < < " Migrating profile preference " < < prf ;
2019-03-25 23:36:41 +00:00
( * p_pref ) [ prf ] = ( * prof ) [ prf ] ;
2018-04-22 12:06:48 +00:00
prof - > Erase ( prf ) ;
cnt + + ;
}
}
if ( cnt > 0 ) {
qDebug ( ) < < " Migrated " < < cnt < < " preferences for profile " < < ( * prof ) [ STR_UI_UserName ] ;
prof - > Save ( ) ;
}
return cnt ;
}
2011-06-26 08:30:44 +00:00
/**
* @ brief Scan Profile directory loading user profiles
*/
void Scan ( )
{
2019-03-25 23:36:41 +00:00
QString path = p_pref - > Get ( " {home}/Profiles " ) ;
2011-06-26 08:30:44 +00:00
QDir dir ( path ) ;
2019-03-25 23:36:41 +00:00
profiles . clear ( ) ;
2011-06-26 08:30:44 +00:00
if ( ! dir . exists ( path ) ) {
return ;
}
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
if ( ! dir . isReadable ( ) ) {
2011-07-01 10:10:44 +00:00
qWarning ( ) < < " Can't open " < < path ;
2011-06-26 08:30:44 +00:00
return ;
}
dir . setFilter ( QDir : : Dirs | QDir : : NoDotAndDotDot ) ;
//dir.setSorting(QDir::Name);
2014-04-17 05:58:57 +00:00
QFileInfoList list = dir . entryInfoList ( ) ;
2011-06-26 08:30:44 +00:00
2018-04-22 12:06:48 +00:00
int cleanup = 0 ;
2011-07-17 07:03:26 +00:00
// Iterate through subdirectories and load profiles..
2018-05-05 21:58:11 +00:00
for ( auto & fi : list ) {
2014-04-17 05:58:57 +00:00
QString npath = fi . canonicalFilePath ( ) ;
2020-09-10 05:03:32 +00:00
QDir profilePath ( npath ) ;
if ( profilePath . isEmpty ( ) ) // skip any empty folders
continue ;
2014-04-17 05:58:57 +00:00
Profile * prof = new Profile ( npath ) ;
2014-09-14 15:29:07 +00:00
//prof->Open();
2014-07-02 03:22:09 +00:00
2014-04-17 05:58:57 +00:00
profiles [ fi . fileName ( ) ] = prof ;
2018-04-22 12:06:48 +00:00
// Migrate any old settings
cleanup + = CleanupProfile ( prof ) ;
2011-06-26 08:30:44 +00:00
}
2018-04-22 12:06:48 +00:00
if ( cleanup > 0 ) {
qDebug ( ) < < " Saving preferences after migration " ;
2019-03-25 23:36:41 +00:00
p_pref - > Save ( ) ;
2018-04-22 12:06:48 +00:00
}
2014-09-14 15:29:07 +00:00
// Update profiles.xml for mobile version
saveProfileList ( ) ;
2011-06-26 08:30:44 +00:00
}
2011-12-06 14:39:14 +00:00
2011-12-18 16:39:36 +00:00
} // namespace Profiles
2011-06-26 08:30:44 +00:00
2014-05-31 21:25:07 +00:00
2022-02-27 16:01:46 +00:00
// Returns a list of all days records matching device type between start and end date
2014-05-31 21:25:07 +00:00
QList < Day * > Profile : : getDays ( MachineType mt , QDate start , QDate end )
{
2014-08-20 17:17:13 +00:00
QList < Day * > list ;
2014-05-31 21:25:07 +00:00
if ( ! start . isValid ( ) ) {
2014-08-20 17:17:13 +00:00
return list ;
2014-05-31 21:25:07 +00:00
}
if ( ! end . isValid ( ) ) {
2014-08-20 17:17:13 +00:00
return list ;
2014-05-31 21:25:07 +00:00
}
QDate date = start ;
if ( date . isNull ( ) ) {
2014-08-20 17:17:13 +00:00
return list ;
2014-05-31 21:25:07 +00:00
}
2014-08-20 17:17:13 +00:00
QMap < QDate , Day * > : : iterator it ;
2014-05-31 21:25:07 +00:00
2014-08-20 17:17:13 +00:00
do {
it = daylist . find ( date ) ;
if ( it ! = daylist . end ( ) ) {
Day * day = it . value ( ) ;
if ( mt ! = MT_UNKNOWN ) {
if ( day - > hasEnabledSessions ( mt ) ) {
list . push_back ( day ) ;
}
} else {
if ( day - > hasEnabledSessions ( ) ) {
list . push_back ( day ) ;
}
2014-05-31 21:25:07 +00:00
}
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2014-08-20 17:17:13 +00:00
return list ;
2014-05-31 21:25:07 +00:00
}
2022-02-27 16:01:46 +00:00
// Counts number of days in range with data for specified device type
2011-12-23 10:52:31 +00:00
int Profile : : countDays ( MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
2012-01-01 14:14:18 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
if ( ! end . isValid ( ) ) {
2012-01-01 14:14:18 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
QDate date = start ;
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
2013-09-11 12:57:06 +00:00
2014-04-17 05:58:57 +00:00
int days = 0 ;
2011-12-23 10:52:31 +00:00
do {
2014-09-11 14:23:08 +00:00
Day * day = FindGoodDay ( date , mt ) ;
2014-04-17 05:58:57 +00:00
2011-12-23 10:52:31 +00:00
if ( day ) {
2014-08-20 17:17:13 +00:00
days + + ;
2011-12-23 10:52:31 +00:00
}
2014-04-17 05:58:57 +00:00
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2011-12-23 10:52:31 +00:00
return days ;
}
2014-05-06 09:11:31 +00:00
int Profile : : countCompliantDays ( MachineType mt , QDate start , QDate end )
{
EventDataType compliance = cpap - > complianceHours ( ) ;
if ( ! start . isValid ( ) ) {
return 0 ;
}
if ( ! end . isValid ( ) ) {
return 0 ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
return 0 ;
}
int days = 0 ;
do {
2014-09-30 08:40:12 +00:00
Day * day = FindGoodDay ( date , mt ) ;
2014-05-06 09:11:31 +00:00
if ( day ) {
2014-08-20 17:17:13 +00:00
if ( day - > hours ( mt ) > compliance ) { days + + ; }
2014-05-06 09:11:31 +00:00
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
return days ;
}
2019-05-27 18:22:38 +00:00
// Count number of events of type code in period
2011-12-22 13:22:40 +00:00
EventDataType Profile : : calcCount ( ChannelID code , MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
double val = 0 ;
2011-12-22 13:22:40 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-22 13:22:40 +00:00
if ( day ) {
2014-04-17 05:58:57 +00:00
val + = day - > count ( code ) ;
2011-12-22 13:22:40 +00:00
}
2014-04-17 05:58:57 +00:00
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2011-12-22 13:22:40 +00:00
return val ;
}
2014-04-22 07:16:24 +00:00
2011-12-22 13:22:40 +00:00
double Profile : : calcSum ( ChannelID code , MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
double val = 0 ;
2011-12-22 13:22:40 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-22 13:22:40 +00:00
if ( day ) {
2014-04-17 05:58:57 +00:00
val + = day - > sum ( code ) ;
2011-12-22 13:22:40 +00:00
}
2014-04-17 05:58:57 +00:00
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2011-12-22 13:22:40 +00:00
return val ;
}
2014-05-15 20:48:05 +00:00
2011-12-22 13:22:40 +00:00
EventDataType Profile : : calcHours ( MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
double val = 0 ;
2011-12-22 13:22:40 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-22 13:22:40 +00:00
if ( day ) {
2020-07-24 18:29:19 +00:00
val + = day - > hours ( mt ) ;
2011-12-22 13:22:40 +00:00
}
2014-04-17 05:58:57 +00:00
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2011-12-22 13:22:40 +00:00
return val ;
}
2014-05-15 20:48:05 +00:00
EventDataType Profile : : calcAboveThreshold ( ChannelID code , EventDataType threshold , MachineType mt ,
QDate start , QDate end )
{
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
return 0 ;
}
2014-05-15 21:48:53 +00:00
EventDataType val = 0 ;
2014-05-15 20:48:05 +00:00
do {
Day * day = GetGoodDay ( date , mt ) ;
if ( day ) {
val + = day - > timeAboveThreshold ( code , threshold ) ;
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
return val ;
}
EventDataType Profile : : calcBelowThreshold ( ChannelID code , EventDataType threshold , MachineType mt ,
QDate start , QDate end )
{
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
return 0 ;
}
2014-05-15 21:48:53 +00:00
EventDataType val = 0 ;
2014-05-15 20:48:05 +00:00
do {
Day * day = GetGoodDay ( date , mt ) ;
if ( day ) {
val + = day - > timeBelowThreshold ( code , threshold ) ;
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
return val ;
}
2014-10-11 03:07:29 +00:00
Day * Profile : : findSessionDay ( Session * session )
{
2018-05-05 21:58:11 +00:00
for ( auto it = p_profile - > daylist . begin ( ) , it_end = p_profile - > daylist . end ( ) ; it ! = it_end ; + + it ) {
2014-10-11 12:27:10 +00:00
Day * day = it . value ( ) ;
2018-05-05 21:58:11 +00:00
for ( auto & sess : day - > sessions ) {
if ( sess = = session ) {
2014-10-11 12:27:10 +00:00
return day ;
2014-10-11 03:07:29 +00:00
}
2014-10-11 12:27:10 +00:00
}
}
2014-10-11 03:07:29 +00:00
return nullptr ;
}
2014-05-15 20:48:05 +00:00
2011-12-22 13:22:40 +00:00
EventDataType Profile : : calcAvg ( ChannelID code , MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
double val = 0 ;
int cnt = 0 ;
2011-12-22 13:22:40 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-22 13:22:40 +00:00
if ( day ) {
2014-07-02 13:45:53 +00:00
if ( ! day - > summaryOnly ( ) | | day - > hasData ( code , ST_AVG ) ) {
val + = day - > sum ( code ) ;
2018-04-17 04:11:28 +00:00
cnt + = day - > count ( code ) ;
2014-07-02 13:45:53 +00:00
}
2011-12-22 13:22:40 +00:00
}
2014-04-17 05:58:57 +00:00
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
if ( ! cnt ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
return val / float ( cnt ) ;
2011-12-22 13:22:40 +00:00
}
2014-04-22 07:16:24 +00:00
2011-12-22 13:22:40 +00:00
EventDataType Profile : : calcWavg ( ChannelID code , MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
double val = 0 , tmp , tmph , hours = 0 ;
2011-12-22 13:22:40 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-22 13:22:40 +00:00
if ( day ) {
2014-07-02 13:45:53 +00:00
if ( ! day - > summaryOnly ( ) | | day - > hasData ( code , ST_WAVG ) ) {
tmph = day - > hours ( ) ;
tmp = day - > wavg ( code ) ;
val + = tmp * tmph ;
hours + = tmph ;
}
2014-04-17 05:58:57 +00:00
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
if ( ! hours ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
val = val / hours ;
2011-12-22 13:22:40 +00:00
return val ;
}
2014-04-22 07:16:24 +00:00
2011-12-23 10:52:31 +00:00
EventDataType Profile : : calcMin ( ChannelID code , MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
2013-09-11 12:57:06 +00:00
2014-07-02 13:58:59 +00:00
bool first = true ;
2011-12-23 10:52:31 +00:00
2014-07-02 13:58:59 +00:00
double min = 0 , tmp ;
2014-04-17 05:58:57 +00:00
2011-12-23 10:52:31 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-23 10:52:31 +00:00
if ( day ) {
2014-07-02 13:45:53 +00:00
if ( ! day - > summaryOnly ( ) | | day - > hasData ( code , ST_MIN ) ) {
tmp = day - > Min ( code ) ;
2014-04-17 05:58:57 +00:00
2014-07-02 13:58:59 +00:00
if ( first | | ( min > tmp ) ) {
2014-07-02 13:45:53 +00:00
min = tmp ;
2014-07-02 13:58:59 +00:00
first = false ;
2014-07-02 13:45:53 +00:00
}
2014-04-17 05:58:57 +00:00
}
2014-07-02 13:45:53 +00:00
2014-04-17 05:58:57 +00:00
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2014-07-02 13:58:59 +00:00
if ( first ) {
2014-04-17 05:58:57 +00:00
min = 0 ;
}
2011-12-23 10:52:31 +00:00
return min ;
}
EventDataType Profile : : calcMax ( ChannelID code , MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
2014-07-02 13:58:59 +00:00
bool first = true ;
double max = 0 , tmp ;
2011-12-23 10:52:31 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-23 10:52:31 +00:00
if ( day ) {
2014-07-02 13:45:53 +00:00
if ( ! day - > summaryOnly ( ) | | day - > hasData ( code , ST_MAX ) ) {
tmp = day - > Max ( code ) ;
2014-04-17 05:58:57 +00:00
2014-07-02 13:58:59 +00:00
if ( first | | ( max < tmp ) ) {
2014-07-02 13:45:53 +00:00
max = tmp ;
2014-07-02 13:58:59 +00:00
first = false ;
2014-07-02 13:45:53 +00:00
}
2014-04-17 05:58:57 +00:00
}
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2014-07-02 13:58:59 +00:00
if ( first ) {
2014-04-17 05:58:57 +00:00
max = 0 ;
}
2011-12-23 10:52:31 +00:00
return max ;
}
EventDataType Profile : : calcSettingsMin ( ChannelID code , MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
2014-07-02 13:58:59 +00:00
bool first = true ;
double min = 0 , tmp ;
2011-12-23 10:52:31 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-23 10:52:31 +00:00
if ( day ) {
2014-04-17 05:58:57 +00:00
tmp = day - > settings_min ( code ) ;
2014-07-02 13:58:59 +00:00
if ( first | | ( min > tmp ) ) {
2014-04-17 05:58:57 +00:00
min = tmp ;
2014-07-02 13:58:59 +00:00
first = false ;
2014-04-17 05:58:57 +00:00
}
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2014-07-02 13:58:59 +00:00
if ( first ) {
2014-04-17 05:58:57 +00:00
min = 0 ;
}
2011-12-23 10:52:31 +00:00
return min ;
}
2014-05-15 20:48:05 +00:00
2011-12-23 10:52:31 +00:00
EventDataType Profile : : calcSettingsMax ( ChannelID code , MachineType mt , QDate start , QDate end )
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
2014-07-02 13:58:59 +00:00
bool first = true ;
double max = 0 , tmp ;
2011-12-23 10:52:31 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-23 10:52:31 +00:00
if ( day ) {
2014-04-17 05:58:57 +00:00
tmp = day - > settings_max ( code ) ;
2014-07-02 13:58:59 +00:00
if ( first | | ( max < tmp ) ) {
2014-04-17 05:58:57 +00:00
max = tmp ;
2014-07-02 13:58:59 +00:00
first = false ;
2014-04-17 05:58:57 +00:00
}
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2014-07-02 13:58:59 +00:00
if ( first ) {
2014-04-17 05:58:57 +00:00
max = 0 ;
}
2011-12-23 10:52:31 +00:00
return max ;
}
2011-12-24 01:22:41 +00:00
struct CountSummary {
2014-04-17 05:58:57 +00:00
CountSummary ( EventStoreType v ) : val ( v ) , count ( 0 ) , time ( 0 ) { }
2011-12-24 01:22:41 +00:00
EventStoreType val ;
EventStoreType count ;
quint32 time ;
} ;
2014-04-17 05:58:57 +00:00
EventDataType Profile : : calcPercentile ( ChannelID code , EventDataType percent , MachineType mt ,
QDate start , QDate end )
2011-12-22 13:22:40 +00:00
{
2014-04-17 05:58:57 +00:00
if ( ! start . isValid ( ) ) {
start = LastGoodDay ( mt ) ;
}
if ( ! end . isValid ( ) ) {
end = LastGoodDay ( mt ) ;
}
QDate date = start ;
2011-12-22 13:22:40 +00:00
2014-04-17 05:58:57 +00:00
if ( date . isNull ( ) ) {
2013-09-11 12:57:06 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
2011-12-22 13:22:40 +00:00
2012-02-01 14:53:31 +00:00
QMap < EventDataType , qint64 > wmap ;
2013-10-13 10:13:14 +00:00
QMap < EventDataType , qint64 > : : iterator wmi ;
2011-12-24 01:22:41 +00:00
2014-04-17 05:58:57 +00:00
QHash < ChannelID , QHash < EventStoreType , EventStoreType > > : : iterator vsi ;
QHash < ChannelID , QHash < EventStoreType , quint32 > > : : iterator tsi ;
2012-01-05 12:12:42 +00:00
EventDataType gain ;
//bool setgain=false;
2012-02-01 14:53:31 +00:00
EventDataType value ;
int weight ;
2012-01-01 14:14:18 +00:00
2014-04-17 05:58:57 +00:00
qint64 SN = 0 ;
2012-02-01 14:53:31 +00:00
bool timeweight ;
2020-09-09 02:54:56 +00:00
bool summaryOnly = true ;
2011-12-22 13:22:40 +00:00
do {
2014-04-17 05:58:57 +00:00
Day * day = GetGoodDay ( date , mt ) ;
2011-12-23 10:52:31 +00:00
if ( day ) {
2014-07-02 13:45:53 +00:00
if ( day - > summaryOnly ( ) ) {
2020-09-09 02:54:56 +00:00
date = date . addDays ( 1 ) ;
continue ;
2014-07-02 13:45:53 +00:00
}
2018-05-05 21:58:11 +00:00
2020-09-09 02:54:56 +00:00
summaryOnly = false ;
2018-05-05 21:58:11 +00:00
// why was this nested like this???
//for (int i = 0; i < day->size(); i++) {
2020-09-09 02:54:56 +00:00
for ( auto & sess : day - > sessions ) {
if ( ! sess - > enabled ( ) ) {
continue ;
}
2014-04-17 05:58:57 +00:00
2020-09-09 02:54:56 +00:00
gain = sess - > m_gain [ code ] ;
2014-04-17 05:58:57 +00:00
2020-09-09 02:54:56 +00:00
if ( ! gain ) { gain = 1 ; }
2011-12-28 12:03:48 +00:00
2020-09-09 02:54:56 +00:00
vsi = sess - > m_valuesummary . find ( code ) ;
2011-12-24 01:22:41 +00:00
2020-09-09 02:54:56 +00:00
if ( vsi = = sess - > m_valuesummary . end ( ) ) { continue ; }
2014-04-17 05:58:57 +00:00
2020-09-09 02:54:56 +00:00
tsi = sess - > m_timesummary . find ( code ) ;
timeweight = ( tsi ! = sess - > m_timesummary . end ( ) ) ;
2014-04-17 05:58:57 +00:00
2020-09-09 02:54:56 +00:00
QHash < EventStoreType , EventStoreType > & vsum = vsi . value ( ) ;
QHash < EventStoreType , quint32 > & tsum = tsi . value ( ) ;
2012-02-01 14:53:31 +00:00
2020-09-09 02:54:56 +00:00
if ( timeweight ) {
for ( auto k = tsum . begin ( ) , tsumend = tsum . end ( ) ; k ! = tsumend ; k + + ) {
weight = k . value ( ) ;
value = EventDataType ( k . key ( ) ) * gain ;
2014-04-17 05:58:57 +00:00
2020-09-09 02:54:56 +00:00
SN + = weight ;
wmi = wmap . find ( value ) ;
2014-04-17 05:58:57 +00:00
2020-09-09 02:54:56 +00:00
if ( wmi = = wmap . end ( ) ) {
wmap [ value ] = weight ;
} else {
wmi . value ( ) + = weight ;
2012-02-01 14:53:31 +00:00
}
2020-09-09 02:54:56 +00:00
}
} else {
for ( auto k = vsum . begin ( ) , vsumend = vsum . end ( ) ; k ! = vsumend ; k + + ) {
weight = k . value ( ) ;
value = EventDataType ( k . key ( ) ) * gain ;
SN + = weight ;
wmi = wmap . find ( value ) ;
if ( wmi = = wmap . end ( ) ) {
wmap [ value ] = weight ;
} else {
wmi . value ( ) + = weight ;
2011-12-24 01:22:41 +00:00
}
}
2011-12-23 10:52:31 +00:00
}
2020-09-09 02:54:56 +00:00
}
// }
2011-12-23 10:52:31 +00:00
}
2014-04-17 05:58:57 +00:00
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
2014-07-02 13:45:53 +00:00
if ( summaryOnly ) {
// abort percentile calculation, there is not enough data
return 0 ;
}
2012-01-05 04:37:22 +00:00
QVector < ValueCount > valcnt ;
// Build sorted list of value/counts
2014-04-17 05:58:57 +00:00
for ( wmi = wmap . begin ( ) ; wmi ! = wmap . end ( ) ; wmi + + ) {
2012-01-05 04:37:22 +00:00
ValueCount vc ;
2014-04-17 05:58:57 +00:00
vc . value = wmi . key ( ) ;
vc . count = wmi . value ( ) ;
vc . p = 0 ;
2012-01-05 04:37:22 +00:00
valcnt . push_back ( vc ) ;
}
2014-04-17 05:58:57 +00:00
2012-01-05 04:37:22 +00:00
// sort by weight, then value
2018-06-07 22:09:05 +00:00
std : : sort ( valcnt . begin ( ) , valcnt . end ( ) ) ;
2011-12-24 01:22:41 +00:00
2012-01-05 04:37:22 +00:00
//double SN=100.0/double(N); // 100% / overall sum
2014-04-17 05:58:57 +00:00
double p = 100.0 * percent ;
2011-12-24 05:55:44 +00:00
2014-04-17 05:58:57 +00:00
double nth = double ( SN ) * percent ; // index of the position in the unweighted set would be
double nthi = floor ( nth ) ;
2011-12-24 05:55:44 +00:00
2014-04-17 05:58:57 +00:00
qint64 sum1 = 0 , sum2 = 0 ;
qint64 w1 , w2 = 0 ;
double v1 = 0 , v2 = 0 ;
2011-12-24 05:55:44 +00:00
2014-04-17 05:58:57 +00:00
int N = valcnt . size ( ) ;
int k = 0 ;
2011-12-24 05:55:44 +00:00
2014-04-17 05:58:57 +00:00
for ( k = 0 ; k < N ; k + + ) {
v1 = valcnt [ k ] . value ;
w1 = valcnt [ k ] . count ;
sum1 + = w1 ;
2012-01-05 04:37:22 +00:00
if ( sum1 > nthi ) {
return v1 ;
}
2014-04-17 05:58:57 +00:00
if ( sum1 = = nthi ) {
2012-01-05 04:37:22 +00:00
break ; // boundary condition
}
}
2014-04-17 05:58:57 +00:00
if ( k > = N ) {
2012-01-05 04:37:22 +00:00
return v1 ;
2014-04-17 05:58:57 +00:00
}
2012-01-05 04:37:22 +00:00
2014-04-17 05:58:57 +00:00
v2 = valcnt [ k + 1 ] . value ;
w2 = valcnt [ k + 1 ] . count ;
sum2 = sum1 + w2 ;
2012-01-05 04:37:22 +00:00
// value lies between v1 and v2
2014-04-17 05:58:57 +00:00
double px = 100.0 / double ( SN ) ; // Percentile represented by one full value
2012-01-05 04:37:22 +00:00
// calculate percentile ranks
2014-04-17 05:58:57 +00:00
double p1 = px * ( double ( sum1 ) - ( double ( w1 ) / 2.0 ) ) ;
double p2 = px * ( double ( sum2 ) - ( double ( w2 ) / 2.0 ) ) ;
2012-01-05 04:37:22 +00:00
// calculate linear interpolation
2014-04-17 05:58:57 +00:00
double v = v1 + ( ( p - p1 ) / ( p2 - p1 ) ) * ( v2 - v1 ) ;
2011-12-24 05:55:44 +00:00
2012-01-05 04:37:22 +00:00
// p1.....p.............p2
// 37 55 70
return v ;
2011-12-22 13:22:40 +00:00
}
2022-02-27 16:01:46 +00:00
// Lookup first day record of the specified device type, or return the first day overall if MT_UNKNOWN
2011-12-22 13:22:40 +00:00
QDate Profile : : FirstDay ( MachineType mt )
{
2014-04-17 05:58:57 +00:00
if ( ( mt = = MT_UNKNOWN ) | | ( ! m_last . isValid ( ) ) | | ( ! m_first . isValid ( ) ) ) {
2011-12-22 13:22:40 +00:00
return m_first ;
2014-04-17 05:58:57 +00:00
}
QDate d = m_first ;
2011-12-22 13:22:40 +00:00
do {
2014-09-11 14:23:08 +00:00
if ( FindDay ( d , mt ) ! = nullptr ) {
2013-09-11 12:57:06 +00:00
return d ;
2014-04-17 05:58:57 +00:00
}
d = d . addDays ( 1 ) ;
} while ( d < = m_last ) ;
2011-12-22 13:22:40 +00:00
return m_last ;
}
2013-09-14 23:32:14 +00:00
2022-02-27 16:01:46 +00:00
// Lookup last day record of the specified device type, or return the last day overall if MT_UNKNOWN
2011-12-22 13:22:40 +00:00
QDate Profile : : LastDay ( MachineType mt )
{
2014-04-17 05:58:57 +00:00
if ( ( mt = = MT_UNKNOWN ) | | ( ! m_last . isValid ( ) ) | | ( ! m_first . isValid ( ) ) ) {
2011-12-22 13:22:40 +00:00
return m_last ;
2014-04-17 05:58:57 +00:00
}
QDate d = m_last ;
2011-12-22 13:22:40 +00:00
do {
2014-09-11 14:23:08 +00:00
if ( FindDay ( d , mt ) ! = nullptr ) {
2013-09-11 12:57:06 +00:00
return d ;
2014-04-17 05:58:57 +00:00
}
d = d . addDays ( - 1 ) ;
} while ( d > = m_first ) ;
2011-12-22 13:22:40 +00:00
return m_first ;
}
2011-12-28 12:36:40 +00:00
QDate Profile : : FirstGoodDay ( MachineType mt )
{
2014-05-31 21:25:07 +00:00
if ( mt = = MT_UNKNOWN ) {
2011-12-28 12:36:40 +00:00
return FirstDay ( ) ;
2014-04-17 05:58:57 +00:00
}
2011-12-28 12:36:40 +00:00
2014-04-17 05:58:57 +00:00
QDate d = FirstDay ( mt ) ;
QDate l = LastDay ( mt ) ;
2014-05-31 21:25:07 +00:00
// No data will return invalid date records
2014-04-17 05:58:57 +00:00
if ( ! d . isValid ( ) | | ! l . isValid ( ) ) {
2012-01-01 14:14:18 +00:00
return QDate ( ) ;
2014-04-17 05:58:57 +00:00
}
2011-12-28 12:36:40 +00:00
do {
2014-09-11 14:23:08 +00:00
if ( FindGoodDay ( d , mt ) ! = nullptr ) {
2013-09-11 12:57:06 +00:00
return d ;
2014-04-17 05:58:57 +00:00
}
d = d . addDays ( 1 ) ;
} while ( d < = l ) ;
2011-12-28 12:36:40 +00:00
return l ; //m_last;
}
QDate Profile : : LastGoodDay ( MachineType mt )
{
2014-05-18 17:53:09 +00:00
if ( mt = = MT_UNKNOWN ) {
2011-12-28 12:36:40 +00:00
return FirstDay ( ) ;
2014-04-17 05:58:57 +00:00
}
QDate d = LastDay ( mt ) ;
QDate f = FirstDay ( mt ) ;
if ( ! ( d . isValid ( ) & & f . isValid ( ) ) ) {
2013-09-11 12:57:06 +00:00
return QDate ( ) ;
2014-04-17 05:58:57 +00:00
}
2011-12-28 12:36:40 +00:00
do {
2014-09-11 14:23:08 +00:00
if ( FindGoodDay ( d , mt ) ! = nullptr ) {
2013-09-11 12:57:06 +00:00
return d ;
2014-04-17 05:58:57 +00:00
}
d = d . addDays ( - 1 ) ;
} while ( d > = f ) ;
2014-05-18 17:53:09 +00:00
return f ;
2011-12-28 12:36:40 +00:00
}
2014-09-11 14:23:08 +00:00
bool Profile : : channelAvailable ( ChannelID code )
{
2018-05-05 21:58:11 +00:00
for ( auto & mach : m_machlist ) {
2014-09-11 14:23:08 +00:00
if ( mach - > hasChannel ( code ) )
return true ;
}
return false ;
}
2012-01-25 22:34:21 +00:00
bool Profile : : hasChannel ( ChannelID code )
{
2014-04-17 05:58:57 +00:00
QDate d = LastDay ( ) ;
QDate f = FirstDay ( ) ;
if ( ! ( d . isValid ( ) & & f . isValid ( ) ) ) {
2013-09-11 12:57:06 +00:00
return false ;
2014-04-17 05:58:57 +00:00
}
2014-08-20 17:17:13 +00:00
QMap < QDate , Day * > : : iterator dit ;
2014-04-22 07:16:24 +00:00
2014-04-17 05:58:57 +00:00
bool found = false ;
2012-01-25 22:34:21 +00:00
do {
2014-04-17 05:58:57 +00:00
dit = daylist . find ( d ) ;
if ( dit ! = daylist . end ( ) ) {
2014-08-20 17:17:13 +00:00
Day * day = dit . value ( ) ;
2014-04-22 07:16:24 +00:00
2014-09-11 14:23:08 +00:00
2014-08-20 17:17:13 +00:00
if ( day - > channelHasData ( code ) ) {
found = true ;
break ;
2012-01-25 22:34:21 +00:00
}
}
2014-04-17 05:58:57 +00:00
d = d . addDays ( - 1 ) ;
} while ( d > = f ) ;
2012-01-25 22:34:21 +00:00
return found ;
}
2018-04-22 12:06:48 +00:00
const quint16 chandata_version = 1 ;
void Profile : : saveChannels ( )
{
2018-04-22 14:22:18 +00:00
// First save the XML version for Mobile versions
schema : : channel . Save ( Get ( " {DataFolder}/ " ) + " channels.xml " ) ;
2018-04-22 12:06:48 +00:00
QString filename = Get ( " {DataFolder}/ " ) + " channels.dat " ;
QFile f ( filename ) ;
qDebug ( ) < < " Saving Channel States " ;
f . open ( QFile : : WriteOnly ) ;
QDataStream out ( & f ) ;
out . setVersion ( QDataStream : : Qt_4_6 ) ;
out . setByteOrder ( QDataStream : : LittleEndian ) ;
out < < ( quint32 ) magic ;
out < < ( quint16 ) chandata_version ;
2018-06-09 00:59:16 +00:00
QSettings settings ;
2018-04-22 12:06:48 +00:00
( * p_profile ) [ STR_PREF_Language ] = settings . value ( LangSetting , " " ) . toString ( ) ;
quint16 size = schema : : channel . channels . size ( ) ;
out < < size ;
2018-05-05 21:58:11 +00:00
for ( auto it = schema : : channel . channels . begin ( ) , it_end = schema : : channel . channels . end ( ) ; it ! = it_end ; + + it ) {
2018-04-22 12:06:48 +00:00
schema : : Channel * chan = it . value ( ) ;
out < < it . key ( ) ;
out < < chan - > code ( ) ;
out < < chan - > enabled ( ) ;
out < < chan - > defaultColor ( ) ;
out < < chan - > fullname ( ) ;
out < < chan - > label ( ) ;
out < < chan - > description ( ) ;
out < < chan - > lowerThreshold ( ) ;
out < < chan - > lowerThresholdColor ( ) ;
out < < chan - > upperThreshold ( ) ;
out < < chan - > upperThresholdColor ( ) ;
out < < chan - > showInOverview ( ) ;
}
f . close ( ) ;
}
void Profile : : loadChannels ( )
{
bool changing_language = false ;
QString filename = Get ( " {DataFolder}/ " ) + " channels.dat " ;
QFile f ( filename ) ;
if ( ! f . open ( QFile : : ReadOnly ) ) {
return ;
}
qDebug ( ) < < " Loading channel.dat States " ;
QDataStream in ( & f ) ;
in . setVersion ( QDataStream : : Qt_4_6 ) ;
in . setByteOrder ( QDataStream : : LittleEndian ) ;
quint32 mag ;
in > > mag ;
if ( magic ! = mag ) {
qDebug ( ) < < " LoadChannels: Faulty data " ;
return ;
}
quint16 version ;
in > > version ;
2018-06-09 00:59:16 +00:00
QSettings settings ;
2018-04-22 12:06:48 +00:00
QString language = Get ( STR_PREF_Language ) ;
if ( settings . value ( LangSetting , " " ) . toString ( ) ! = language ) {
qDebug ( ) < < " Language change detected, resetting default channel names " ;
changing_language = true ;
}
quint16 size ;
in > > size ;
QString name ;
ChannelID code ;
bool enabled ;
QColor color ;
EventDataType lowerThreshold ;
QColor lowerThresholdColor ;
EventDataType upperThreshold ;
QColor upperThresholdColor ;
QString fullname ;
QString label ;
QString description ;
bool showOverview = false ;
for ( int i = 0 ; i < size ; i + + ) {
in > > code ;
schema : : Channel * chan = & schema : : channel [ code ] ;
in > > name ;
if ( chan - > code ( ) ! = name ) {
qDebug ( ) < < " Looking up channel " < < name < < " by name, as it's ChannedID must have changed " ;
chan = & schema : : channel [ name ] ;
}
in > > enabled ;
in > > color ;
in > > fullname ;
in > > label ;
in > > description ;
in > > lowerThreshold ;
in > > lowerThresholdColor ;
in > > upperThreshold ;
in > > upperThresholdColor ;
if ( version > = 1 ) {
in > > showOverview ;
}
if ( chan - > isNull ( ) ) {
qDebug ( ) < < " loadChannels has no idea about channel " < < name ;
2023-01-26 12:48:25 +00:00
if ( in . atEnd ( ) ) break ;
2018-04-22 12:06:48 +00:00
continue ;
}
chan - > setEnabled ( enabled ) ;
chan - > setDefaultColor ( color ) ;
// Don't import channel descriptions if event renaming is turned off. (helps pick up new translations)
if ( changing_language ) {
// Nothing
} else {
chan - > setFullname ( fullname ) ;
chan - > setLabel ( label ) ;
chan - > setDescription ( description ) ;
}
chan - > setLowerThreshold ( lowerThreshold ) ;
chan - > setLowerThresholdColor ( lowerThresholdColor ) ;
chan - > setUpperThreshold ( upperThreshold ) ;
chan - > setUpperThresholdColor ( upperThresholdColor ) ;
chan - > setShowInOverview ( showOverview ) ;
2023-01-26 12:48:25 +00:00
if ( in . atEnd ( ) ) break ;
2018-04-22 12:06:48 +00:00
}
f . close ( ) ;
2023-01-27 01:05:18 +00:00
resetOxiChannelPref ( ) ;
2018-04-22 12:06:48 +00:00
}
2023-01-26 12:48:25 +00:00
2023-01-27 01:05:18 +00:00
void Profile : : resetOxiChannelPref ( ) {
2023-01-26 12:48:25 +00:00
schema : : channel [ OXI_Pulse ] . setLowerThreshold ( oxi - > flagPulseBelow ( ) ) ;
schema : : channel [ OXI_Pulse ] . setUpperThreshold ( oxi - > flagPulseAbove ( ) ) ;
schema : : channel [ OXI_SPO2 ] . setLowerThreshold ( oxi - > oxiDesaturationThreshold ( ) ) ;
} ;