OSCAR-code/SleepLib/profiles.cpp

371 lines
9.7 KiB
C++
Raw Normal View History

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"
#include "tinyxml/tinyxml.h"
Preferences *p_pref;
Preferences *p_layout;
Profile::Profile()
:Preferences(),is_first_day(true)
{
p_name="Profile";
p_path=pref.Get("{home}/Profiles");
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()
{
for (map<MachineID,Machine *>::iterator i=machlist.begin(); i!=machlist.end(); i++) {
delete i->second;
}
}
void Profile::LoadMachineData()
{
for (map<MachineID,Machine *>::iterator i=machlist.begin(); i!=machlist.end(); i++) {
Machine *m=i->second;
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) {
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\nNo attempt will be made to load previous data.\n\n";
msg=msg+"Importing ALL of your data for this machine again will rectify this problem.\n\n";
msg=msg+"However, if you have more than one seperate datacard/stash for this machine, it would be best if the machine data was purged first.\n\nWould you like me to do this for you?";
if (QMessageBox::warning(NULL,"Machine Database Changes",msg,QMessageBox::Yes | QMessageBox::No,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
QString s;
s.sprintf("%li",v);
m->properties["DataVersion"]=s; // Dont need to nag again if they are too lazy.
}
}
} else m->Load();
} else {
m->Load();
}
}
}
/**
* @brief Machine XML section in profile.
* @param root
*/
void Profile::ExtraLoad(TiXmlHandle *root)
{
TiXmlElement *elem;
elem=root->FirstChild("Machines").FirstChild().Element();
if (!elem) {
// qDebug("ExtraLoad: Elem is empty.");
2011-06-26 08:30:44 +00:00
}
for(; elem; elem=elem->NextSiblingElement()) {
QString pKey=elem->Value();
assert(pKey=="Machine");
int m_id;
elem->QueryIntAttribute("id",&m_id);
int mt;
elem->QueryIntAttribute("type",&mt);
MachineType m_type=(MachineType)mt;
QString m_class=elem->Attribute("class");
Machine *m;
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);
TiXmlElement *e=elem->FirstChildElement();
for (; e; e=e->NextSiblingElement()) {
QString pKey=e->Value();
m->properties[pKey]=e->GetText();
}
}
}
void Profile::AddMachine(Machine *m) {
assert(m!=NULL);
machlist[m->id()]=m;
};
void Profile::DelMachine(Machine *m) {
assert(m!=NULL);
machlist.erase(m->id());
};
TiXmlElement * Profile::ExtraSave()
{
TiXmlElement *mach=new TiXmlElement("Machines");
for (map<MachineID,Machine *>::iterator i=machlist.begin(); i!=machlist.end(); i++) {
TiXmlElement *me=new TiXmlElement("Machine");
Machine *m=i->second;
//QString t=wxT("0x")+m->hexid();
me->SetAttribute("id",m->id());
me->SetAttribute("type",(int)m->GetType());
char *cc=m->GetClass().toLatin1().data();
me->SetAttribute("class",cc);
i->second->properties["path"]="{DataFolder}/"+m->hexid();
for (map<QString,QString>::iterator j=i->second->properties.begin(); j!=i->second->properties.end(); j++) {
TiXmlElement *mp=new TiXmlElement(j->first.toLatin1());
mp->LinkEndChild(new TiXmlText(j->second.toLatin1()));
me->LinkEndChild(mp);
}
mach->LinkEndChild(me);
}
//root->LinkEndChild(mach);
return mach;
}
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.
vector<Day *> & dl=daylist[date];
for (vector<Day *>::iterator a=dl.begin();a!=dl.end();a++) {
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()) {
for (vector<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-06-26 08:30:44 +00:00
list<MachineLoader *>loaders=GetLoaders();
for (list<MachineLoader *>::iterator i=loaders.begin(); i!=loaders.end(); i++) {
if (c+=(*i)->Open(path,this)) break;
}
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;
list<MachineLoader *>loaders=GetLoaders();
for (list<MachineLoader *>::iterator i=loaders.begin(); i!=loaders.end(); i++) {
if ((*i)->ClassName()==name) {
l=*i;
break;
}
}
return l;
}
vector<Machine *> Profile::GetMachines(MachineType t)
// Returns a vector containing all machine objects regisered of type t
{
vector<Machine *> vec;
map<MachineID,Machine *>::iterator i;
for (i=machlist.begin(); i!=machlist.end(); i++) {
assert(i->second!=NULL);
if (i->second->GetType()==t) {
vec.push_back(i->second);
}
}
return vec;
}
Machine * Profile::GetMachine(MachineType t)
{
vector<Machine *>vec=GetMachines(t);
if (vec.size()==0) return NULL;
return vec[0];
}
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
{
std::map<QString,Profile *> profiles;
void Done()
{
pref.Save();
laypref.Save();
for (map<QString,Profile *>::iterator i=profiles.begin(); i!=profiles.end(); i++) {
i->second->Save();
delete i->second;
}
profiles.clear();
delete p_pref;
delete p_layout;
}
Profile *Get(QString name)
{
if (profiles.find(name)!=profiles.end())
return profiles[name];
return NULL;
}
Profile *Create(QString name,QString realname,QString password)
{
QString path=pref.Get("{home}/Profiles/")+name;
QDir dir(path);
if (!dir.exists(path)) dir.mkpath(path);
//path+="/"+name;
//if (!dir.exists(path)) wxMkdir(path);
Profile *prof=new Profile(path);
prof->Open();
profiles[name]=prof;
prof->Set("Username",name);
prof->Set("Realname",realname);
if (!password.isEmpty()) prof->Set("Password",SHA1(password));
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()
{
InitMapsWithoutAwesomeInitializerLists();
p_pref=new Preferences("Preferences");
p_layout=new Preferences("Layout");
pref.Open();
laypref.Open();
QString path=pref.Get("{home}/Profiles");
QDir dir(path);
if (!dir.exists(path)) {
dir.mkpath(path);
// Just silently create a new user record and get on with it.
Create(getUserName(),getUserName(),"");
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();
QString username=getUserName();
2011-07-17 07:03:26 +00:00
if (list.size()==0) { // No profiles.. Create one.
2011-06-26 08:30:44 +00:00
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;
}
}
}; // namespace Profiles