2014-04-09 21:01:57 +00:00
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* vim : set ts = 8 sts = 4 et sw = 4 tw = 99 :
*
* SleepLib Profiles Implementation
*
* Copyright ( c ) 2011 - 2014 Mark Watkins < jedimark @ users . sourceforge . net >
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file COPYING in the main directory of the 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>
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"
# include "machine_loader.h"
2011-12-11 09:45:28 +00:00
# include <QApplication>
# include "mainwindow.h"
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 ) ,
m_opened ( 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 ;
2011-06-26 08:30:44 +00:00
machlist . clear ( ) ;
2011-12-21 11:09:50 +00:00
2014-04-23 13:19:56 +00:00
doctor = nullptr ;
user = nullptr ;
cpap = nullptr ;
oxi = nullptr ;
appearance = nullptr ;
session = nullptr ;
general = nullptr ;
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 ( )
{
2014-07-12 10:52:14 +00:00
QString lockfile = p_path + " /lockfile " ;
QFile file ( lockfile ) ;
file . remove ( ) ;
2014-04-25 05:28:10 +00:00
if ( m_opened ) {
delete user ;
delete doctor ;
delete cpap ;
delete oxi ;
delete appearance ;
delete session ;
delete general ;
2014-05-06 18:03:13 +00:00
for ( QHash < MachineID , Machine * > : : iterator it = machlist . begin ( ) ; it ! = machlist . end ( ) ; it + + ) {
2014-04-25 05:28:10 +00:00
delete it . value ( ) ;
}
m_opened = false ;
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 ) {
return Preferences : : Save ( filename ) ;
} 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 ;
}
2012-01-12 22:46:06 +00:00
bool Profile : : Open ( QString filename )
{
2014-04-25 05:28:10 +00:00
if ( filename . isEmpty ( ) ) {
filename = p_filename ;
}
if ( m_opened ) {
qDebug ( ) < < " Profile " < < filename < < " all ready open " ;
return true ;
}
2014-04-17 05:58:57 +00:00
bool b = Preferences : : Open ( filename ) ;
2014-04-25 05:28:10 +00:00
2014-07-12 10:52:14 +00:00
QString lockfile = p_path + " /lockfile " ;
QFile file ( lockfile ) ;
file . open ( QFile : : WriteOnly ) ;
QByteArray ba ;
ba . append ( QHostInfo : : localHostName ( ) ) ;
file . write ( ba ) ;
file . close ( ) ;
2014-04-25 05:28:10 +00:00
m_opened = true ;
2014-04-17 05:58:57 +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 ) ;
2012-01-12 22:46:06 +00:00
return b ;
}
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 ;
foreach ( const QString & exec , execs ) {
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 ( ) ;
foreach ( const QString & p , path ( ) ) {
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)
void showInGraphicalShell ( const QString & pathIn )
{
QWidget * parent = NULL ;
// Mac, Windows support folder or file.
# if defined(Q_OS_WIN)
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)
Q_UNUSED ( parent )
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
// 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 " ) .
arg ( FullVersionString ) .
arg ( m - > properties [ STR_PROP_Brand ] ) . arg ( m - > properties [ STR_PROP_Model ] ) . arg ( m - > properties [ STR_PROP_Serial ] )
+ " </font><br/><br/> " ;
2014-04-17 05:58:57 +00:00
2014-07-02 03:22:09 +00:00
bool backups = false ;
if ( p_profile - > session - > backupCardData ( ) & & m - > properties . contains ( STR_PROP_BackupPath ) ) {
2014-07-03 01:59:50 +00:00
QString bpath = Get ( m - > properties [ STR_PROP_BackupPath ] ) ;
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 ) {
mainwin - > importCPAP ( Get ( m - > properties [ STR_PROP_BackupPath ] ) , QObject : : tr ( " Rebuilding from %1 Backup " ) . arg ( m - > properties [ STR_PROP_Brand ] ) ) ;
} 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..
if ( m - > GetClass ( ) ! = STR_MACH_Intellipap ) {
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 ,
QObject : : tr ( " SleepyHead will now start the import wizard so you can reinstall your %1 data. " ) . arg ( m - > properties [ STR_PROP_Brand ] )
, 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 ;
}
2011-06-26 08:30:44 +00:00
void Profile : : LoadMachineData ( )
{
2014-04-17 05:58:57 +00:00
QHash < MachineID , QMap < QDate , QHash < ChannelID , EventDataType > > > cache ;
2011-12-11 09:45:28 +00:00
2014-04-17 05:58:57 +00:00
for ( QHash < MachineID , Machine * > : : iterator i = machlist . begin ( ) ; i ! = machlist . end ( ) ; i + + ) {
Machine * m = i . value ( ) ;
MachineLoader * loader = GetLoader ( m - > GetClass ( ) ) ;
2011-06-26 08:30:44 +00:00
if ( loader ) {
2014-04-17 05:58:57 +00:00
long v = loader - > Version ( ) ;
long cv = 0 ;
if ( m - > properties . find ( STR_PROP_DataVersion ) = = m - > properties . end ( ) ) {
m - > properties [ STR_PROP_DataVersion ] = " 0 " ;
2011-06-26 08:30:44 +00:00
}
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
bool ok ;
2014-04-17 05:58:57 +00:00
cv = m - > properties [ STR_PROP_DataVersion ] . toLong ( & ok ) ;
if ( ! ok | | cv < v ) {
2011-07-29 22:55:24 +00:00
DataFormatError ( m ) ;
// It may exit above and not return here..
QString s ;
2014-04-17 05:58:57 +00:00
s . sprintf ( " %li " , v ) ;
m - > properties [ STR_PROP_DataVersion ] = s ; // Dont need to nag again if they are too lazy.
2011-07-29 22:55:24 +00:00
} else {
try {
m - > Load ( ) ;
2014-04-17 05:58:57 +00:00
} catch ( OldDBVersion e ) {
2014-05-08 04:46:23 +00:00
Q_UNUSED ( e )
2011-07-29 22:55:24 +00:00
DataFormatError ( m ) ;
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 {
m - > Load ( ) ;
}
}
}
/**
* @ brief Machine XML section in profile .
* @ param root
*/
2014-04-17 05:58:57 +00:00
void Profile : : ExtraLoad ( QDomElement & root )
2011-06-26 08:30:44 +00:00
{
2014-04-17 05:58:57 +00:00
if ( root . tagName ( ) ! = " Machines " ) {
2011-07-24 16:34:53 +00:00
qDebug ( ) < < " No Machines Tag in Profiles.xml " ;
return ;
2011-06-26 08:30:44 +00:00
}
2014-04-17 05:58:57 +00:00
QDomElement elem = root . firstChildElement ( ) ;
2011-07-24 16:34:53 +00:00
while ( ! elem . isNull ( ) ) {
2014-04-17 05:58:57 +00:00
QString pKey = elem . tagName ( ) ;
2011-07-24 16:34:53 +00:00
2014-04-17 05:58:57 +00:00
if ( pKey ! = " Machine " ) {
2011-07-19 15:31:51 +00:00
qWarning ( ) < < " Profile::ExtraLoad() pKey!= \" Machine \" " ;
2014-04-17 05:58:57 +00:00
elem = elem . nextSiblingElement ( ) ;
2011-07-19 15:31:51 +00:00
continue ;
}
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
int m_id ;
2011-07-24 16:34:53 +00:00
bool ok ;
2014-04-17 05:58:57 +00:00
m_id = elem . attribute ( " id " , " " ) . toInt ( & ok ) ;
2011-06-26 08:30:44 +00:00
int mt ;
2014-04-17 05:58:57 +00:00
mt = elem . attribute ( " type " , " " ) . toInt ( & ok ) ;
MachineType m_type = ( MachineType ) mt ;
2011-07-24 16:34:53 +00:00
2014-04-17 05:58:57 +00:00
QString m_class = elem . attribute ( " class " , " " ) ;
2011-07-24 16:34:53 +00:00
//MachineLoader *ml=GetLoader(m_class);
2011-06-26 08:30:44 +00:00
Machine * m ;
2014-04-17 05:58:57 +00:00
2011-07-24 16:34:53 +00:00
//if (ml) {
// ml->CreateMachine
//}
2014-04-17 05:58:57 +00:00
if ( m_type = = MT_CPAP ) {
2014-07-11 12:09:38 +00:00
m = new CPAP ( m_id ) ;
2014-04-17 05:58:57 +00:00
} else if ( m_type = = MT_OXIMETER ) {
2014-07-11 12:09:38 +00:00
m = new Oximeter ( m_id ) ;
2014-04-17 05:58:57 +00:00
} else if ( m_type = = MT_SLEEPSTAGE ) {
2014-07-11 12:09:38 +00:00
m = new SleepStage ( m_id ) ;
2014-04-17 05:58:57 +00:00
} else if ( m_type = = MT_POSITION ) {
2014-07-11 12:09:38 +00:00
m = new PositionSensor ( m_id ) ;
2014-04-17 05:58:57 +00:00
} else {
2014-07-11 12:09:38 +00:00
m = new Machine ( m_id ) ;
2011-06-26 08:30:44 +00:00
m - > SetType ( m_type ) ;
}
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
m - > SetClass ( m_class ) ;
AddMachine ( m ) ;
2014-04-17 05:58:57 +00:00
QDomElement e = elem . firstChildElement ( ) ;
for ( ; ! e . isNull ( ) ; e = e . nextSiblingElement ( ) ) {
QString pKey = e . tagName ( ) ;
m - > properties [ pKey ] = e . text ( ) ;
2011-06-26 08:30:44 +00:00
}
2014-04-17 05:58:57 +00:00
elem = elem . nextSiblingElement ( ) ;
2011-06-26 08:30:44 +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 ;
}
2014-04-17 05:58:57 +00:00
machlist [ m - > id ( ) ] = m ;
2011-06-26 08:30:44 +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
2011-07-31 20:24:43 +00:00
machlist . erase ( machlist . find ( m - > id ( ) ) ) ;
2011-06-26 08:30:44 +00:00
} ;
2011-07-24 16:34:53 +00:00
// Potential Memory Leak Here..
2014-04-17 05:58:57 +00:00
QDomElement Profile : : ExtraSave ( QDomDocument & doc )
{
QDomElement mach = doc . createElement ( " Machines " ) ;
for ( QHash < MachineID , Machine * > : : iterator i = machlist . begin ( ) ; i ! = machlist . end ( ) ; i + + ) {
QDomElement me = doc . createElement ( " Machine " ) ;
Machine * m = i . value ( ) ;
me . setAttribute ( " id " , ( int ) m - > id ( ) ) ;
me . setAttribute ( " type " , ( int ) m - > GetType ( ) ) ;
me . setAttribute ( " class " , m - > GetClass ( ) ) ;
if ( ! m - > properties . contains ( STR_PROP_Path ) ) { m - > properties [ STR_PROP_Path ] = " {DataFolder}/ " + m - > GetClass ( ) + " _ " + m - > hexid ( ) ; }
for ( QHash < QString , QString > : : iterator j = i . value ( ) - > properties . begin ( ) ;
j ! = i . value ( ) - > properties . end ( ) ; j + + ) {
QDomElement mp = doc . createElement ( j . key ( ) ) ;
2011-07-31 20:24:43 +00:00
mp . appendChild ( doc . createTextNode ( j . value ( ) ) ) ;
2011-07-24 16:34:53 +00:00
//mp->LinkEndChild(new QDomText(j->second.toLatin1()));
me . appendChild ( mp ) ;
2011-06-26 08:30:44 +00:00
}
2014-04-17 05:58:57 +00:00
2011-07-24 16:34:53 +00:00
mach . appendChild ( me ) ;
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 mach ;
}
2014-04-17 05:58:57 +00:00
void Profile : : AddDay ( QDate date , Day * day , MachineType mt )
{
2011-06-26 08:30:44 +00:00
//date+=wxTimeSpan::Day();
2012-01-08 03:29:22 +00:00
if ( ! day ) {
qDebug ( ) < < " Profile::AddDay called with null day object " ;
return ;
}
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
}
// Check for any other machines of same type.. Throw an exception if one already exists.
2014-04-17 05:58:57 +00:00
QList < Day * > & dl = daylist [ date ] ;
for ( QList < Day * > : : iterator a = dl . begin ( ) ; a ! = dl . end ( ) ; a + + ) {
if ( ( * a ) - > machine - > GetType ( ) = = mt ) {
2012-04-05 03:14:02 +00:00
// disabled this because two machines isn't all that bad
2014-04-23 13:19:56 +00:00
// if (QMessageBox::question(nullptr,"Different Machine Detected","This data comes from another machine to what's usually imported, and has overlapping data.\nThis new data will override any older data from the old machine. Are you sure you want to do this?",QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) {
2012-04-05 03:14:02 +00:00
// throw OneTypePerDay();
// }
2011-12-24 05:55:44 +00:00
daylist [ date ] . erase ( a ) ;
break ;
2011-06-26 08:30:44 +00:00
}
}
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
daylist [ date ] . push_back ( day ) ;
}
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
// Just return the day if not matching for a machine.
if ( type = = MT_UNKNOWN )
return day ;
// For a machine match, find at least one enabled Session.
Q_ASSERT ( day - > machine_type ( ) = = type ) ;
for ( int i = 0 ; i < day - > size ( ) ; + + i ) {
if ( ( * day ) [ i ] - > enabled ( ) )
return day ;
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-04-17 05:58:57 +00:00
Day * Profile : : GetDay ( QDate date , MachineType type )
2011-06-26 08:30:44 +00:00
{
2014-04-11 22:22:23 +00:00
if ( ! daylist . contains ( date ) )
2014-04-23 13:19:56 +00:00
return nullptr ;
2014-04-17 05:58:57 +00:00
2014-04-11 22:22:23 +00:00
QList < Day * > list ( daylist . value ( date ) ) ;
2014-04-22 07:16:24 +00:00
QList < Day * > : : iterator it = list . begin ( ) ;
QList < Day * > : : iterator list_end = list . end ( ) ;
for ( ; it ! = list_end ; + + it ) {
Day * day = ( * it ) ;
2014-04-23 13:19:56 +00:00
Q_ASSERT ( day ! = nullptr ) ;
2014-04-17 05:58:57 +00:00
2014-04-11 22:22:23 +00:00
// Just return the day if not matching for a machine.
if ( day - > machine_type ( ) = = type | | type = = MT_UNKNOWN ) {
return day ;
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-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
2014-04-22 07:16:24 +00:00
Q_FOREACH ( MachineLoader * 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
{
2014-05-25 07:07:08 +00:00
QList < MachineLoader * > loaders = GetLoaders ( MT_CPAP ) ;
2014-04-17 05:58:57 +00:00
2014-04-22 07:16:24 +00:00
Q_FOREACH ( MachineLoader * loader , loaders ) {
if ( loader - > ClassName ( ) = = name ) {
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 ;
2014-04-17 05:58:57 +00:00
QHash < MachineID , Machine * > : : iterator i ;
2014-04-22 07:16:24 +00:00
QHash < MachineID , Machine * > : : iterator machlist_end = machlist . end ( ) ;
2011-06-26 08:30:44 +00:00
2014-04-22 07:16:24 +00:00
for ( i = machlist . begin ( ) ; i ! = machlist_end ; i + + ) {
2011-07-31 20:24:43 +00:00
if ( ! i . value ( ) ) {
2014-04-23 13:19:56 +00:00
qWarning ( ) < < " Profile::GetMachines() i->second == nullptr " ;
2011-07-19 15:31:51 +00:00
continue ;
}
2014-04-17 05:58:57 +00:00
MachineType mt = i . value ( ) - > GetType ( ) ;
if ( ( t = = MT_UNKNOWN ) | | ( mt = = t ) ) {
2011-07-31 20:24:43 +00:00
vec . push_back ( i . value ( ) ) ;
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-04-17 05:58:57 +00:00
void Profile : : RemoveSession ( Session * sess )
2011-12-06 14:39:14 +00:00
{
2014-04-17 05:58:57 +00:00
QMap < QDate , QList < Day * > > : : iterator di ;
2014-04-22 07:16:24 +00:00
QMap < QDate , QList < Day * > > : : iterator daylist_end = daylist . end ( ) ;
2014-05-28 17:45:42 +00:00
Machine * mach = sess - > machine ( ) ;
2011-12-06 14:39:14 +00:00
2014-04-22 07:16:24 +00:00
for ( di = daylist . begin ( ) ; di ! = daylist_end ; di + + ) {
2014-04-17 05:58:57 +00:00
for ( int d = 0 ; d < di . value ( ) . size ( ) ; d + + ) {
Day * day = di . value ( ) [ d ] ;
2011-12-06 14:39:14 +00:00
2014-04-17 05:58:57 +00:00
int i = day - > getSessions ( ) . indexOf ( sess ) ;
if ( i > = 0 ) {
for ( ; i < day - > getSessions ( ) . size ( ) - 1 ; i + + ) {
day - > getSessions ( ) [ i ] = day - > getSessions ( ) [ i + 1 ] ;
2011-12-06 14:39:14 +00:00
}
2014-04-17 05:58:57 +00:00
2011-12-06 14:39:14 +00:00
day - > getSessions ( ) . pop_back ( ) ;
2014-04-17 05:58:57 +00:00
qint64 first = 0 , last = 0 ;
for ( int i = 0 ; i < day - > getSessions ( ) . size ( ) ; i + + ) {
Session & sess = * day - > getSessions ( ) [ i ] ;
if ( ! first | | first > sess . first ( ) ) {
first = sess . first ( ) ;
}
if ( ! last | | last < sess . last ( ) ) {
last = sess . last ( ) ;
}
2011-12-06 14:39:14 +00:00
}
2014-04-17 05:58:57 +00:00
2014-05-18 17:06:58 +00:00
if ( day - > size ( ) = = 0 ) {
di . value ( ) . removeAll ( day ) ;
2014-05-28 17:45:42 +00:00
mach - > day . erase ( mach - > day . find ( di . key ( ) ) ) ;
2014-05-18 17:06:58 +00:00
delete day ;
}
2012-01-19 15:18:34 +00:00
// day->setFirst(first);
// day->setLast(last);
2011-12-06 14:39:14 +00:00
return ;
}
2014-05-18 17:06:58 +00:00
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 ( ) ;
LAYOUT . Save ( ) ;
2014-04-17 05:58:57 +00:00
2014-05-26 19:29:01 +00:00
p_profile - > Save ( ) ;
delete p_profile ;
2014-04-17 05:58:57 +00:00
2011-06-26 08:30:44 +00:00
profiles . clear ( ) ;
delete p_pref ;
delete p_layout ;
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
}
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 ) ;
p_profile - > Open ( ) ;
profiles [ name ] = p_profile ;
p_profile - > user - > setUserName ( name ) ;
//p_profile->Set("Realname",realname);
//if (!password.isEmpty()) p_profile.user->setPassword(password);
p_profile - > Set ( STR_GEN_DataFolder , QString ( " {home}/Profiles/{ " ) + QString ( STR_UI_UserName ) +
2014-04-17 05:58:57 +00:00
QString ( " } " ) ) ;
2011-06-26 08:30:44 +00:00
2014-07-11 12:09:38 +00:00
Machine * m = new Machine ( 0 ) ;
2011-06-26 08:30:44 +00:00
m - > SetClass ( " Journal " ) ;
2014-04-17 05:58:57 +00:00
m - > properties [ STR_PROP_Brand ] = " Journal " ;
m - > properties [ STR_PROP_Model ] = " Journal Data Machine Object " ;
m - > properties [ STR_PROP_Serial ] = m - > hexid ( ) ;
m - > properties [ STR_PROP_Path ] = " {DataFolder}/ " + m - > GetClass ( ) + " _ " + m - > hexid ( ) ;
2011-06-26 08:30:44 +00:00
m - > SetType ( MT_JOURNAL ) ;
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
}
/**
* @ 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
2011-07-17 07:03:26 +00:00
// Iterate through subdirectories and load profiles..
2014-04-17 05:58:57 +00:00
for ( int i = 0 ; i < list . size ( ) ; i + + ) {
QFileInfo fi = list . at ( i ) ;
QString npath = fi . canonicalFilePath ( ) ;
Profile * prof = new Profile ( npath ) ;
2014-07-02 03:22:09 +00:00
2014-04-17 05:58:57 +00:00
profiles [ fi . fileName ( ) ] = prof ;
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 )
{
QList < Day * > daylist ;
if ( ! start . isValid ( ) ) {
return daylist ;
}
if ( ! end . isValid ( ) ) {
return daylist ;
}
QDate date = start ;
if ( date . isNull ( ) ) {
return daylist ;
}
do {
Day * day = GetGoodDay ( date , mt ) ;
if ( day ) {
if ( ( mt = = MT_UNKNOWN ) | | ( day - > machine - > GetType ( ) = = mt ) ) {
daylist . push_back ( day ) ;
}
}
date = date . addDays ( 1 ) ;
} while ( date < = end ) ;
return daylist ;
}
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-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
if ( ( mt = = MT_UNKNOWN ) | | ( day - > machine - > GetType ( ) = = mt ) ) { 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 {
Day * day = GetGoodDay ( date , mt ) ;
if ( day ) {
if ( ( day - > machine - > GetType ( ) = = mt ) & & ( day - > hours ( ) > compliance ) ) { days + + ; }
}
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 ;
}
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 ) ;
cnt + + ;
}
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 ;
}
2014-04-17 05:58:57 +00:00
for ( int i = 0 ; i < day - > size ( ) ; i + + ) {
for ( QList < Session * > : : iterator s = day - > begin ( ) ; s ! = day - > end ( ) ; s + + ) {
if ( ! ( * s ) - > enabled ( ) ) {
2012-01-01 14:14:18 +00:00
continue ;
2014-04-17 05:58:57 +00:00
}
Session * sess = * s ;
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 ) {
2014-04-17 05:58:57 +00:00
for ( QHash < EventStoreType , quint32 > : : iterator k = tsum . begin ( ) ; k ! = tsum . end ( ) ; k + + ) {
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 {
2014-04-17 05:58:57 +00:00
for ( QHash < EventStoreType , EventStoreType > : : iterator k = vsum . begin ( ) ; k ! = vsum . end ( ) ; k + + ) {
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
}
}
}
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-04-23 13:19:56 +00:00
if ( GetDay ( 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-04-23 13:19:56 +00:00
if ( GetDay ( 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-04-23 13:19:56 +00:00
if ( GetGoodDay ( 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-04-23 13:19:56 +00:00
if ( GetGoodDay ( 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
}
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
}
QMap < QDate , QList < Day * > > : : iterator dit ;
2014-04-22 07:16:24 +00:00
QList < Day * > : : iterator di ;
QList < Day * > : : iterator di_end ;
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-04-22 07:16:24 +00:00
di = dit . value ( ) . begin ( ) ;
di_end = dit . value ( ) . end ( ) ;
for ( ; di ! = di_end ; + + di ) {
Day * day = ( * di ) ;
2014-04-17 05:58:57 +00:00
2012-01-25 22:34:21 +00:00
if ( day - > channelHasData ( code ) ) {
2014-04-17 05:58:57 +00:00
found = true ;
2012-01-25 22:34:21 +00:00
break ;
}
}
}
2014-04-17 05:58:57 +00:00
if ( found ) {
2013-09-11 12:57:06 +00:00
break ;
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 ;
}