OSCAR-code/SleepLib/profiles.cpp

865 lines
25 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 <QApplication>
#include "mainwindow.h"
extern MainWindow * mainwin;
2011-06-26 08:30:44 +00:00
Preferences *p_pref;
Preferences *p_layout;
Profile * p_profile;
2011-06-26 08:30:44 +00:00
Profile::Profile()
:Preferences(),is_first_day(true)
{
2011-12-21 14:24:09 +00:00
p_name=STR_GEN_Profile;
p_path=PREF.Get("{home}/Profiles");
2011-06-26 08:30:44 +00:00
machlist.clear();
doctor=new DoctorInfo(this);
user=new UserInfo(this);
cpap=new CPAPSettings(this);
oxi=new OxiSettings(this);
appearance=new AppearanceSettings(this);
2011-12-21 12:47:47 +00:00
session=new SessionSettings(this);
general=new UserSettings(this);
2011-06-26 08:30:44 +00:00
}
Profile::Profile(QString path)
:Preferences(),is_first_day(true)
{
const QString xmlext=".xml";
2011-12-21 14:24:09 +00:00
p_name=STR_GEN_Profile;
2011-06-26 08:30:44 +00:00
if (path.isEmpty()) p_path=GetAppRoot();
else p_path=path;
2011-12-21 14:24:09 +00:00
(*this)[STR_GEN_DataFolder]=p_path;
2011-06-26 08:30:44 +00:00
if (!p_path.endsWith("/")) p_path+="/";
p_filename=p_path+p_name+xmlext;
machlist.clear();
doctor=new DoctorInfo(this);
user=new UserInfo(this);
cpap=new CPAPSettings(this);
oxi=new OxiSettings(this);
appearance=new AppearanceSettings(this);
2011-12-21 12:47:47 +00:00
session=new SessionSettings(this);
general=new UserSettings(this);
2011-06-26 08:30:44 +00:00
}
bool Profile::Save(QString filename)
2011-06-26 08:30:44 +00:00
{
if (p_profile==this) {
QString cachefile=Get("{DataFolder}")+QDir::separator()+"cache.day";
QFile f(cachefile);
if (ExistsAndTrue("TrashDayCache")) {
if (f.exists()) {
f.remove();
}
} else {
QMap<QDate,QList<Day *> >::iterator di;
QHash<MachineID,QMap<QDate,QHash<ChannelID, EventDataType> > > cache;
QHash<MachineID,QMap<QDate,QHash<ChannelID, EventDataType> > >::iterator ci;
for (di=daylist.begin();di!=daylist.end();di++) {
QDate date=di.key();
for (QList<Day *>::iterator d=di.value().begin();d!=di.value().end();d++) {
Day *day=*d;
MachineID mach=day->machine->id();
QHash<ChannelID, EventDataType>::iterator i;
for (i=day->m_p90.begin();i!=day->m_p90.end();i++) {
cache[mach][date][i.key()]=day->m_p90[i.key()];
}
}
}
if (f.open(QFile::WriteOnly)) {
QDataStream out(&f);
out.setVersion(QDataStream::Qt_4_6);
out.setByteOrder(QDataStream::LittleEndian);
quint16 size=cache.size();
out << size;
for (ci=cache.begin();ci!=cache.end();ci++) {
quint32 mid=ci.key();
out << mid;
out << ci.value();
}
f.close();
}
}
(*this)["TrashDayCache"]=false;
}
return Preferences::Save(filename);
}
Profile::~Profile()
{
delete user;
delete doctor;
delete cpap;
delete oxi;
delete appearance;
2011-12-21 12:47:47 +00:00
delete session;
delete general;
for (QHash<MachineID,Machine *>::iterator i=machlist.begin(); i!=machlist.end(); i++) {
delete i.value();
2011-06-26 08:30:44 +00:00
}
}
void Profile::DataFormatError(Machine *m)
{
QString msg=QObject::tr("Software changes have been made that require the reimporting of the following machines data:\n\n");
2011-12-21 14:24:09 +00:00
msg=msg+m->properties[STR_PROP_Brand]+" "+m->properties[STR_PROP_Model]+" "+m->properties[STR_PROP_Serial];
msg=msg+QObject::tr("\n\nThis is still only alpha software and these changes are sometimes necessary.\n\n");
msg=msg+QObject::tr("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+QObject::tr("Would you like me to purge this data this for you so you can run the new version?");
if (QMessageBox::warning(NULL,QObject::tr("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-12-21 14:24:09 +00:00
QMessageBox::critical(NULL,QObject::tr("Purge Failed"),QObject::tr("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\nThis folder currently resides at the following location:\n")+PREF[STR_GEN_DataFolder].toString(),QMessageBox::Ok);
exit(-1);
}
} else {
exit(-1);
}
return;
}
2011-06-26 08:30:44 +00:00
void Profile::LoadMachineData()
{
QHash<MachineID,QMap<QDate,QHash<ChannelID, EventDataType> > > cache;
QString filename=Get("{DataFolder}")+QDir::separator()+"cache.day";
QFile f(filename);
if (f.exists(filename) && (f.open(QFile::ReadOnly))) {
QDataStream in(&f);
in.setVersion(QDataStream::Qt_4_6);
in.setByteOrder(QDataStream::LittleEndian);
quint16 size;
quint32 mid;
in >> size;
for (int i=0;i<size;i++) {
in >> mid;
in >> cache[mid];
}
2011-12-21 12:47:47 +00:00
PROFILE.general->setRebuildCache(false);
} else {
if (mainwin) {
mainwin->Notify(QObject::tr("Caching session data, this may take a little while."));
2011-12-21 12:47:47 +00:00
PROFILE.general->setRebuildCache(true);
QApplication::processEvents();
}
}
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;
2011-12-21 14:24:09 +00:00
if (m->properties.find(STR_PROP_DataVersion)==m->properties.end()) {
m->properties[STR_PROP_DataVersion]="0";
2011-06-26 08:30:44 +00:00
}
bool ok;
2011-12-21 14:24:09 +00:00
cv=m->properties[STR_PROP_DataVersion].toLong(&ok);
2011-06-26 08:30:44 +00:00
if (!ok || cv<v) {
DataFormatError(m);
// It may exit above and not return here..
QString s;
s.sprintf("%li",v);
2011-12-21 14:24:09 +00:00
m->properties[STR_PROP_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-06-26 08:30:44 +00:00
} else {
m->Load();
}
}
for (QMap<QDate,QList<Day *> >::iterator di=daylist.begin();di!=daylist.end();di++) {
for (QList<Day *>::iterator d=di.value().begin();d!=di.value().end();d++) {
Day *day=*d;
MachineID mid=day->machine->id();
if (cache.contains(mid)) {
if (cache[mid].contains(di.key())) {
day->m_p90=cache[mid][di.key()];
}
}
}
}
// Load Day Cache here..
2011-06-26 08:30:44 +00:00
}
/**
* @brief Machine XML section in profile.
* @param root
*/
void Profile::ExtraLoad(QDomElement & root)
2011-06-26 08:30:44 +00:00
{
if (root.tagName()!="Machines") {
qDebug() << "No Machines Tag in Profiles.xml";
return;
2011-06-26 08:30:44 +00:00
}
QDomElement elem=root.firstChildElement();
while (!elem.isNull()) {
QString pKey=elem.tagName();
if (pKey!="Machine") {
qWarning() << "Profile::ExtraLoad() pKey!=\"Machine\"";
elem=elem.nextSiblingElement();
continue;
}
2011-06-26 08:30:44 +00:00
int m_id;
bool ok;
m_id=elem.attribute("id","").toInt(&ok);
2011-06-26 08:30:44 +00:00
int mt;
mt=elem.attribute("type","").toInt(&ok);
2011-06-26 08:30:44 +00:00
MachineType m_type=(MachineType)mt;
QString m_class=elem.attribute("class","");
//MachineLoader *ml=GetLoader(m_class);
2011-06-26 08:30:44 +00:00
Machine *m;
//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);
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
}
elem=elem.nextSiblingElement();
2011-06-26 08:30:44 +00:00
}
}
void Profile::AddMachine(Machine *m) {
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) {
if (!m) {
qWarning() << "Empty Machine in Profile::AddMachine()";
return;
}
machlist.erase(machlist.find(m->id()));
2011-06-26 08:30:44 +00:00
};
// Potential Memory Leak Here..
QDomElement Profile::ExtraSave(QDomDocument & doc)
2011-06-26 08:30:44 +00:00
{
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();
2011-06-26 08:30:44 +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()));
//mp->LinkEndChild(new QDomText(j->second.toLatin1()));
me.appendChild(mp);
2011-06-26 08:30:44 +00:00
}
mach.appendChild(me);
2011-06-26 08:30:44 +00:00
}
return mach;
}
#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.
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()) {
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;
}
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;
path=path.replace("\\","/");
if (path.endsWith("/")) path.chop(1);
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;
}
//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;
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;
}
QList<Machine *> Profile::GetMachines(MachineType t)
// Returns a QVector containing all machine objects regisered of type t
2011-06-26 08:30:44 +00:00
{
QList<Machine *> vec;
QHash<MachineID,Machine *>::iterator i;
2011-06-26 08:30:44 +00:00
for (i=machlist.begin(); i!=machlist.end(); i++) {
if (!i.value()) {
qWarning() << "Profile::GetMachines() i->second == NULL";
continue;
}
MachineType mt=i.value()->GetType();
if ((t==MT_UNKNOWN) || (mt==t)) {
vec.push_back(i.value());
2011-06-26 08:30:44 +00:00
}
}
return vec;
}
Machine * Profile::GetMachine(MachineType t)
{
QList<Machine *>vec=GetMachines(t);
2011-06-26 08:30:44 +00:00
if (vec.size()==0) return NULL;
return vec[0];
}
void Profile::RemoveSession(Session * sess)
{
QMap<QDate,QList<Day *> >::iterator di;
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
{
QHash<QString,Profile *> profiles;
2011-06-26 08:30:44 +00:00
void Done()
{
PREF.Save();
LAYOUT.Save();
// Only save the open profile..
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
{
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;
2011-12-21 14:24:09 +00:00
prof->user->setUserName(name);
//prof->Set("Realname",realname);
2011-12-21 14:24:09 +00:00
//if (!password.isEmpty()) prof.user->setPassword(password);
prof->Set(STR_GEN_DataFolder,QString("{home}/Profiles/{")+QString(UI_STR_UserName)+QString("}"));
2011-06-26 08:30:44 +00:00
Machine *m=new Machine(prof,0);
m->SetClass("Journal");
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);
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");
PREF.Open();
LAYOUT.Open();
2011-06-26 08:30:44 +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;
}
}
} // namespace Profiles
2011-06-26 08:30:44 +00:00
2011-12-21 12:47:47 +00:00
// DoctorInfo Strings
const char * DI_STR_Name="DoctorName";
const char * DI_STR_Phone="DoctorPhone";
2011-12-21 12:47:47 +00:00
const char * DI_STR_Email="DoctorEmail";
const char * DI_STR_Practice="DoctorPractice";
const char * DI_STR_Address="DoctorAddress";
const char * DI_STR_PatientID="DoctorPatientID";
2011-12-21 12:47:47 +00:00
// UserInfo Strings
const char * UI_STR_DOB="DOB";
const char * UI_STR_FirstName="FirstName";
const char * UI_STR_LastName="LastName";
const char * UI_STR_UserName="UserName";
const char * UI_STR_Password="Password";
const char * UI_STR_Address="Address";
const char * UI_STR_Phone="Phone";
const char * UI_STR_EmailAddress="EmailAddress";
const char * UI_STR_Country="Country";
const char * UI_STR_Height="Height";
const char * UI_STR_Gender="Gender";
const char * UI_STR_TimeZone="TimeZone";
const char * UI_STR_Language="Language";
const char * UI_STR_DST="DST";
2011-12-21 12:47:47 +00:00
// OxiSettings Strings
const char * OS_STR_EnableOximetry="EnableOximetry";
const char * OS_STR_SyncOximetry="SyncOximetry";
const char * OS_STR_OximeterType="OximeterType";
const char * OS_STR_OxiDiscardThreshold="OxiDiscardThreshold";
const char * OS_STR_SPO2DropDuration="SPO2DropDuration";
const char * OS_STR_SPO2DropPercentage="SPO2DropPercentage";
const char * OS_STR_PulseChangeDuration="PulseChangeDuration";
const char * OS_STR_PulseChangeBPM="PulseChangeBPM";
2011-12-21 12:47:47 +00:00
// CPAPSettings Strings
const char * CS_STR_ComplianceHours="ComplianceHours";
const char * CS_STR_ShowCompliance="ShowCompliance";
const char * CS_STR_ShowLeaksMode="ShowLeaksMode";
const char * CS_STR_MaskStartDate="MaskStartDate";
const char * CS_STR_MaskDescription="MaskDescription";
const char * CS_STR_MaskType="MaskType";
const char * CS_STR_PrescribedMode="CPAPPrescribedMode";
const char * CS_STR_PrescribedMinPressure="CPAPPrescribedMinPressure";
const char * CS_STR_PrescribedMaxPressure="CPAPPrescribedMaxPressure";
const char * CS_STR_UntreatedAHI="UntreatedAHI";
const char * CS_STR_Notes="CPAPNotes";
const char * CS_STR_DateDiagnosed="DateDiagnosed";
2011-12-21 12:47:47 +00:00
// ImportSettings Strings
const char * IS_STR_DaySplitTime="DaySplitTime";
const char * IS_STR_CacheSessions="MemoryHog";
const char * IS_STR_CombineCloseSessions="CombineCloserSessions";
const char * IS_STR_IgnoreShorterSessions="IgnoreShorterSessions";
const char * IS_STR_Multithreading="EnableMultithreading";
2011-12-21 12:47:47 +00:00
const char * IS_STR_TrashDayCache="TrashDayCache";
const char * IS_STR_ShowSerialNumbers="ShowSerialNumbers";
2011-12-21 12:47:47 +00:00
// AppearanceSettings Strings
const char * AS_STR_GraphHeight="GraphHeight";
const char * AS_STR_AntiAliasing="UseAntiAliasing";
const char * AS_STR_HighResPrinting="HighResPrinting";
const char * AS_STR_GraphSnapshots="EnableGraphSnapshots";
const char * AS_STR_Animations="AnimationsAndTransitions";
const char * AS_STR_SquareWave="SquareWavePlots";
const char * AS_STR_OverlayType="OverlayType";
2011-12-21 12:47:47 +00:00
// UserSettings Strings
const char * US_STR_UnitSystem="UnitSystem";
const char * US_STR_EventWindowSize="EventWindowSize";
const char * US_STR_SkipEmptyDays="SkipEmptyDays";
const char * US_STR_RebuildCache="RebuildCache";
const char * US_STR_ShowDebug="ShowDebug";
const char * US_STR_LinkGroups="LinkGroups";
2011-12-22 13:22:40 +00:00
int Profile::countDays(MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
int days=0;
do {
Day * day=GetDay(date,mt);
if (day) {
if ((mt==MT_UNKNOWN) || (day->machine->GetType()==mt)) 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)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
double val=0;
do {
Day * day=GetDay(date,mt);
if (day) {
val+=day->count(code);
}
date=date.addDays(1);
} while (date<end);
return val;
}
double Profile::calcSum(ChannelID code, MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
double val=0;
do {
Day * day=GetDay(date,mt);
if (day) {
val+=day->sum(code);
}
date=date.addDays(1);
} while (date<end);
return val;
}
EventDataType Profile::calcHours(MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
double val=0;
do {
Day * day=GetDay(date,mt);
if (day) {
val+=day->hours();
}
date=date.addDays(1);
} while (date<end);
return val;
}
EventDataType Profile::calcAvg(ChannelID code, MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
double val=0;
int cnt=0;
do {
Day * day=GetDay(date,mt);
if (day) {
val+=day->sum(code);
cnt++;
}
date=date.addDays(1);
} while (date<end);
if (!cnt) return 0;
return val/float(cnt);
}
EventDataType Profile::calcWavg(ChannelID code, MachineType mt, QDate start, QDate end)
{
if (!start.isValid())
start=LastDay(mt);
if (!end.isValid())
end=LastDay(mt);
QDate date=start;
double val=0,tmp,tmph,hours=0;
do {
Day * day=GetDay(date,mt);
if (day) {
tmph=day->hours();
tmp=day->wavg(code);
val+=tmp*tmph;
hours+=tmph;
}
date=date.addDays(1);
} while (date<end);
if (!hours) return 0;
val=val/hours;
return val;
}
EventDataType Profile::calcMin(ChannelID code, MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
double min=99999999,tmp;
do {
Day * day=GetDay(date,mt);
if (day) {
tmp=day->Min(code);
if (min>tmp) min=tmp;
}
date=date.addDays(1);
} while (date<end);
if (min>=99999999) min=0;
return min;
}
EventDataType Profile::calcMax(ChannelID code, MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
double max=-99999999,tmp;
do {
Day * day=GetDay(date,mt);
if (day) {
tmp=day->Max(code);
if (max<tmp) max=tmp;
}
date=date.addDays(1);
} while (date<end);
if (max<=-99999999) max=0;
return max;
}
EventDataType Profile::calcSettingsMin(ChannelID code, MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
double min=99999999,tmp;
do {
Day * day=GetDay(date,mt);
if (day) {
tmp=day->settings_min(code);
if (min>tmp) min=tmp;
}
date=date.addDays(1);
} while (date<end);
if (min>=99999999) min=0;
return min;
}
EventDataType Profile::calcSettingsMax(ChannelID code, MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
double max=-99999999,tmp;
do {
Day * day=GetDay(date,mt);
if (day) {
tmp=day->settings_max(code);
if (max<tmp) max=tmp;
}
date=date.addDays(1);
} while (date<end);
if (max<=-99999999) max=0;
return max;
}
2011-12-22 13:22:40 +00:00
EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, MachineType mt, QDate start, QDate end)
{
if (!start.isValid()) start=LastDay(mt);
if (!end.isValid()) end=LastDay(mt);
QDate date=start;
// This is one messy function.. It requires all data to be loaded.. :(
QVector<EventDataType> array;
2011-12-22 13:22:40 +00:00
//double val=0,tmp,hours=0;
do {
Day * day=GetDay(date,mt);
if (day) {
bool open=day->eventsLoaded();
if (!open)
day->OpenEvents();
for (int i=0;i<day->size();i++) {
for (QVector<Session *>::iterator s=day->begin();s!=day->end();s++) {
QHash<ChannelID,QVector<EventList *> >::iterator el=(*s)->eventlist.find(code);
if (el==(*s)->eventlist.end()) continue;
for (int j=0;j<el.value().size();j++) {
EventList & e=*el.value()[j];
for (unsigned k=0;k<e.count();k++) {
array.push_back(e.data(k));
}
}
}
}
//if (!open)
//day->CloseEvents();
}
2011-12-22 13:22:40 +00:00
date=date.addDays(1);
} while (date<end);
qSort(array);
int idx=array.size()*percent;
if (idx>array.size()-1) idx=array.size()-1;
return array[idx];
2011-12-22 13:22:40 +00:00
}
QDate Profile::FirstDay(MachineType mt)
{
if ((mt==MT_UNKNOWN) || (!m_last.isValid()) || (!m_first.isValid()))
return m_first;
QDate d=m_first;
do {
if (GetDay(d,mt)!=NULL) return d;
d=d.addDays(1);
} while (d<=m_last);
return m_last;
}
QDate Profile::LastDay(MachineType mt)
{
if ((mt==MT_UNKNOWN) || (!m_last.isValid()) || (!m_first.isValid()))
return m_last;
QDate d=m_last;
do {
if (GetDay(d,mt)!=NULL) return d;
d=d.addDays(-1);
} while (d>=m_first);
return m_first;
}