2018-04-22 12:06:48 +00:00
/* SleepLib Profiles Implementation
2014-04-09 21:01:57 +00:00
*
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
* License . See the file COPYING in the main directory of the Linux
* distribution for more details . */
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"
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
Profile : : Profile ( QString path )
2014-04-25 05:28:10 +00:00
: is_first_day ( true ) ,
2014-09-14 15:29:07 +00:00
m_opened ( false ) ,
m_machopened ( false )
2014-04-17 05:58:57 +00:00
{
p_name = STR_GEN_Profile ;
if ( path . isEmpty ( ) ) {
p_path = GetAppRoot ( ) ;
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 ( ) ;
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} " ) ) ;
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 ) ;
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 ( )
{
2018-04-22 12:06:48 +00:00
removeLock ( ) ;
2014-04-25 05:28:10 +00:00
2018-04-22 12:06:48 +00:00
delete user ;
delete doctor ;
delete cpap ;
delete oxi ;
delete appearance ;
delete session ;
delete general ;
2014-07-26 12:46:11 +00:00
2018-04-22 12:06:48 +00:00
// delete machine 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
}
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 ;
}
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 " ;
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 ( )
{
if ( m_machopened )
return true ;
2014-09-14 15:29:07 +00:00
QString filename = p_path + " machines.xml " ;
QFile file ( filename ) ;
if ( ! file . open ( QFile : : ReadOnly ) ) {
qWarning ( ) < < " Could not open " < < QDir : : toNativeSeparators ( filename ) ;
return false ;
}
QDomDocument doc ( " machines.xml " ) ;
if ( ! doc . setContent ( & file ) ) {
qWarning ( ) < < " Invalid XML Content in " < < QDir : : toNativeSeparators ( filename ) ;
return false ;
}
file . close ( ) ;
QDomElement root = doc . firstChild ( ) . toElement ( ) ;
if ( root . tagName ( ) . toLower ( ) ! = " machines " ) {
//qDebug() << "No Machines Tag in Profiles.xml";
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 ) ;
} 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 ;
2018-04-22 12:06:48 +00:00
// Create Machine needs a profile passed to it..
m = CreateMachine ( info , m_id ) ;
2014-09-14 15:29:07 +00:00
if ( m ) m - > properties = prop ;
elem = elem . nextSiblingElement ( ) ;
}
m_machopened = true ;
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 ) ;
2018-04-22 12:06:48 +00:00
for ( QHash < QString , QString > : : iterator j = m - > properties . begin ( ) ; j ! = m - > 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 ) ;
mach . appendChild ( me ) ;
}
doc . appendChild ( mach ) ;
QString filename = p_path + " machines.xml " ;
QFile file ( filename ) ;
if ( ! file . open ( QFile : : WriteOnly ) ) {
return false ;
}
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 ( ) ) ;
}
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
2014-07-02 03:22:09 +00:00
msg = " <font size=+1> " + QObject : : tr ( " SleepyHead (%1) needs to upgrade its database for %2 %3 %4 " ) .
2016-03-06 02:50:22 +00:00
arg ( VersionString ) .
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 ) {
msg = msg + QObject : : tr ( " <b>SleepyHead maintains a backup of your devices data card that it uses for this purpose.</b> " ) + " <br/><br/> " ;
msg = msg + QObject : : tr ( " <i>Your old machine data should be regenerated provided this backup feature has not been disabled in preferences during a previous data import.</i> " ) + " <br/><br/> " ;
backups = true ;
} else {
2014-07-03 01:59:50 +00:00
msg = msg + " <font size=+1> " + STR_MessageBox_Warning + " :</font> " + QObject : : tr ( " SleepyHead does not yet have any automatic card backups stored for this device. " ) + " <br/><br/> " ;
2014-07-02 03:22:09 +00:00
msg = msg + QObject : : tr ( " This means you will need to import this machine data again afterwards from your own backups or data card. " ) + " <br/><br/> " ;
}
2014-04-17 05:58:57 +00:00
2014-07-02 03:22:09 +00:00
msg + = " <font size=+1> " + QObject : : tr ( " Important: " ) + " </font> " + QObject : : tr ( " Once you upgrade, you <font size=+1>can not</font> use this profile with the previous version anymore. " ) + " <br/><br/> " +
QObject : : tr ( " If you are concerned, click No to exit, and backup your profile manually, before starting SleepyHead again. " ) + " <br/><br/> " ;
msg = msg + " <font size=+1> " + QObject : : tr ( " Are you ready to upgrade, so you can run the new version of SleepyHead? " ) + " </font> " ;
QMessageBox * question = new QMessageBox ( QMessageBox : : Warning , QObject : : tr ( " Machine Database Changes " ) , msg , QMessageBox : : Yes | QMessageBox : : No ) ;
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 ,
QObject : : tr ( " Sorry, the purge operation failed, which means this version of SleepyHead can't start. " ) + " \n \n " +
QObject : : tr ( " The machine 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 ) {
2014-07-28 13:56:29 +00:00
mainwin - > importCPAP ( ImportPath ( m - > getBackupPath ( ) , lookupLoader ( m ) ) , QObject : : tr ( " Rebuilding from %1 Backup " ) . arg ( m - > brand ( ) ) ) ;
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 ) {
2014-07-03 01:59:50 +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 SleepyHead needs to do so, it can rebuild from these? " ) ,
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 ,
2014-07-28 13:56:29 +00:00
QObject : : tr ( " SleepyHead 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 ,
QObject : : tr ( " SleepyHead will now exit, then (attempt to) launch your computers file manager so you can manually back your profile up: " ) + " \n \n " +
QDir : : toNativeSeparators ( Get ( p_preferences [ STR_GEN_DataFolder ] . toString ( ) ) ) + " \n \n " +
QObject : : tr ( " Use your file manager to make a copy of your profile directory, then afterwards, restart Sleepyhead and complete the upgrade process. " )
, 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-04-25 10:34:23 +00:00
if ( ! m_machopened ) {
qCritical ( ) < < " Profile::UnloadMachineData() called with m_machopened==false " ;
return ;
}
2018-05-05 21:58:11 +00:00
for ( auto & day : daylist ) {
delete day ;
2018-04-22 12:06:48 +00:00
}
daylist . clear ( ) ;
2018-05-05 21:58:11 +00:00
for ( auto & mach : m_machlist ) {
mach - > sessionlist . clear ( ) ;
mach - > day . clear ( ) ;
2018-04-22 12:06:48 +00:00
}
removeLock ( ) ;
}
2011-06-26 08:30:44 +00:00
void Profile : : LoadMachineData ( )
{
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 ( ) ) {
DataFormatError ( mach ) ;
2011-07-29 22:55:24 +00:00
} else {
try {
2018-05-05 21:58:11 +00:00
mach - > Load ( ) ;
2016-01-04 06:32:37 +00:00
} catch ( OldDBVersion & e ) {
2014-05-08 04:46:23 +00:00
Q_UNUSED ( e )
2018-05-05 21:58:11 +00:00
DataFormatError ( mach ) ;
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-05-05 21:58:11 +00:00
mach - > Load ( ) ;
2011-06-26 08:30:44 +00:00
}
}
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 :
m = new CPAP ( id ) ;
break ;
case MT_SLEEPSTAGE :
m = new SleepStage ( id ) ;
break ;
case MT_OXIMETER :
m = new Oximeter ( id ) ;
break ;
case MT_POSITION :
m = new PositionSensor ( id ) ;
break ;
case MT_JOURNAL :
m = new Machine ( id ) ;
m - > setType ( MT_JOURNAL ) ;
break ;
default :
m = new Machine ( id ) ;
break ;
}
m - > setInfo ( info ) ;
qDebug ( ) < < " Added " < < info . loadername < < " Machine Record " < < ( info . serial . isEmpty ( ) ? m - > hexid ( ) : info . serial ) ;
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
}
2014-04-11 22:22:23 +00:00
// Get Day record if data available for date and machine 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
// For a machine 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 ;
// For a machine 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
}
2011-07-15 13:30:41 +00:00
int Profile : : Import ( QString path )
2011-06-26 08:30:44 +00:00
{
2014-04-17 05:58:57 +00:00
int c = 0 ;
2011-07-01 10:10:44 +00:00
qDebug ( ) < < " Importing " < < path ;
2014-04-17 05:58:57 +00:00
path = path . replace ( " \\ " , " / " ) ;
if ( path . endsWith ( " / " ) ) {
2013-09-11 12:57:06 +00:00
path . chop ( 1 ) ;
2014-04-17 05:58:57 +00:00
}
2014-05-25 07:07:08 +00:00
QList < MachineLoader * > loaders = GetLoaders ( MT_CPAP ) ;
2011-11-30 06:01:38 +00:00
2018-05-05 21:58:11 +00:00
for ( auto & loader : loaders ) {
2014-07-11 12:09:38 +00:00
if ( c + = loader - > Open ( path ) ) {
2013-09-11 12:57:06 +00:00
break ;
2014-04-17 05:58:57 +00:00
}
2011-06-26 08:30:44 +00:00
}
2014-04-17 05:58:57 +00:00
2011-07-15 13:30:41 +00:00
return c ;
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
}
2011-07-31 20:24:43 +00:00
// Returns a QVector containing all machine 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
}
2011-06-26 08:30:44 +00:00
return vec [ 0 ] ;
}
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;
// QList<QDate> datelist;
// QList<Day *> days;
// 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());
// }
// }
// }
// 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));
// }
//}
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 ( )
{
2011-10-05 07:41:56 +00:00
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 )
{
2014-04-17 05:58:57 +00:00
if ( profiles . find ( name ) ! = profiles . end ( ) ) {
2011-06-26 08:30:44 +00:00
return profiles [ name ] ;
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
2011-10-01 12:54:20 +00:00
Profile * Create ( QString name )
2011-06-26 08:30:44 +00:00
{
2014-04-17 05:58:57 +00:00
QString path = PREF . Get ( " {home}/Profiles/ " ) + name ;
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
2014-07-11 12:09:38 +00:00
Machine * m = new Machine ( 0 ) ;
2014-08-20 17:17:13 +00:00
m - > setType ( MT_JOURNAL ) ;
2014-08-03 13:00:13 +00:00
MachineInfo info ( MT_JOURNAL , 0 , STR_MACH_Journal , " SleepyHead " , 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 ( )
{
QString filename = PREF . Get ( " {home}/profiles.xml " ) ;
QDomDocument doc ( " profiles " ) ;
QDomElement root = doc . createElement ( " profiles " ) ;
doc . appendChild ( root ) ;
2018-04-22 12:06:48 +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 SleepyHead. " ) ) ;
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 ) ;
file . open ( QFile : : WriteOnly ) ;
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
< < STR_CS_UserEventPieChart < < STR_AS_OverlayType < < STR_AS_OverviewLinechartMode ;
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 ;
PREF [ prf ] = ( * prof ) [ prf ] ;
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 ( )
{
2014-04-17 05:58:57 +00:00
QString path = PREF . Get ( " {home}/Profiles " ) ;
2011-06-26 08:30:44 +00:00
QDir dir ( path ) ;
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 ( ) ;
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 " ;
PREF . Save ( ) ;
}
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
// Returns a list of all days records matching machine type between start and end date
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
}
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 ;
}
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 ) {
2014-04-17 05:58:57 +00:00
val + = day - > hours ( ) ;
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 ;
2014-07-02 13:45:53 +00:00
bool summaryOnly = false ;
2014-04-17 05:58:57 +00:00
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 ( ) ) {
summaryOnly = true ;
break ;
}
2018-05-05 21:58:11 +00:00
// why was this nested like this???
//for (int i = 0; i < day->size(); i++) {
for ( auto & sess : day - > sessions ) {
if ( ! sess - > enabled ( ) ) {
2012-01-01 14:14:18 +00:00
continue ;
2014-04-17 05:58:57 +00:00
}
gain = sess - > m_gain [ code ] ;
if ( ! gain ) { gain = 1 ; }
2011-12-28 12:03:48 +00:00
2014-04-17 05:58:57 +00:00
vsi = sess - > m_valuesummary . find ( code ) ;
2011-12-24 01:22:41 +00:00
2014-04-17 05:58:57 +00:00
if ( vsi = = sess - > m_valuesummary . end ( ) ) { continue ; }
tsi = sess - > m_timesummary . find ( code ) ;
timeweight = ( tsi ! = sess - > m_timesummary . end ( ) ) ;
QHash < EventStoreType , EventStoreType > & vsum = vsi . value ( ) ;
QHash < EventStoreType , quint32 > & tsum = tsi . value ( ) ;
2012-02-01 14:53:31 +00:00
if ( timeweight ) {
2018-05-05 21:58:11 +00:00
for ( auto k = tsum . begin ( ) , tsumend = tsum . end ( ) ; k ! = tsumend ; k + + ) {
2014-04-17 05:58:57 +00:00
weight = k . value ( ) ;
value = EventDataType ( k . key ( ) ) * gain ;
SN + = weight ;
wmi = wmap . find ( value ) ;
if ( wmi = = wmap . end ( ) ) {
wmap [ value ] = weight ;
2013-10-13 10:13:14 +00:00
} else {
2014-04-17 05:58:57 +00:00
wmi . value ( ) + = weight ;
2012-02-01 14:53:31 +00:00
}
}
} else {
2018-05-05 21:58:11 +00:00
for ( auto k = vsum . begin ( ) , vsumend = vsum . end ( ) ; k ! = vsumend ; k + + ) {
2014-04-17 05:58:57 +00:00
weight = k . value ( ) ;
value = EventDataType ( k . key ( ) ) * gain ;
SN + = weight ;
wmi = wmap . find ( value ) ;
if ( wmi = = wmap . end ( ) ) {
wmap [ value ] = weight ;
2013-10-13 10:13:14 +00:00
} else {
2014-04-17 05:58:57 +00:00
wmi . value ( ) + = weight ;
2012-02-01 14:53:31 +00:00
}
2011-12-24 01:22:41 +00:00
}
}
2011-12-23 10:52:31 +00:00
}
2018-05-05 21:58:11 +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
qSort ( valcnt ) ;
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
}
2013-09-14 23:32:14 +00:00
// Lookup first day record of the specified machine 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
// Lookup last day record of the specified machine type, or return the first 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 ;
QSettings settings ( getDeveloperName ( ) , getAppName ( ) ) ;
( * 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 ;
QSettings settings ( getDeveloperName ( ) , getAppName ( ) ) ;
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 ;
if ( in . atEnd ( ) ) return ;
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 ) ;
if ( in . atEnd ( ) ) return ;
}
f . close ( ) ;
}