2011-06-26 08:30:44 +00:00
/*
SleepLib Profiles Implementation
Author : Mark Watkins < jedimark64 @ users . sourceforge . net >
License : GPL
*/
# include <QString>
# include <QDateTime>
# include <QDir>
# include <QMessageBox>
2011-07-01 10:10:44 +00:00
# include <QDebug>
2011-06-26 08:30:44 +00:00
# include "preferences.h"
# include "profiles.h"
# include "machine.h"
# include "machine_loader.h"
Preferences * p_pref ;
Preferences * p_layout ;
2011-10-05 07:41:56 +00:00
Profile * p_profile ;
2011-06-26 08:30:44 +00:00
Profile : : Profile ( )
: Preferences ( ) , is_first_day ( true )
{
p_name = " Profile " ;
2011-10-05 07:41:56 +00:00
p_path = PREF . Get ( " {home}/Profiles " ) ;
2011-06-26 08:30:44 +00:00
machlist . clear ( ) ;
//m_first=m_last=
}
Profile : : Profile ( QString path )
: Preferences ( ) , is_first_day ( true )
{
const QString xmlext = " .xml " ;
p_name = " Profile " ;
if ( path . isEmpty ( ) ) p_path = GetAppRoot ( ) ;
else p_path = path ;
( * this ) [ " DataFolder " ] = p_path ;
if ( ! p_path . endsWith ( " / " ) ) p_path + = " / " ;
p_filename = p_path + p_name + xmlext ;
machlist . clear ( ) ;
//m_first=m_last=NULL;
}
Profile : : ~ Profile ( )
{
2011-07-31 20:24:43 +00:00
for ( QHash < MachineID , Machine * > : : iterator i = machlist . begin ( ) ; i ! = machlist . end ( ) ; i + + ) {
delete i . value ( ) ;
2011-06-26 08:30:44 +00:00
}
}
2011-07-29 22:55:24 +00:00
void Profile : : DataFormatError ( Machine * m )
{
QString msg = " Software changes have been made that require the reimporting of the following machines data: \n \n " ;
msg = msg + m - > properties [ " Brand " ] + " " + m - > properties [ " Model " ] + " " + m - > properties [ " Serial " ] ;
msg = msg + " \n \n This is still only alpha software and these changes are sometimes necessary. \n \n " ;
msg = msg + " I can automatically purge this data for you, or you can cancel now and continue to run in a previous version. \n \n " ;
msg = msg + " Would you like me to purge this data this for you so you can run the new version? " ;
if ( QMessageBox : : warning ( NULL , " Machine Database Changes " , msg , QMessageBox : : Yes | QMessageBox : : Cancel , QMessageBox : : Yes ) = = QMessageBox : : Yes ) {
if ( ! m - > Purge ( 3478216 ) ) { // Do not copy this line without thinking.. You will be eaten by a Grue if you do
2011-10-05 07:41:56 +00:00
QMessageBox : : critical ( NULL , " Purge Failed " , " Sorry, I could not purge this data, which means this version of SleepyHead can't start.. SleepyHead's Data folder needs to be removed manually \n \n This folder currently resides at the following location: \n " + PREF [ " DataFolder " ] . toString ( ) , QMessageBox : : Ok ) ;
2011-07-29 22:55:24 +00:00
exit ( - 1 ) ;
}
} else {
exit ( - 1 ) ;
}
return ;
}
2011-06-26 08:30:44 +00:00
void Profile : : LoadMachineData ( )
{
2011-07-31 20:24:43 +00:00
for ( QHash < MachineID , Machine * > : : iterator i = machlist . begin ( ) ; i ! = machlist . end ( ) ; i + + ) {
Machine * m = i . value ( ) ;
2011-06-26 08:30:44 +00:00
MachineLoader * loader = GetLoader ( m - > GetClass ( ) ) ;
if ( loader ) {
long v = loader - > Version ( ) ;
long cv = 0 ;
if ( m - > properties . find ( " DataVersion " ) = = m - > properties . end ( ) ) {
m - > properties [ " DataVersion " ] = " 0 " ;
}
bool ok ;
cv = m - > properties [ " 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 ;
s . sprintf ( " %li " , v ) ;
m - > properties [ " DataVersion " ] = s ; // Dont need to nag again if they are too lazy.
} else {
try {
m - > Load ( ) ;
} catch ( OldDBVersion e ) {
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
*/
2011-07-24 16:34:53 +00:00
void Profile : : ExtraLoad ( QDomElement & root )
2011-06-26 08:30:44 +00:00
{
2011-07-24 16:34:53 +00:00
if ( root . tagName ( ) ! = " Machines " ) {
qDebug ( ) < < " No Machines Tag in Profiles.xml " ;
return ;
2011-06-26 08:30:44 +00:00
}
2011-07-24 16:34:53 +00:00
QDomElement elem = root . firstChildElement ( ) ;
while ( ! elem . isNull ( ) ) {
QString pKey = elem . tagName ( ) ;
2011-07-19 15:31:51 +00:00
if ( pKey ! = " Machine " ) {
qWarning ( ) < < " Profile::ExtraLoad() pKey!= \" Machine \" " ;
2011-07-24 16:34:53 +00:00
elem = elem . nextSiblingElement ( ) ;
2011-07-19 15:31:51 +00:00
continue ;
}
2011-06-26 08:30:44 +00:00
int m_id ;
2011-07-24 16:34:53 +00:00
bool ok ;
m_id = elem . attribute ( " id " , " " ) . toInt ( & ok ) ;
2011-06-26 08:30:44 +00:00
int mt ;
2011-07-24 16:34:53 +00:00
mt = elem . attribute ( " type " , " " ) . toInt ( & ok ) ;
2011-06-26 08:30:44 +00:00
MachineType m_type = ( MachineType ) mt ;
2011-07-24 16:34:53 +00:00
QString m_class = elem . attribute ( " class " , " " ) ;
//MachineLoader *ml=GetLoader(m_class);
2011-06-26 08:30:44 +00:00
Machine * m ;
2011-07-24 16:34:53 +00:00
//if (ml) {
// ml->CreateMachine
//}
2011-06-26 08:30:44 +00:00
if ( m_type = = MT_CPAP ) m = new CPAP ( this , m_id ) ;
else if ( m_type = = MT_OXIMETER ) m = new Oximeter ( this , m_id ) ;
else if ( m_type = = MT_SLEEPSTAGE ) m = new SleepStage ( this , m_id ) ;
else {
m = new Machine ( this , m_id ) ;
m - > SetType ( m_type ) ;
}
m - > SetClass ( m_class ) ;
AddMachine ( m ) ;
2011-07-24 16:34:53 +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
}
2011-07-24 16:34:53 +00:00
elem = elem . nextSiblingElement ( ) ;
2011-06-26 08:30:44 +00:00
}
}
void Profile : : AddMachine ( Machine * m ) {
2011-07-24 16:34:53 +00:00
if ( ! m ) {
qWarning ( ) < < " Empty Machine in Profile::AddMachine() " ;
return ;
}
2011-06-26 08:30:44 +00:00
machlist [ m - > id ( ) ] = m ;
} ;
void Profile : : DelMachine ( Machine * m ) {
2011-07-24 16:34:53 +00:00
if ( ! m ) {
qWarning ( ) < < " Empty Machine in Profile::AddMachine() " ;
return ;
}
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..
QDomElement Profile : : ExtraSave ( QDomDocument & doc )
2011-06-26 08:30:44 +00:00
{
2011-07-24 16:34:53 +00:00
QDomElement mach = doc . createElement ( " Machines " ) ;
2011-07-31 20:24:43 +00:00
for ( QHash < MachineID , Machine * > : : iterator i = machlist . begin ( ) ; i ! = machlist . end ( ) ; i + + ) {
2011-07-24 16:34:53 +00:00
QDomElement me = doc . createElement ( " Machine " ) ;
2011-07-31 20:24:43 +00:00
Machine * m = i . value ( ) ;
2011-06-26 08:30:44 +00:00
//QString t=wxT("0x")+m->hexid();
2011-07-24 16:34:53 +00:00
me . setAttribute ( " id " , ( int ) m - > id ( ) ) ;
me . setAttribute ( " type " , ( int ) m - > GetType ( ) ) ;
me . setAttribute ( " class " , m - > GetClass ( ) ) ;
2011-07-31 20:24:43 +00:00
i . value ( ) - > properties [ " path " ] = " {DataFolder}/ " + m - > hexid ( ) ;
2011-06-26 08:30:44 +00:00
2011-07-31 20:24:43 +00:00
for ( QHash < QString , QString > : : iterator j = i . value ( ) - > properties . begin ( ) ; j ! = i . value ( ) - > properties . end ( ) ; j + + ) {
QDomElement mp = doc . createElement ( j . key ( ) ) ;
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
}
2011-07-24 16:34:53 +00:00
mach . appendChild ( me ) ;
2011-06-26 08:30:44 +00:00
}
return mach ;
}
2011-07-22 13:46:17 +00:00
# include <QMessageBox>
2011-06-26 08:30:44 +00:00
void Profile : : AddDay ( QDate date , Day * day , MachineType mt ) {
//date+=wxTimeSpan::Day();
if ( is_first_day ) {
m_first = m_last = date ;
is_first_day = false ;
}
if ( m_first > date ) m_first = date ;
if ( m_last < date ) m_last = date ;
// Check for any other machines of same type.. Throw an exception if one already exists.
2011-12-08 04:10:35 +00:00
QList < Day * > & dl = daylist [ date ] ;
for ( QList < Day * > : : iterator a = dl . begin ( ) ; a ! = dl . end ( ) ; a + + ) {
2011-06-26 08:30:44 +00:00
if ( ( * a ) - > machine - > GetType ( ) = = mt ) {
throw OneTypePerDay ( ) ;
}
}
daylist [ date ] . push_back ( day ) ;
}
Day * Profile : : GetDay ( QDate date , MachineType type )
{
Day * day = NULL ;
2011-06-27 06:26:29 +00:00
// profile-> why did I d that??
if ( daylist . find ( date ) ! = daylist . end ( ) ) {
2011-12-08 04:10:35 +00:00
for ( QList < Day * > : : iterator di = daylist [ date ] . begin ( ) ; di ! = daylist [ date ] . end ( ) ; di + + ) {
2011-06-26 08:30:44 +00:00
if ( type = = MT_UNKNOWN ) { // Who cares.. We just want to know there is data available.
day = ( * di ) ;
break ;
}
if ( ( * di ) - > machine_type ( ) = = type ) {
day = ( * di ) ;
break ;
}
}
}
return day ;
}
/**
* @ brief Import Machine Data
* @ param path
*/
2011-07-15 13:30:41 +00:00
int Profile : : Import ( QString path )
2011-06-26 08:30:44 +00:00
{
int c = 0 ;
2011-07-01 10:10:44 +00:00
qDebug ( ) < < " Importing " < < path ;
2011-11-30 06:01:38 +00:00
path = path . replace ( " \\ " , " / " ) ;
if ( path . endsWith ( " / " ) ) path . chop ( 1 ) ;
2011-12-08 04:10:35 +00:00
QList < MachineLoader * > loaders = GetLoaders ( ) ;
for ( QList < MachineLoader * > : : iterator i = loaders . begin ( ) ; i ! = loaders . end ( ) ; i + + ) {
2011-06-26 08:30:44 +00:00
if ( c + = ( * i ) - > Open ( path , this ) ) break ;
}
2011-07-22 13:46:17 +00:00
//qDebug() << "Import Done";
2011-07-15 13:30:41 +00:00
return c ;
2011-06-26 08:30:44 +00:00
}
MachineLoader * GetLoader ( QString name )
{
MachineLoader * l = NULL ;
2011-12-08 04:10:35 +00:00
QList < MachineLoader * > loaders = GetLoaders ( ) ;
for ( QList < MachineLoader * > : : iterator i = loaders . begin ( ) ; i ! = loaders . end ( ) ; i + + ) {
2011-06-26 08:30:44 +00:00
if ( ( * i ) - > ClassName ( ) = = name ) {
l = * i ;
break ;
}
}
return l ;
}
2011-12-08 04:10:35 +00:00
QList < Machine * > Profile : : GetMachines ( MachineType t )
2011-07-31 20:24:43 +00:00
// Returns a QVector containing all machine objects regisered of type t
2011-06-26 08:30:44 +00:00
{
2011-12-08 04:10:35 +00:00
QList < Machine * > vec ;
2011-07-31 20:24:43 +00:00
QHash < MachineID , Machine * > : : iterator i ;
2011-06-26 08:30:44 +00:00
for ( i = machlist . begin ( ) ; i ! = machlist . end ( ) ; i + + ) {
2011-07-31 20:24:43 +00:00
if ( ! i . value ( ) ) {
2011-07-19 15:31:51 +00:00
qWarning ( ) < < " Profile::GetMachines() i->second == NULL " ;
continue ;
}
2011-07-31 20:24:43 +00:00
if ( i . value ( ) - > GetType ( ) = = t ) {
vec . push_back ( i . value ( ) ) ;
2011-06-26 08:30:44 +00:00
}
}
return vec ;
}
Machine * Profile : : GetMachine ( MachineType t )
{
2011-12-08 04:10:35 +00:00
QList < Machine * > vec = GetMachines ( t ) ;
2011-06-26 08:30:44 +00:00
if ( vec . size ( ) = = 0 ) return NULL ;
return vec [ 0 ] ;
}
2011-12-06 14:39:14 +00:00
void Profile : : RemoveSession ( Session * sess )
{
2011-12-08 04:10:35 +00:00
QMap < QDate , QList < Day * > > : : iterator di ;
2011-12-06 14:39:14 +00:00
for ( di = daylist . begin ( ) ; di ! = daylist . end ( ) ; di + + ) {
for ( int d = 0 ; d < di . value ( ) . size ( ) ; d + + ) {
Day * day = di . value ( ) [ d ] ;
int i = day - > getSessions ( ) . indexOf ( sess ) ;
if ( i > = 0 ) {
for ( ; i < day - > getSessions ( ) . size ( ) - 1 ; i + + ) {
day - > getSessions ( ) [ i ] = day - > getSessions ( ) [ i + 1 ] ;
}
day - > getSessions ( ) . pop_back ( ) ;
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 ( ) ;
}
day - > setFirst ( first ) ;
day - > setLast ( last ) ;
return ;
}
}
}
}
2011-06-27 10:23:24 +00:00
//Profile *profile=NULL;
2011-06-26 08:30:44 +00:00
QString SHA1 ( QString pass )
{
return pass ;
}
namespace Profiles
{
2011-07-31 20:24:43 +00:00
QHash < 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 ( ) ;
// Only save the open profile..
2011-07-31 20:24:43 +00:00
for ( QHash < QString , Profile * > : : iterator i = profiles . begin ( ) ; i ! = profiles . end ( ) ; i + + ) {
i . value ( ) - > Save ( ) ;
delete i . value ( ) ;
2011-06-26 08:30:44 +00:00
}
profiles . clear ( ) ;
delete p_pref ;
delete p_layout ;
}
Profile * Get ( QString name )
{
if ( profiles . find ( name ) ! = profiles . end ( ) )
return profiles [ name ] ;
return NULL ;
}
2011-10-01 12:54:20 +00:00
Profile * Create ( QString name )
2011-06-26 08:30:44 +00:00
{
2011-10-05 07:41:56 +00:00
QString path = PREF . Get ( " {home}/Profiles/ " ) + name ;
2011-06-26 08:30:44 +00:00
QDir dir ( path ) ;
if ( ! dir . exists ( path ) ) dir . mkpath ( path ) ;
//path+="/"+name;
Profile * prof = new Profile ( path ) ;
prof - > Open ( ) ;
profiles [ name ] = prof ;
prof - > Set ( " Username " , name ) ;
2011-09-08 09:50:05 +00:00
//prof->Set("Realname",realname);
2011-10-01 12:54:20 +00:00
//if (!password.isEmpty()) prof->Set("Password",SHA1(password));
2011-06-26 08:30:44 +00:00
prof - > Set ( " DataFolder " , " {home}/Profiles/{Username} " ) ;
Machine * m = new Machine ( prof , 0 ) ;
m - > SetClass ( " Journal " ) ;
m - > properties [ " Brand " ] = " Virtual " ;
m - > SetType ( MT_JOURNAL ) ;
prof - > AddMachine ( m ) ;
prof - > Save ( ) ;
return prof ;
}
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 ( )
{
2011-09-17 12:39:00 +00:00
//InitMapsWithoutAwesomeInitializerLists();
2011-06-26 08:30:44 +00:00
p_pref = new Preferences ( " Preferences " ) ;
p_layout = new Preferences ( " Layout " ) ;
2011-10-05 07:41:56 +00:00
PREF . Open ( ) ;
LAYOUT . Open ( ) ;
2011-06-26 08:30:44 +00:00
2011-10-05 07:41:56 +00:00
QString path = PREF . Get ( " {home}/Profiles " ) ;
2011-06-26 08:30:44 +00:00
QDir dir ( path ) ;
if ( ! dir . exists ( path ) ) {
2011-10-01 12:54:20 +00:00
//dir.mkpath(path);
2011-06-26 08:30:44 +00:00
// Just silently create a new user record and get on with it.
2011-10-01 12:54:20 +00:00
//Create(getUserName(),getUserName(),"");
2011-06-26 08:30:44 +00:00
return ;
}
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);
QFileInfoList list = dir . entryInfoList ( ) ;
2011-10-01 12:54:20 +00:00
//QString username=getUserName();
//if (list.size()==0) { // No profiles.. Create one.
//Create(username,username,"");
//return;
//}
2011-07-17 07:03:26 +00:00
// Iterate through subdirectories and load profiles..
2011-06-26 08:30:44 +00:00
for ( int i = 0 ; i < list . size ( ) ; i + + ) {
QFileInfo fi = list . at ( i ) ;
QString npath = fi . canonicalFilePath ( ) ;
Profile * prof = new Profile ( npath ) ;
2011-07-17 07:03:26 +00:00
prof - > Open ( ) ; // Read it's XML file..
2011-06-26 08:30:44 +00:00
profiles [ fi . fileName ( ) ] = prof ;
}
}
2011-12-06 14:39:14 +00:00
2011-06-26 08:30:44 +00:00
} ; // namespace Profiles