mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
Some profiling & optimisations. Implemented Session Events Compression, Backup edf File compression for ResMed, preference changed around a bit, and new options for the compression and backup stuff. And more efficient Weighted Percentile calculations
This commit is contained in:
parent
88fbc9d746
commit
3e7684efac
@ -427,6 +427,7 @@ void SummaryChart::paint(gGraph & w,int left, int top, int width, int height)
|
||||
if (!m_hours.contains(zd))
|
||||
goto jumpnext;
|
||||
//continue;
|
||||
|
||||
hours=m_hours[zd];
|
||||
|
||||
int x1=px;
|
||||
@ -473,12 +474,13 @@ void SummaryChart::paint(gGraph & w,int left, int top, int width, int height)
|
||||
outlines->add(x1,py,x2,py,x2,py,x2,py-h);
|
||||
} // if (bar
|
||||
//py-=h;
|
||||
totalvalues[0]+=tmp;
|
||||
totalvalues[0]+=hours*tmp;
|
||||
}
|
||||
totalcounts[0]++;
|
||||
totalcounts[0]+=hours;
|
||||
totalvalues[1]+=j;
|
||||
totalcounts[1]++;
|
||||
total_val+=hours;
|
||||
total_hours+=hours;
|
||||
total_days++;
|
||||
} else {
|
||||
if (!d.value().contains(0)) goto jumpnext;
|
||||
|
@ -409,7 +409,6 @@ int calcAHIGraph(Session *session)
|
||||
const qint64 window_size=3600000L;
|
||||
const qint64 window_step=30000; // 30 second windows
|
||||
|
||||
|
||||
if (session->machine()->GetType()!=MT_CPAP) return 0;
|
||||
if (session->eventlist.contains(CPAP_AHI)) return 0; // abort if already there
|
||||
|
||||
|
@ -36,3 +36,8 @@ QString weightString(float kg, UnitSystem us)
|
||||
}
|
||||
return("Bad UnitSystem");
|
||||
}
|
||||
|
||||
bool operator <(const ValueCount & a, const ValueCount & b)
|
||||
{
|
||||
return a.value < b.value;
|
||||
}
|
||||
|
@ -6,11 +6,29 @@
|
||||
|
||||
enum UnitSystem { US_Undefined, US_Metric, US_Archiac };
|
||||
|
||||
typedef float EventDataType;
|
||||
|
||||
struct ValueCount {
|
||||
ValueCount() { value=0; count=0; p=0; }
|
||||
ValueCount(const ValueCount & copy) {
|
||||
value=copy.value;
|
||||
count=copy.count;
|
||||
p=copy.p;
|
||||
}
|
||||
EventDataType value;
|
||||
int count;
|
||||
double p;
|
||||
};
|
||||
|
||||
// Primarily sort by value
|
||||
bool operator <(const ValueCount & a, const ValueCount & b);
|
||||
|
||||
const float ounce_convert=28.3495231; // grams
|
||||
const float pound_convert=ounce_convert*16;
|
||||
|
||||
QString weightString(float kg, UnitSystem us=US_Undefined);
|
||||
|
||||
|
||||
const QString STR_UNIT_CM=QObject::tr("cm");
|
||||
const QString STR_UNIT_INCH=QObject::tr("\"");
|
||||
const QString STR_UNIT_FOOT=QObject::tr("ft");
|
||||
@ -40,6 +58,7 @@ const QString STR_PROP_SubModel="SubModel";
|
||||
const QString STR_PROP_Serial="Serial";
|
||||
const QString STR_PROP_DataVersion="DataVersion";
|
||||
const QString STR_PROP_Path="Path";
|
||||
const QString STR_PROP_BackupPath="BackupPath";
|
||||
const QString STR_PROP_LastImported="LastImported";
|
||||
|
||||
const QString STR_MACH_ResMed="ResMed";
|
||||
|
119
SleepLib/day.cpp
119
SleepLib/day.cpp
@ -6,6 +6,7 @@
|
||||
|
||||
#include "day.h"
|
||||
#include "profiles.h"
|
||||
#include <cmath>
|
||||
#include <algorithm>
|
||||
|
||||
Day::Day(Machine *m)
|
||||
@ -149,60 +150,110 @@ EventDataType Day::settings_wavg(ChannelID code)
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
EventDataType Day::percentile(ChannelID code,EventDataType percentile)
|
||||
{
|
||||
// Cache this calculation
|
||||
//if (percentile>=1) return 0; // probably better to crash and burn.
|
||||
// QHash<ChannelID, QHash<EventDataType, EventDataType> >::iterator pi;
|
||||
// pi=perc_cache.find(code);
|
||||
// if (pi!=perc_cache.end()) {
|
||||
// QHash<EventDataType, EventDataType> & hsh=pi.value();
|
||||
// QHash<EventDataType, EventDataType>::iterator hi=hsh.find(
|
||||
// if (hi!=pi.value().end()) {
|
||||
// return hi.value();
|
||||
// }
|
||||
// }
|
||||
// Cache this calculation?
|
||||
|
||||
QVector<Session *>::iterator s;
|
||||
|
||||
// Don't assume sessions are in order.
|
||||
QVector<EventDataType> ar;
|
||||
QMap<EventDataType, int> wmap;
|
||||
|
||||
int SN=0;
|
||||
|
||||
// First Calculate count of all events
|
||||
for (s=sessions.begin();s!=sessions.end();s++) {
|
||||
if (!(*s)->enabled()) continue;
|
||||
|
||||
Session & sess=*(*s);
|
||||
QHash<ChannelID,QHash<EventStoreType, EventStoreType> > ::iterator ei=sess.m_valuesummary.find(code);
|
||||
if (ei==sess.m_valuesummary.end()) continue;
|
||||
|
||||
EventDataType gain=sess.m_gain[code];
|
||||
EventDataType weight,value;
|
||||
for (QHash<EventStoreType, EventStoreType>::iterator i=ei.value().begin();i!=ei.value().end();i++) {
|
||||
for (int j=0;j<i.value();j++) {
|
||||
ar.push_back(float(i.key())*gain);
|
||||
weight=i.value();
|
||||
value=EventDataType(i.key())*gain;
|
||||
|
||||
SN+=weight;
|
||||
if (wmap.contains(value)) {
|
||||
wmap[value]+=weight;
|
||||
} else {
|
||||
wmap[value]=weight;
|
||||
}
|
||||
}
|
||||
}
|
||||
int size=ar.size();
|
||||
if (!size)
|
||||
return 0;
|
||||
size--;
|
||||
|
||||
QVector<EventDataType>::iterator first=ar.begin();
|
||||
QVector<EventDataType>::iterator last=ar.end();
|
||||
QVector<EventDataType>::iterator middle = first + int((last-first) * percentile);
|
||||
std::nth_element(first,middle,last);
|
||||
EventDataType val=*middle;
|
||||
QVector<ValueCount> valcnt;
|
||||
|
||||
// qSort(ar);
|
||||
// int p=EventDataType(size)*percentile;
|
||||
// float p2=EventDataType(size)*percentile;
|
||||
// float diff=p2-p;
|
||||
// EventDataType val=ar[p];
|
||||
// if (diff>0) {
|
||||
// int s=p+1;
|
||||
// if (s>size-1) s=size-1;
|
||||
// EventDataType v2=ar[s];
|
||||
// EventDataType v3=v2-val;
|
||||
// if (v3>0) {
|
||||
// val+=v3*diff;
|
||||
// }
|
||||
// Build sorted list of value/counts
|
||||
for (QMap<EventDataType, int>::iterator n=wmap.begin();n!=wmap.end();n++) {
|
||||
ValueCount vc;
|
||||
vc.value=n.key();
|
||||
vc.count=n.value();
|
||||
vc.p=0;
|
||||
valcnt.push_back(vc);
|
||||
}
|
||||
// sort by weight, then value
|
||||
qSort(valcnt);
|
||||
|
||||
//double SN=100.0/double(N); // 100% / overall sum
|
||||
double p=100.0*percentile;
|
||||
|
||||
double nth=double(SN)*percentile; // index of the position in the unweighted set would be
|
||||
double nthi=floor(nth);
|
||||
|
||||
int sum1=0,sum2=0;
|
||||
int w1,w2=0;
|
||||
EventDataType v1,v2;
|
||||
|
||||
int N=valcnt.size();
|
||||
int k=0;
|
||||
|
||||
for (k=0;k < N;k++) {
|
||||
v1=valcnt[k].value;
|
||||
w1=valcnt[k].count;
|
||||
sum1+=w1;
|
||||
|
||||
if (sum1 > nthi) {
|
||||
return v1;
|
||||
}
|
||||
if (sum1 == nthi){
|
||||
break; // boundary condition
|
||||
}
|
||||
}
|
||||
if (k>=N)
|
||||
return v1;
|
||||
|
||||
v2=valcnt[k+1].value;
|
||||
w2=valcnt[k+1].count;
|
||||
sum2=sum1+w2;
|
||||
// value lies between v1 and v2
|
||||
|
||||
double px=100.0/double(SN); // Percentile represented by one full value
|
||||
|
||||
// calculate percentile ranks
|
||||
double p1=px * (double(sum1)-(double(w1)/2.0));
|
||||
double p2=px * (double(sum2)-(double(w2)/2.0));
|
||||
|
||||
// calculate linear interpolation
|
||||
double v=v1 + ((p-p1)/(p2-p1)) * (v2-v1);
|
||||
|
||||
return v;
|
||||
|
||||
|
||||
// p1.....p.............p2
|
||||
// 37 55 70
|
||||
|
||||
// }
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
EventDataType Day::p90(ChannelID code)
|
||||
|
@ -155,6 +155,7 @@ public:
|
||||
protected:
|
||||
//! \brief A Vector containing all sessions for this day
|
||||
QVector<Session *> sessions;
|
||||
QHash<ChannelID, QHash<EventDataType, EventDataType> > perc_cache;
|
||||
qint64 d_first,d_last;
|
||||
private:
|
||||
bool d_firstsession;
|
||||
|
@ -22,6 +22,8 @@ EventList::EventList(EventListType et,EventDataType gain, EventDataType offset,
|
||||
m_update_minmax=false;
|
||||
}
|
||||
|
||||
m_data.reserve(2048);
|
||||
|
||||
// Reserve a few to increase performace??
|
||||
}
|
||||
EventList::~EventList()
|
||||
@ -112,17 +114,33 @@ void EventList::AddWaveform(qint64 start, qint16 * data, int recs, qint64 durati
|
||||
//double rate=duration/recs;
|
||||
|
||||
//realloc buffers.
|
||||
int r=m_count;
|
||||
m_count+=recs;
|
||||
m_data.reserve(m_count);
|
||||
m_data.resize(m_count);
|
||||
|
||||
EventStoreType *edata=m_data.data();
|
||||
|
||||
EventStoreType raw;
|
||||
EventDataType val;
|
||||
for (int i=0;i<recs;i++) {
|
||||
val=EventDataType(data[i])*m_gain+m_offset;
|
||||
qint16 * sp=data;
|
||||
EventStoreType * dp=&edata[r];
|
||||
|
||||
if (m_update_minmax) {
|
||||
for (int i=0;i<recs;i++) {
|
||||
raw=*sp++;
|
||||
val=EventDataType(raw)*m_gain+m_offset;
|
||||
if (m_min>val) m_min=val;
|
||||
if (m_max<val) m_max=val;
|
||||
*dp=raw;
|
||||
dp++;
|
||||
}
|
||||
} else {
|
||||
for (int i=0;i<recs;i++) {
|
||||
raw=*(sp++);
|
||||
val=EventDataType(raw)*m_gain+m_offset;
|
||||
*dp=raw;
|
||||
dp++;
|
||||
}
|
||||
m_data.push_back(data[i]);
|
||||
}
|
||||
|
||||
}
|
||||
@ -155,17 +173,33 @@ void EventList::AddWaveform(qint64 start, unsigned char * data, int recs, qint64
|
||||
// TODO: Check waveform chunk really is contiguos
|
||||
|
||||
//realloc buffers.
|
||||
int r=m_count;
|
||||
m_count+=recs;
|
||||
m_data.reserve(m_count);
|
||||
m_data.resize(m_count);
|
||||
|
||||
EventStoreType *edata=m_data.data();
|
||||
EventStoreType raw;
|
||||
EventDataType val;
|
||||
for (int i=0;i<recs;i++) {
|
||||
val=EventDataType(data[i])*m_gain+m_offset;
|
||||
|
||||
unsigned char * sp=data;
|
||||
EventStoreType * dp=&edata[r];
|
||||
|
||||
if (m_update_minmax) {
|
||||
for (int i=0;i<recs;i++) {
|
||||
raw=*sp++;
|
||||
val=EventDataType(val)*m_gain+m_offset;
|
||||
if (m_min>val) m_min=val;
|
||||
if (m_max<val) m_max=val;
|
||||
*dp=raw;
|
||||
dp++;
|
||||
}
|
||||
} else {
|
||||
for (int i=0;i<recs;i++) {
|
||||
raw=*sp++;
|
||||
val=EventDataType(val)*m_gain+m_offset;
|
||||
*dp=raw;
|
||||
dp++;
|
||||
}
|
||||
m_data.push_back(data[i]);
|
||||
}
|
||||
|
||||
}
|
||||
@ -199,17 +233,33 @@ void EventList::AddWaveform(qint64 start, char * data, int recs, qint64 duration
|
||||
// TODO: Check waveform chunk really is contiguos
|
||||
|
||||
//realloc buffers.
|
||||
m_count+=recs;
|
||||
m_data.reserve(m_count);
|
||||
|
||||
int r=m_count;
|
||||
m_count+=recs;
|
||||
m_data.resize(m_count);
|
||||
|
||||
EventStoreType *edata=m_data.data();
|
||||
EventStoreType raw;
|
||||
EventDataType val;
|
||||
for (int i=0;i<recs;i++) {
|
||||
val=EventDataType(data[i])*m_gain+m_offset;
|
||||
|
||||
char * sp=data;
|
||||
EventStoreType * dp=&edata[r];
|
||||
|
||||
if (m_update_minmax) {
|
||||
for (int i=0;i<recs;i++) {
|
||||
raw=*sp++;
|
||||
val=EventDataType(val)*m_gain+m_offset;
|
||||
if (m_min>val) m_min=val;
|
||||
if (m_max<val) m_max=val;
|
||||
*dp=raw;
|
||||
dp++;
|
||||
}
|
||||
} else {
|
||||
for (int i=0;i<recs;i++) {
|
||||
raw=*sp++;
|
||||
val=EventDataType(val)*m_gain+m_offset;
|
||||
*dp=raw;
|
||||
dp++;
|
||||
}
|
||||
m_data.push_back(data[i]);
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -138,6 +138,14 @@ public:
|
||||
|
||||
//! \brief Returns the time storage vector (only used in EVL_Event types)
|
||||
QVector<quint32> & getTime() { return m_time; }
|
||||
|
||||
// Don't mess with these without considering the consequences
|
||||
void rawDataResize(quint32 i) { m_data.resize(i); m_count=i; }
|
||||
void rawData2Resize(quint32 i) { m_data2.resize(i); m_count=i; }
|
||||
void rawTimeResize(quint32 i) { m_time.resize(i); m_count=i; }
|
||||
EventStoreType * rawData() { return m_data.data(); }
|
||||
EventStoreType * rawData2() { return m_data2.data(); }
|
||||
quint32 * rawTime() { return m_time.data(); }
|
||||
protected:
|
||||
|
||||
//! \brief The time storage vector, in 32bits delta format, added as offsets to m_first
|
||||
|
@ -324,11 +324,11 @@ Machine *CMS50Loader::CreateMachine(Profile *profile)
|
||||
m->SetClass(cms50_class_name);
|
||||
m->properties[STR_PROP_Brand]="Contec";
|
||||
m->properties[STR_PROP_Model]="CMS50X";
|
||||
QString a;
|
||||
a.sprintf("%i",cms50_data_version);
|
||||
m->properties[STR_PROP_DataVersion]=a;
|
||||
m->properties[STR_PROP_DataVersion]=QString::number(cms50_data_version);
|
||||
|
||||
profile->AddMachine(m);
|
||||
m->properties[STR_PROP_Path]="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+m->hexid()+"/";
|
||||
QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+m->hexid()+"/";
|
||||
m->properties[STR_PROP_Path]=path;
|
||||
|
||||
return m;
|
||||
}
|
||||
|
@ -399,7 +399,11 @@ Machine *IntellipapLoader::CreateMachine(QString serial,Profile *profile)
|
||||
profile->AddMachine(m);
|
||||
|
||||
m->properties[STR_PROP_Serial]=serial;
|
||||
m->properties[STR_PROP_Path]="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/";
|
||||
m->properties[STR_PROP_DataVersion]=QString::number(intellipap_data_version);
|
||||
|
||||
QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/";
|
||||
m->properties[STR_PROP_Path]=path;
|
||||
m->properties[STR_PROP_BackupPath]=path+"Backup/";
|
||||
|
||||
return m;
|
||||
}
|
||||
|
@ -152,7 +152,11 @@ Machine *PRS1Loader::CreateMachine(QString serial,Profile *profile)
|
||||
profile->AddMachine(m);
|
||||
|
||||
m->properties[STR_PROP_Serial]=serial;
|
||||
m->properties[STR_PROP_Path]="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/";;
|
||||
m->properties[STR_PROP_DataVersion]=QString::number(prs1_data_version);
|
||||
|
||||
QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/";
|
||||
m->properties[STR_PROP_Path]=path;
|
||||
m->properties[STR_PROP_BackupPath]=path+"Backup/";
|
||||
|
||||
return m;
|
||||
}
|
||||
@ -419,7 +423,7 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile)
|
||||
|
||||
}
|
||||
|
||||
m->properties[STR_PROP_DataVersion]=QString().sprintf("%i",prs1_data_version);
|
||||
m->properties[STR_PROP_DataVersion]=QString::number(prs1_data_version);
|
||||
m->properties[STR_PROP_LastImported]=QDateTime::currentDateTime().toString(Qt::ISODate);
|
||||
m->Save(); // Save any new sessions to disk in our format
|
||||
if (qprogress) qprogress->setValue(100);
|
||||
|
@ -198,21 +198,45 @@ bool EDFParser::Parse()
|
||||
}
|
||||
bool EDFParser::Open(QString name)
|
||||
{
|
||||
|
||||
//Urk.. This needs fixing for VC++, as it doesn't have packed attribute type..
|
||||
|
||||
if (name.endsWith(".gz")) {
|
||||
filename=name.mid(0,-3);
|
||||
QFile fi(name);
|
||||
fi.open(QFile::ReadOnly);
|
||||
fi.seek(fi.size()-4);
|
||||
unsigned char ch[4];
|
||||
fi.read((char *)ch,4);
|
||||
filesize=ch[0] | (ch [1] << 8) | (ch[2] << 16) | (ch[3] << 24);
|
||||
datasize=filesize-EDFHeaderSize;
|
||||
if (datasize<0) return false;
|
||||
qDebug() << "Size of" << name << "uncompressed=" << filesize;
|
||||
gzFile f=gzopen(name.toAscii(),"rb");
|
||||
if (!f) {
|
||||
qDebug() << "EDFParser::Open() Couldn't open file" << name;
|
||||
return false;
|
||||
}
|
||||
gzread(f,(char *)&header,EDFHeaderSize);
|
||||
buffer=new char [datasize];
|
||||
gzbuffer(f,65536*2);
|
||||
gzread(f,buffer,datasize);
|
||||
gzclose(f);
|
||||
} else {
|
||||
QFile f(name);
|
||||
if (!f.open(QIODevice::ReadOnly)) return false;
|
||||
if (!f.isReadable()) return false;
|
||||
if (!f.open(QIODevice::ReadOnly))
|
||||
return false;
|
||||
filename=name;
|
||||
filesize=f.size();
|
||||
datasize=filesize-EDFHeaderSize;
|
||||
if (datasize<0) return false;
|
||||
|
||||
//Urk.. This needs fixing for VC++, as it doesn't have packed attribute type..
|
||||
|
||||
f.read((char *)&header,EDFHeaderSize);
|
||||
//qDebug() << "Opening " << name;
|
||||
buffer=new char [datasize];
|
||||
f.read(buffer,datasize);
|
||||
f.close();
|
||||
}
|
||||
pos=0;
|
||||
return true;
|
||||
}
|
||||
@ -248,10 +272,11 @@ Machine *ResmedLoader::CreateMachine(QString serial,Profile *profile)
|
||||
|
||||
m->properties[STR_PROP_Serial]=serial;
|
||||
m->properties[STR_PROP_Brand]=STR_MACH_ResMed;
|
||||
QString a;
|
||||
a.sprintf("%i",resmed_data_version);
|
||||
m->properties[STR_PROP_DataVersion]=a;
|
||||
m->properties[STR_PROP_Path]="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/";
|
||||
m->properties[STR_PROP_DataVersion]=QString::number(resmed_data_version);
|
||||
|
||||
QString path="{"+STR_GEN_DataFolder+"}/"+m->GetClass()+"_"+serial+"/";
|
||||
m->properties[STR_PROP_Path]=path;
|
||||
m->properties[STR_PROP_BackupPath]=path+"Backup/";
|
||||
|
||||
return m;
|
||||
|
||||
@ -268,6 +293,7 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
const QString ext_TGT="tgt";
|
||||
const QString ext_CRC="crc";
|
||||
const QString ext_EDF="edf";
|
||||
const QString ext_gz=".gz";
|
||||
|
||||
QString serial; // Serial number
|
||||
QString key,value;
|
||||
@ -330,16 +356,31 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
// Early check for STR.edf file, so we can early exit before creating faulty machine record.
|
||||
QString strpath=path+strfile+ext_EDF; // STR.edf file
|
||||
f.setFileName(strpath);
|
||||
if (!f.exists()) { // No STR.edf.. Do we have a STR.edf.gz?
|
||||
strpath+=ext_gz;
|
||||
f.setFileName(strpath);
|
||||
if (!f.exists()) {
|
||||
qDebug() << "Missing STR.edf file";
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
// Create machine object (unless it's already registered)
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
Machine *m=CreateMachine(serial,profile);
|
||||
|
||||
bool create_backups=PROFILE.session->backupCardData();
|
||||
bool compress_backups=PROFILE.session->compressBackupData();
|
||||
|
||||
QString backup_path=PROFILE.Get(m->properties[STR_PROP_BackupPath]);
|
||||
if (backup_path.isEmpty())
|
||||
backup_path=PROFILE.Get(m->properties[STR_PROP_Path])+"Backup/";
|
||||
|
||||
if (path==backup_path) {
|
||||
create_backups=false;
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
// Parse the idmap into machine objects properties, (overwriting any old values)
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
@ -369,10 +410,12 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
// Creating early as we need the object
|
||||
QDir dir(newpath);
|
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
// Create the backup folder for storing a copy of everything in..
|
||||
// (Unless we are importing from this backup folder)
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
QString backup_path=PROFILE.Get(m->properties[STR_PROP_Path])+"Backup/";
|
||||
if (create_backups) {
|
||||
if (!dir.exists(backup_path)) {
|
||||
if (!dir.mkpath(backup_path+datalog)) {
|
||||
qDebug() << "Could not create S9 backup directory :-/";
|
||||
@ -383,9 +426,19 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
QFile::copy(path+idfile+ext_TGT,backup_path+idfile+ext_TGT);
|
||||
QFile::copy(path+idfile+ext_CRC,backup_path+idfile+ext_CRC);
|
||||
|
||||
// Copy STR files to backup folder
|
||||
|
||||
//copy STR files to backup folder
|
||||
if (strpath.endsWith(ext_gz)) // Already compressed.
|
||||
QFile::copy(strpath,backup_path+strfile+ext_EDF+ext_gz);
|
||||
else { // Compress STR file to backup folder
|
||||
compress_backups ?
|
||||
compressFile(strpath,backup_path+strfile+ext_EDF)
|
||||
:
|
||||
QFile::copy(strpath,backup_path+strfile+ext_EDF);
|
||||
QFile::copy(path+strfile+ext_CRC,backup_path+strfile+ext_CRC);
|
||||
}
|
||||
|
||||
QFile::copy(path+"STR.crc",backup_path+"STR.crc");
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
// Process the actual STR.edf data
|
||||
@ -475,14 +528,14 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
dir.setSorting(QDir::Name);
|
||||
QFileInfoList flist=dir.entryInfoList();
|
||||
|
||||
QString ext,rest,datestr;//,s,codestr;
|
||||
QString datestr;
|
||||
SessionID sessionid;
|
||||
QDateTime date;
|
||||
int size=flist.size();
|
||||
|
||||
sessfiles.clear();
|
||||
|
||||
// For each file in filelist...
|
||||
// For each file in flist...
|
||||
for (int i=0;i<size;i++) {
|
||||
QFileInfo fi=flist.at(i);
|
||||
filename=fi.fileName();
|
||||
@ -491,24 +544,28 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
if (!fi.isReadable())
|
||||
continue;
|
||||
|
||||
// Check the file extnsion
|
||||
ext=filename.section(".",1).toLower();
|
||||
if (ext!=ext_EDF) continue;
|
||||
// Accept only .edf and .edf.gz files
|
||||
if (!((filename.right(4).toLower() == "."+ext_EDF)
|
||||
|| (filename.right(7).toLower() == "."+ext_EDF+ext_gz)))
|
||||
continue;
|
||||
|
||||
// Split the filename into components
|
||||
rest=filename.section(".",0,0);
|
||||
// Extract the session date out of the filename
|
||||
datestr=filename.section("_",0,1);
|
||||
|
||||
// Take the filename's date, and convert it to epoch to form the sessionID.
|
||||
// Take the filename's date, and
|
||||
date=QDateTime::fromString(datestr,"yyyyMMdd_HHmmss");
|
||||
if (!date.isValid())
|
||||
continue; // Skip file if dates invalid
|
||||
|
||||
// Skip file if dates invalid, the filename is clearly wrong..
|
||||
if (!date.isValid())
|
||||
continue;
|
||||
|
||||
// convert this date to UNIX epoch to form the sessionID
|
||||
sessionid=date.toTime_t();
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////////////////
|
||||
// Resmed bugs up on the session filenames.. 1 or 2 seconds either way
|
||||
// Moral of the story, when writing firmware and saving in batches, use the same datetimes.
|
||||
// Moral of the story, when writing firmware and saving in batches, use the same datetimes,
|
||||
// and provide firmware updates for free to your customers.
|
||||
////////////////////////////////////////////////////////////////////////////////////////////
|
||||
if (sessfiles.find(sessionid)==sessfiles.end()) {
|
||||
if (sessfiles.find(sessionid+2)!=sessfiles.end()) sessionid+=2;
|
||||
@ -517,8 +574,8 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
else if (sessfiles.find(sessionid-2)!=sessfiles.end()) sessionid-=2;
|
||||
}
|
||||
|
||||
// Push current filename to sanitized by-session list
|
||||
sessfiles[sessionid].push_back(rest);
|
||||
// Push current filename to ordered-by-sessionid list
|
||||
sessfiles[sessionid].push_back(filename);
|
||||
|
||||
// Update the progress bar
|
||||
if (qprogress) qprogress->setValue((float(i+1)/float(size)*10.0));
|
||||
@ -536,6 +593,7 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
// Scan over file list and knock out of dayused list
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
int dn;
|
||||
for (QMap<SessionID,QVector<QString> >::iterator si=sessfiles.begin();si!=sessfiles.end();si++) {
|
||||
sessionid=si.key();
|
||||
|
||||
@ -544,17 +602,18 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
if (edn<0) edn=0;
|
||||
|
||||
// Find real day number from str.edf mask on/off data.
|
||||
int dn=-1;
|
||||
dn=-1;
|
||||
for (int j=edn;j<strfirst.size();j++){
|
||||
time_t st=strfirst.at(j);
|
||||
time_t et=strlast.at(j);
|
||||
st=strfirst.at(j);
|
||||
if (sessionid>=st) {
|
||||
et=strlast.at(j);
|
||||
if (sessionid<(et+300)) {
|
||||
dn=j;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
// If found, mark day off so STR.edf summary data isn't used instead of the real thing.
|
||||
if (dn>=0) {
|
||||
dayused[dn]=0;
|
||||
}
|
||||
@ -565,16 +624,21 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
// For all days not in session lists, (to get at days without data records)
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
for (int dn=0;dn<days;dn++) {
|
||||
if (!dayused[dn]) continue; // Skip days with loadable data.
|
||||
for (dn=0;dn<days;dn++) {
|
||||
// Skip days with loadable data.
|
||||
if (!dayused[dn])
|
||||
continue;
|
||||
|
||||
if (!daystarttimes.contains(dn)) continue;
|
||||
int j;
|
||||
int scnt=daystarttimes[dn].size();
|
||||
if (!daystarttimes.contains(dn))
|
||||
continue;
|
||||
|
||||
sess=NULL;
|
||||
// For each mask on/off segment.
|
||||
for (j=0;j<scnt;j++) {
|
||||
|
||||
int scnt=daystarttimes[dn].size(); // count of sessions for this day
|
||||
|
||||
// Create a new session for each mask on/off segment in a day
|
||||
// But only populate the last one with summary data.
|
||||
for (int j=0;j<scnt;j++) {
|
||||
st=daystarttimes[dn].at(j);
|
||||
|
||||
// Skip if session already exists
|
||||
@ -583,13 +647,12 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
|
||||
et=dayendtimes[dn].at(j);
|
||||
|
||||
// Create session
|
||||
// Create the session
|
||||
sess=new Session(m,st);
|
||||
sess->really_set_first(qint64(st)*1000L);
|
||||
sess->really_set_last(qint64(et)*1000L);
|
||||
sess->SetChanged(true);
|
||||
m->AddSession(sess,profile);
|
||||
|
||||
}
|
||||
// Add the actual data to the last session
|
||||
EventDataType tmp,dur;
|
||||
@ -693,12 +756,14 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->settings[CPAP_PressureMax]=pressure;
|
||||
//sess->setMax(CPAP_Pressure,pressure);
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
EventDataType valmed=0,valmax=0,val95=0;
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Leak Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("Leak Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_Leak]=sig->gain*60.0;
|
||||
@ -714,6 +779,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->m_valuesummary[CPAP_Leak][valmax]=4;
|
||||
}
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Minute Ventilation Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("Min Vent Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_MinuteVent]=sig->gain;
|
||||
@ -728,6 +796,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->setMax(CPAP_MinuteVent,valmax*sig->gain);
|
||||
sess->m_valuesummary[CPAP_MinuteVent][valmax]=4;
|
||||
}
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Respiratory Rate Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("RR Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_RespRate]=sig->gain;
|
||||
@ -743,6 +814,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->m_valuesummary[CPAP_RespRate][valmax]=4;
|
||||
}
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Tidal Volume Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("Tid Vol Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_TidalVolume]=sig->gain*1000.0;
|
||||
@ -758,6 +832,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->m_valuesummary[CPAP_TidalVolume][valmax]=4;
|
||||
}
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Target Minute Ventilation Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("Targ Vent Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_TgMV]=sig->gain;
|
||||
@ -774,6 +851,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
}
|
||||
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// I:E Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("I:E Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_IE]=sig->gain;
|
||||
@ -789,6 +869,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->m_valuesummary[CPAP_IE][valmax]=4;
|
||||
}
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Mask Pressure Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("Mask Pres Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_Pressure]=sig->gain;
|
||||
@ -804,6 +887,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->m_valuesummary[CPAP_Pressure][valmax]=4;
|
||||
}
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Inspiratory Pressure (IPAP) Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("Insp Pres Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_IPAP]=sig->gain;
|
||||
@ -818,6 +904,9 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->setMax(CPAP_IPAP,valmax*sig->gain);
|
||||
sess->m_valuesummary[CPAP_IPAP][valmax]=4;
|
||||
}
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Expiratory Pressure (EPAP) Summary
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("Exp Pres Med"))) {
|
||||
valmed=sig->data[dn];
|
||||
sess->m_gain[CPAP_EPAP]=sig->gain;
|
||||
@ -833,33 +922,39 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
sess->m_valuesummary[CPAP_EPAP][valmax]=4;
|
||||
}
|
||||
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
// Duration and Event Indices
|
||||
/////////////////////////////////////////////////////////////////////
|
||||
if ((sig=stredf.lookupName("Mask Dur"))) {
|
||||
dur=sig->data[dn]*sig->gain;
|
||||
dur/=60.0f; // convert to hours.
|
||||
}
|
||||
if ((sig=stredf.lookupName("OAI"))) {
|
||||
if ((sig=stredf.lookupName("OAI"))) { // Obstructive Apnea Index
|
||||
tmp=sig->data[dn]*sig->gain;
|
||||
sess->setCph(CPAP_Obstructive,tmp);
|
||||
sess->setCount(CPAP_Obstructive,tmp*(dur/60.0));
|
||||
sess->setCount(CPAP_Obstructive,tmp*dur); // Converting from indice to counts..
|
||||
}
|
||||
if ((sig=stredf.lookupName("HI"))) {
|
||||
if ((sig=stredf.lookupName("HI"))) { // Hypopnea Index
|
||||
tmp=sig->data[dn]*sig->gain;
|
||||
sess->setCph(CPAP_Hypopnea,tmp);
|
||||
sess->setCount(CPAP_Hypopnea,tmp*(dur/60.0));
|
||||
sess->setCount(CPAP_Hypopnea,tmp*dur);
|
||||
}
|
||||
if ((sig=stredf.lookupName("UAI"))) {
|
||||
if ((sig=stredf.lookupName("UAI"))) { // Unspecified Apnea Index
|
||||
tmp=sig->data[dn]*sig->gain;
|
||||
sess->setCph(CPAP_Apnea,tmp);
|
||||
sess->setCount(CPAP_Apnea,tmp*(dur/60.0));
|
||||
sess->setCount(CPAP_Apnea,tmp*dur);
|
||||
}
|
||||
if ((sig=stredf.lookupName("CAI"))) {
|
||||
if ((sig=stredf.lookupName("CAI"))) { // "Central" Apnea Index
|
||||
tmp=sig->data[dn]*sig->gain;
|
||||
sess->setCph(CPAP_ClearAirway,tmp);
|
||||
sess->setCount(CPAP_ClearAirway,tmp*(dur/60.0));
|
||||
sess->setCount(CPAP_ClearAirway,tmp*dur);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
bool gz;
|
||||
backup_path+=datalog+"/";
|
||||
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
// Scan through new file list and import sessions
|
||||
@ -873,18 +968,31 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
|
||||
// Create the session
|
||||
sess=new Session(m,sessionid);
|
||||
QString filename,fullpath,backupfile,backfile, crcfile;
|
||||
|
||||
// Process EDF File List
|
||||
for (int i=0;i<si.value().size();++i) {
|
||||
QString filename=si.value()[i]+".";
|
||||
QString fullpath=newpath+filename;
|
||||
filename=si.value()[i];
|
||||
gz=(filename.right(3).toLower()==ext_gz);
|
||||
fullpath=newpath+filename;
|
||||
|
||||
// Copy the EDF file to the backup folder
|
||||
QString backup_file=backup_path+datalog+"/"+filename;
|
||||
QFile().copy(fullpath+ext_EDF, backup_file+ext_EDF);
|
||||
QFile().copy(fullpath+ext_CRC, backup_file+ext_CRC);
|
||||
if (create_backups) {
|
||||
backupfile=backup_path+filename;
|
||||
if (!gz) {
|
||||
compress_backups ?
|
||||
compressFile(fullpath, backupfile)
|
||||
:
|
||||
QFile::copy(fullpath, backupfile);
|
||||
} else // already compressed, just copy it.
|
||||
QFile::copy(fullpath, backupfile);
|
||||
|
||||
backfile=filename.replace(".edf",".crc",Qt::CaseInsensitive);
|
||||
backupfile=backup_path+backfile;
|
||||
crcfile=newpath+backfile;
|
||||
QFile::copy(crcfile, backupfile);
|
||||
}
|
||||
|
||||
fullpath+=ext_EDF;
|
||||
EDFParser edf(fullpath);
|
||||
|
||||
// Parse the actual file
|
||||
@ -896,12 +1004,12 @@ int ResmedLoader::Open(QString & path,Profile *profile)
|
||||
qDebug() << "edf Serial number doesn't match Identification.tgt";
|
||||
}
|
||||
|
||||
fn=fullpath.section("_",-1).toLower();
|
||||
fn=filename.section("_",-1).section(".",0,0).toLower();
|
||||
|
||||
if (fn=="eve.edf") LoadEVE(sess,edf);
|
||||
else if (fn=="pld.edf") LoadPLD(sess,edf);
|
||||
else if (fn=="brp.edf") LoadBRP(sess,edf);
|
||||
else if (fn=="sad.edf") LoadSAD(sess,edf);
|
||||
if (fn=="eve") LoadEVE(sess,edf);
|
||||
else if (fn=="pld") LoadPLD(sess,edf);
|
||||
else if (fn=="brp") LoadBRP(sess,edf);
|
||||
else if (fn=="sad") LoadSAD(sess,edf);
|
||||
}
|
||||
if (qprogress) qprogress->setValue(10.0+(float(++cnt)/float(size)*90.0));
|
||||
QApplication::processEvents();
|
||||
|
@ -28,6 +28,7 @@ class BoundsError {};
|
||||
class OldDBVersion {};
|
||||
|
||||
const quint32 magic=0xC73216AB; // Magic number for Sleepyhead Data Files.. Don't touch!
|
||||
const quint32 compmagic=0xC73216AC; // Magic number for Compressed Sleepyhead Data Files.. Don't touch!
|
||||
|
||||
//const int max_number_event_fields=10;
|
||||
// This should probably move somewhere else
|
||||
|
@ -44,6 +44,45 @@ MachineLoader::~MachineLoader()
|
||||
delete *m;
|
||||
}
|
||||
}
|
||||
|
||||
bool MachineLoader::compressFile(QString inpath, QString outpath)
|
||||
{
|
||||
if (outpath.isEmpty())
|
||||
outpath=inpath+".gz";
|
||||
else if (!outpath.endsWith(".gz")) {
|
||||
outpath+=".gz";
|
||||
}
|
||||
|
||||
QFile f(inpath);
|
||||
if (!f.exists(inpath)) {
|
||||
qDebug() << "compressFile()" << inpath << "does not exist";
|
||||
return false;
|
||||
}
|
||||
qint64 size=f.size();
|
||||
if (!f.open(QFile::ReadOnly)) {
|
||||
qDebug() << "compressFile() Couldn't open" << inpath;
|
||||
return false;
|
||||
}
|
||||
char * buf=new char [size];
|
||||
if (!f.read(buf,size)) {
|
||||
delete buf;
|
||||
qDebug() << "compressFile() Couldn't read all of" << inpath;
|
||||
return false;
|
||||
}
|
||||
f.close();
|
||||
gzFile gz=gzopen(outpath.toAscii(),"wb");
|
||||
gzbuffer(gz,65536*2);
|
||||
if (!gz) {
|
||||
qDebug() << "compressFile() Couldn't open" << outpath <<"for writing";
|
||||
delete buf;
|
||||
return false;
|
||||
}
|
||||
gzwrite(gz,buf,size);
|
||||
gzclose(gz);
|
||||
delete buf;
|
||||
return true;
|
||||
}
|
||||
|
||||
/*const QString machine_profile_name="MachineList.xml";
|
||||
|
||||
void MachineLoader::LoadMachineList()
|
||||
|
@ -11,6 +11,7 @@ License: GPL
|
||||
#define MACHINE_LOADER_H
|
||||
#include "profiles.h"
|
||||
#include "machine.h"
|
||||
#include "zlib.h"
|
||||
|
||||
/*! \class MachineLoader
|
||||
\brief Base class to derive a new Machine importer from
|
||||
@ -33,6 +34,8 @@ public:
|
||||
//! \brief Override to returns the class name of this MachineLoader
|
||||
virtual const QString & ClassName()=0;
|
||||
|
||||
bool compressFile(QString inpath, QString outpath="");
|
||||
|
||||
|
||||
/*
|
||||
MachineLoader(Profile *profile,QString & classname);
|
||||
|
@ -13,6 +13,7 @@ License: GPL
|
||||
#include <QMessageBox>
|
||||
#include <QDebug>
|
||||
#include <algorithm>
|
||||
#include <cmath>
|
||||
|
||||
#include "preferences.h"
|
||||
#include "profiles.h"
|
||||
@ -27,6 +28,8 @@ Preferences *p_pref;
|
||||
Preferences *p_layout;
|
||||
Profile * p_profile;
|
||||
|
||||
|
||||
|
||||
Profile::Profile()
|
||||
:Preferences(),is_first_day(true)
|
||||
{
|
||||
@ -428,7 +431,7 @@ Profile *Create(QString name)
|
||||
prof->user->setUserName(name);
|
||||
//prof->Set("Realname",realname);
|
||||
//if (!password.isEmpty()) prof.user->setPassword(password);
|
||||
prof->Set(STR_GEN_DataFolder,QString("{home}/Profiles/{")+QString(UI_STR_UserName)+QString("}"));
|
||||
prof->Set(STR_GEN_DataFolder,QString("{home}/Profiles/{")+QString(STR_UI_UserName)+QString("}"));
|
||||
|
||||
Machine *m=new Machine(prof,0);
|
||||
m->SetClass("Journal");
|
||||
@ -504,80 +507,6 @@ void Scan()
|
||||
|
||||
} // namespace Profiles
|
||||
|
||||
// DoctorInfo Strings
|
||||
const char * DI_STR_Name="DoctorName";
|
||||
const char * DI_STR_Phone="DoctorPhone";
|
||||
const char * DI_STR_Email="DoctorEmail";
|
||||
const char * DI_STR_Practice="DoctorPractice";
|
||||
const char * DI_STR_Address="DoctorAddress";
|
||||
const char * DI_STR_PatientID="DoctorPatientID";
|
||||
|
||||
// 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";
|
||||
|
||||
// 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";
|
||||
|
||||
// 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";
|
||||
|
||||
// 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";
|
||||
const char * IS_STR_TrashDayCache="TrashDayCache";
|
||||
const char * IS_STR_ShowSerialNumbers="ShowSerialNumbers";
|
||||
|
||||
// AppearanceSettings Strings
|
||||
const char * AS_STR_GraphHeight="GraphHeight";
|
||||
const char * AS_STR_AntiAliasing="UseAntiAliasing";
|
||||
const char * AS_STR_GraphSnapshots="EnableGraphSnapshots";
|
||||
const char * AS_STR_Animations="AnimationsAndTransitions";
|
||||
const char * AS_STR_SquareWave="SquareWavePlots";
|
||||
const char * AS_STR_OverlayType="OverlayType";
|
||||
|
||||
// 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";
|
||||
const char * US_STR_CalculateRDI="CalculateRDI";
|
||||
|
||||
int Profile::countDays(MachineType mt, QDate start, QDate end)
|
||||
{
|
||||
if (!start.isValid())
|
||||
@ -778,17 +707,14 @@ EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, Mac
|
||||
|
||||
QDate date=start;
|
||||
|
||||
// This is one messy function.. It requires all data to be loaded.. :(
|
||||
|
||||
QHash<EventStoreType,EventStoreType> summary;
|
||||
QHash<EventStoreType,EventStoreType>::iterator sumi;
|
||||
QVector<EventDataType> array;
|
||||
QMap<EventDataType, int> wmap;
|
||||
|
||||
QHash<ChannelID,QHash<EventStoreType, EventStoreType> >::iterator vsi;
|
||||
EventDataType val,gain;
|
||||
bool setgain=false;
|
||||
EventDataType weight,value;
|
||||
|
||||
//double val=0,tmp,hours=0;
|
||||
int SN=0;
|
||||
do {
|
||||
Day * day=GetGoodDay(date,mt);
|
||||
if (day) {
|
||||
@ -800,81 +726,86 @@ EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, Mac
|
||||
Session *sess=*s;
|
||||
gain=sess->m_gain[code];
|
||||
if (!gain) gain=1;
|
||||
setgain=true;
|
||||
vsi=sess->m_valuesummary.find(code);
|
||||
if (vsi==sess->m_valuesummary.end()) continue;
|
||||
|
||||
QHash<EventStoreType, EventStoreType> & vsum=vsi.value();
|
||||
|
||||
for (QHash<EventStoreType, EventStoreType>::iterator k=vsum.begin();k!=vsum.end();k++) {
|
||||
for (int z=0;z<k.value();z++) {
|
||||
val=k.key();
|
||||
val*=gain;
|
||||
weight=k.value();
|
||||
value=EventDataType(k.key())*gain;
|
||||
|
||||
array.push_back(val);
|
||||
SN+=weight;
|
||||
if (wmap.contains(value)) {
|
||||
wmap[value]+=weight;
|
||||
} else {
|
||||
wmap[value]=weight;
|
||||
}
|
||||
/*sumi=summary.find(k.key());
|
||||
if (sumi==summary.end()) summary[k.key()]=k.value();
|
||||
else {
|
||||
sumi.value()+=k.value();
|
||||
}*/
|
||||
}
|
||||
/*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));
|
||||
}
|
||||
}*/
|
||||
}
|
||||
}
|
||||
}
|
||||
date=date.addDays(1);
|
||||
} while (date<=end);
|
||||
// for (QHash<EventStoreType, EventStoreType>::iterator k=summary.begin();k!=summary.end();k++) {
|
||||
// for (int i=0;i<k.value();i++) {
|
||||
// array.push_back(k.key());
|
||||
// }
|
||||
// }
|
||||
QVector<ValueCount> valcnt;
|
||||
|
||||
/*if (array.size()==0) return 0;
|
||||
qSort(array);
|
||||
// Build sorted list of value/counts
|
||||
for (QMap<EventDataType, int>::iterator n=wmap.begin();n!=wmap.end();n++) {
|
||||
ValueCount vc;
|
||||
vc.value=n.key();
|
||||
vc.count=n.value();
|
||||
vc.p=0;
|
||||
valcnt.push_back(vc);
|
||||
}
|
||||
// sort by weight, then value
|
||||
qSort(valcnt);
|
||||
|
||||
//double SN=100.0/double(N); // 100% / overall sum
|
||||
double p=100.0*percent;
|
||||
|
||||
int idx=array.size()*percent;
|
||||
if (idx>array.size()-1) idx=array.size()-1;
|
||||
return array[idx]; */
|
||||
double nth=double(SN)*percent; // index of the position in the unweighted set would be
|
||||
double nthi=floor(nth);
|
||||
|
||||
QVector<EventDataType>::iterator first=array.begin();
|
||||
QVector<EventDataType>::iterator last=array.end();
|
||||
QVector<EventDataType>::iterator middle = first + int((last-first) * percent);
|
||||
std::nth_element(first,middle,last);
|
||||
val=*middle;
|
||||
int sum1=0,sum2=0;
|
||||
int w1,w2=0;
|
||||
EventDataType v1,v2;
|
||||
|
||||
return val;
|
||||
// int size=array.size();
|
||||
// if (!size)
|
||||
// return 0;
|
||||
// size--;
|
||||
// qSort(array);
|
||||
// int p=EventDataType(size)*percent;
|
||||
// float p2=EventDataType(size)*percent;
|
||||
// float diff=p2-p;
|
||||
// val=array[p];
|
||||
// if (diff>0) {
|
||||
// int s=p+1;
|
||||
// if (s>size-1) s=size-1;
|
||||
// EventDataType v2=array[s];
|
||||
// EventDataType v3=v2-val;
|
||||
// if (v3>0) {
|
||||
// val+=v3*diff;
|
||||
// }
|
||||
int N=valcnt.size();
|
||||
int k=0;
|
||||
|
||||
// }
|
||||
for (k=0;k < N;k++) {
|
||||
v1=valcnt[k].value;
|
||||
w1=valcnt[k].count;
|
||||
sum1+=w1;
|
||||
|
||||
// return val;
|
||||
if (sum1 > nthi) {
|
||||
return v1;
|
||||
}
|
||||
if (sum1 == nthi){
|
||||
break; // boundary condition
|
||||
}
|
||||
}
|
||||
if (k>=N)
|
||||
return v1;
|
||||
|
||||
v2=valcnt[k+1].value;
|
||||
w2=valcnt[k+1].count;
|
||||
sum2=sum1+w2;
|
||||
// value lies between v1 and v2
|
||||
|
||||
double px=100.0/double(SN); // Percentile represented by one full value
|
||||
|
||||
// calculate percentile ranks
|
||||
double p1=px * (double(sum1)-(double(w1)/2.0));
|
||||
double p2=px * (double(sum2)-(double(w2)/2.0));
|
||||
|
||||
// calculate linear interpolation
|
||||
double v=v1 + ((p-p1)/(p2-p1)) * (v2-v1);
|
||||
|
||||
// p1.....p.............p2
|
||||
// 37 55 70
|
||||
|
||||
return v;
|
||||
}
|
||||
|
||||
QDate Profile::FirstDay(MachineType mt)
|
||||
@ -929,3 +860,81 @@ QDate Profile::LastGoodDay(MachineType mt)
|
||||
} while (d>=f);
|
||||
return f; //m_first;
|
||||
}
|
||||
|
||||
|
||||
// DoctorInfo Strings
|
||||
const char * STR_DI_Name="DoctorName";
|
||||
const char * STR_DI_Phone="DoctorPhone";
|
||||
const char * STR_DI_Email="DoctorEmail";
|
||||
const char * STR_DI_Practice="DoctorPractice";
|
||||
const char * STR_DI_Address="DoctorAddress";
|
||||
const char * STR_DI_PatientID="DoctorPatientID";
|
||||
|
||||
// UserInfo Strings
|
||||
const char * STR_UI_DOB="DOB";
|
||||
const char * STR_UI_FirstName="FirstName";
|
||||
const char * STR_UI_LastName="LastName";
|
||||
const char * STR_UI_UserName="UserName";
|
||||
const char * STR_UI_Password="Password";
|
||||
const char * STR_UI_Address="Address";
|
||||
const char * STR_UI_Phone="Phone";
|
||||
const char * STR_UI_EmailAddress="EmailAddress";
|
||||
const char * STR_UI_Country="Country";
|
||||
const char * STR_UI_Height="Height";
|
||||
const char * STR_UI_Gender="Gender";
|
||||
const char * STR_UI_TimeZone="TimeZone";
|
||||
const char * STR_UI_Language="Language";
|
||||
const char * STR_UI_DST="DST";
|
||||
|
||||
// OxiSettings Strings
|
||||
const char * STR_OS_EnableOximetry="EnableOximetry";
|
||||
const char * STR_OS_SyncOximetry="SyncOximetry";
|
||||
const char * STR_OS_OximeterType="OximeterType";
|
||||
const char * STR_OS_OxiDiscardThreshold="OxiDiscardThreshold";
|
||||
const char * STR_OS_SPO2DropDuration="SPO2DropDuration";
|
||||
const char * STR_OS_SPO2DropPercentage="SPO2DropPercentage";
|
||||
const char * STR_OS_PulseChangeDuration="PulseChangeDuration";
|
||||
const char * STR_OS_PulseChangeBPM="PulseChangeBPM";
|
||||
|
||||
// CPAPSettings Strings
|
||||
const char * STR_CS_ComplianceHours="ComplianceHours";
|
||||
const char * STR_CS_ShowCompliance="ShowCompliance";
|
||||
const char * STR_CS_ShowLeaksMode="ShowLeaksMode";
|
||||
const char * STR_CS_MaskStartDate="MaskStartDate";
|
||||
const char * STR_CS_MaskDescription="MaskDescription";
|
||||
const char * STR_CS_MaskType="MaskType";
|
||||
const char * STR_CS_PrescribedMode="CPAPPrescribedMode";
|
||||
const char * STR_CS_PrescribedMinPressure="CPAPPrescribedMinPressure";
|
||||
const char * STR_CS_PrescribedMaxPressure="CPAPPrescribedMaxPressure";
|
||||
const char * STR_CS_UntreatedAHI="UntreatedAHI";
|
||||
const char * STR_CS_Notes="CPAPNotes";
|
||||
const char * STR_CS_DateDiagnosed="DateDiagnosed";
|
||||
|
||||
// ImportSettings Strings
|
||||
const char * STR_IS_DaySplitTime="DaySplitTime";
|
||||
const char * STR_IS_CacheSessions="MemoryHog";
|
||||
const char * STR_IS_CombineCloseSessions="CombineCloserSessions";
|
||||
const char * STR_IS_IgnoreShorterSessions="IgnoreShorterSessions";
|
||||
const char * STR_IS_Multithreading="EnableMultithreading";
|
||||
const char * STR_IS_BackupCardData="BackupCardData";
|
||||
const char * STR_IS_CompressBackupData="CompressBackupData";
|
||||
const char * STR_IS_CompressSessionData="CompressSessionData";
|
||||
|
||||
// AppearanceSettings Strings
|
||||
const char * STR_AS_GraphHeight="GraphHeight";
|
||||
const char * STR_AS_AntiAliasing="UseAntiAliasing";
|
||||
const char * STR_AS_GraphSnapshots="EnableGraphSnapshots";
|
||||
const char * STR_AS_Animations="AnimationsAndTransitions";
|
||||
const char * STR_AS_SquareWave="SquareWavePlots";
|
||||
const char * STR_AS_OverlayType="OverlayType";
|
||||
|
||||
// UserSettings Strings
|
||||
const char * STR_US_UnitSystem="UnitSystem";
|
||||
const char * STR_US_EventWindowSize="EventWindowSize";
|
||||
const char * STR_US_SkipEmptyDays="SkipEmptyDays";
|
||||
const char * STR_US_RebuildCache="RebuildCache";
|
||||
const char * STR_US_ShowDebug="ShowDebug";
|
||||
const char * STR_US_LinkGroups="LinkGroups";
|
||||
const char * STR_US_CalculateRDI="CalculateRDI";
|
||||
const char * STR_US_ShowSerialNumbers="ShowSerialNumbers";
|
||||
|
||||
|
@ -151,59 +151,59 @@ extern Profile * p_profile;
|
||||
#define LAYOUT (*p_layout)
|
||||
#define PROFILE (*p_profile)
|
||||
|
||||
extern const char * DI_STR_Name;
|
||||
extern const char * DI_STR_Phone;
|
||||
extern const char * DI_STR_Email;
|
||||
extern const char * DI_STR_Practice;
|
||||
extern const char * DI_STR_Address;
|
||||
extern const char * DI_STR_PatientID;
|
||||
extern const char * STR_DI_Name;
|
||||
extern const char * STR_DI_Phone;
|
||||
extern const char * STR_DI_Email;
|
||||
extern const char * STR_DI_Practice;
|
||||
extern const char * STR_DI_Address;
|
||||
extern const char * STR_DI_PatientID;
|
||||
|
||||
class DoctorInfo
|
||||
{
|
||||
public:
|
||||
DoctorInfo(Profile *p) : m_profile(p)
|
||||
{
|
||||
if (!m_profile->contains(DI_STR_Name)) (*m_profile)[DI_STR_Name]=QString();
|
||||
if (!m_profile->contains(DI_STR_Phone)) (*m_profile)[DI_STR_Phone]=QString();
|
||||
if (!m_profile->contains(DI_STR_Email)) (*m_profile)[DI_STR_Email]=QString();
|
||||
if (!m_profile->contains(DI_STR_Practice)) (*m_profile)[DI_STR_Practice]=QString();
|
||||
if (!m_profile->contains(DI_STR_Address)) (*m_profile)[DI_STR_Address]=QString();
|
||||
if (!m_profile->contains(DI_STR_PatientID)) (*m_profile)[DI_STR_PatientID]=QString();
|
||||
if (!m_profile->contains(STR_DI_Name)) (*m_profile)[STR_DI_Name]=QString();
|
||||
if (!m_profile->contains(STR_DI_Phone)) (*m_profile)[STR_DI_Phone]=QString();
|
||||
if (!m_profile->contains(STR_DI_Email)) (*m_profile)[STR_DI_Email]=QString();
|
||||
if (!m_profile->contains(STR_DI_Practice)) (*m_profile)[STR_DI_Practice]=QString();
|
||||
if (!m_profile->contains(STR_DI_Address)) (*m_profile)[STR_DI_Address]=QString();
|
||||
if (!m_profile->contains(STR_DI_PatientID)) (*m_profile)[STR_DI_PatientID]=QString();
|
||||
}
|
||||
~DoctorInfo() {}
|
||||
void setProfile(Profile *p) { m_profile=p; }
|
||||
|
||||
const QString name() { return (*m_profile)[DI_STR_Name].toString(); }
|
||||
const QString phone() { return (*m_profile)[DI_STR_Phone].toString(); }
|
||||
const QString email() { return (*m_profile)[DI_STR_Email].toString(); }
|
||||
const QString practiceName() { return (*m_profile)[DI_STR_Practice].toString(); }
|
||||
const QString address() { return (*m_profile)[DI_STR_Address].toString(); }
|
||||
const QString patientID() { return (*m_profile)[DI_STR_PatientID].toString(); }
|
||||
const QString name() { return (*m_profile)[STR_DI_Name].toString(); }
|
||||
const QString phone() { return (*m_profile)[STR_DI_Phone].toString(); }
|
||||
const QString email() { return (*m_profile)[STR_DI_Email].toString(); }
|
||||
const QString practiceName() { return (*m_profile)[STR_DI_Practice].toString(); }
|
||||
const QString address() { return (*m_profile)[STR_DI_Address].toString(); }
|
||||
const QString patientID() { return (*m_profile)[STR_DI_PatientID].toString(); }
|
||||
|
||||
void setName(QString name) { (*m_profile)[DI_STR_Name]=name; }
|
||||
void setPhone(QString phone) { (*m_profile)[DI_STR_Phone]=phone; }
|
||||
void setEmail(QString phone) { (*m_profile)[DI_STR_Email]=phone; }
|
||||
void setPracticeName(QString practice) { (*m_profile)[DI_STR_Practice]=practice; }
|
||||
void setAddress(QString address) { (*m_profile)[DI_STR_Address]=address; }
|
||||
void setPatientID(QString pid) { (*m_profile)[DI_STR_PatientID]=pid; }
|
||||
void setName(QString name) { (*m_profile)[STR_DI_Name]=name; }
|
||||
void setPhone(QString phone) { (*m_profile)[STR_DI_Phone]=phone; }
|
||||
void setEmail(QString phone) { (*m_profile)[STR_DI_Email]=phone; }
|
||||
void setPracticeName(QString practice) { (*m_profile)[STR_DI_Practice]=practice; }
|
||||
void setAddress(QString address) { (*m_profile)[STR_DI_Address]=address; }
|
||||
void setPatientID(QString pid) { (*m_profile)[STR_DI_PatientID]=pid; }
|
||||
|
||||
Profile *m_profile;
|
||||
};
|
||||
|
||||
extern const char * UI_STR_DOB;
|
||||
extern const char * UI_STR_FirstName;
|
||||
extern const char * UI_STR_LastName;
|
||||
extern const char * UI_STR_UserName;
|
||||
extern const char * UI_STR_Password;
|
||||
extern const char * UI_STR_Address;
|
||||
extern const char * UI_STR_Phone;
|
||||
extern const char * UI_STR_EmailAddress;
|
||||
extern const char * UI_STR_Country;
|
||||
extern const char * UI_STR_Height;
|
||||
extern const char * UI_STR_Gender;
|
||||
extern const char * UI_STR_TimeZone;
|
||||
extern const char * UI_STR_Language;
|
||||
extern const char * UI_STR_DST;
|
||||
extern const char * STR_UI_DOB;
|
||||
extern const char * STR_UI_FirstName;
|
||||
extern const char * STR_UI_LastName;
|
||||
extern const char * STR_UI_UserName;
|
||||
extern const char * STR_UI_Password;
|
||||
extern const char * STR_UI_Address;
|
||||
extern const char * STR_UI_Phone;
|
||||
extern const char * STR_UI_EmailAddress;
|
||||
extern const char * STR_UI_Country;
|
||||
extern const char * STR_UI_Height;
|
||||
extern const char * STR_UI_Gender;
|
||||
extern const char * STR_UI_TimeZone;
|
||||
extern const char * STR_UI_Language;
|
||||
extern const char * STR_UI_DST;
|
||||
|
||||
/*! \class UserInfo
|
||||
\brief Profile Options relating to the User Information
|
||||
@ -214,78 +214,78 @@ public:
|
||||
//! \brief Create UserInfo object given Profile *p, and initialize the defaults
|
||||
UserInfo(Profile *p) : m_profile(p)
|
||||
{
|
||||
if (!m_profile->contains(UI_STR_DOB)) (*m_profile)[UI_STR_DOB]=QDate(1970,1,1);
|
||||
if (!m_profile->contains(UI_STR_FirstName)) (*m_profile)[UI_STR_FirstName]=QString();
|
||||
if (!m_profile->contains(UI_STR_LastName)) (*m_profile)[UI_STR_LastName]=QString();
|
||||
if (!m_profile->contains(UI_STR_UserName)) (*m_profile)[UI_STR_UserName]=QString();
|
||||
if (!m_profile->contains(UI_STR_Password)) (*m_profile)[UI_STR_Password]=QString();
|
||||
if (!m_profile->contains(UI_STR_Address)) (*m_profile)[UI_STR_Address]=QString();
|
||||
if (!m_profile->contains(UI_STR_Phone)) (*m_profile)[UI_STR_Phone]=QString();
|
||||
if (!m_profile->contains(UI_STR_EmailAddress)) (*m_profile)[UI_STR_EmailAddress]=QString();
|
||||
if (!m_profile->contains(UI_STR_Country)) (*m_profile)[UI_STR_Country]=QString();
|
||||
if (!m_profile->contains(UI_STR_Height)) (*m_profile)[UI_STR_Height]=0.0;
|
||||
if (!m_profile->contains(UI_STR_Gender)) (*m_profile)[UI_STR_Gender]=(int)GenderNotSpecified;
|
||||
if (!m_profile->contains(UI_STR_TimeZone)) (*m_profile)[UI_STR_TimeZone]=QString();
|
||||
if (!m_profile->contains(UI_STR_Language)) (*m_profile)[UI_STR_Language]="English";
|
||||
if (!m_profile->contains(UI_STR_DST)) (*m_profile)[UI_STR_DST]=false;
|
||||
if (!m_profile->contains(STR_UI_DOB)) (*m_profile)[STR_UI_DOB]=QDate(1970,1,1);
|
||||
if (!m_profile->contains(STR_UI_FirstName)) (*m_profile)[STR_UI_FirstName]=QString();
|
||||
if (!m_profile->contains(STR_UI_LastName)) (*m_profile)[STR_UI_LastName]=QString();
|
||||
if (!m_profile->contains(STR_UI_UserName)) (*m_profile)[STR_UI_UserName]=QString();
|
||||
if (!m_profile->contains(STR_UI_Password)) (*m_profile)[STR_UI_Password]=QString();
|
||||
if (!m_profile->contains(STR_UI_Address)) (*m_profile)[STR_UI_Address]=QString();
|
||||
if (!m_profile->contains(STR_UI_Phone)) (*m_profile)[STR_UI_Phone]=QString();
|
||||
if (!m_profile->contains(STR_UI_EmailAddress)) (*m_profile)[STR_UI_EmailAddress]=QString();
|
||||
if (!m_profile->contains(STR_UI_Country)) (*m_profile)[STR_UI_Country]=QString();
|
||||
if (!m_profile->contains(STR_UI_Height)) (*m_profile)[STR_UI_Height]=0.0;
|
||||
if (!m_profile->contains(STR_UI_Gender)) (*m_profile)[STR_UI_Gender]=(int)GenderNotSpecified;
|
||||
if (!m_profile->contains(STR_UI_TimeZone)) (*m_profile)[STR_UI_TimeZone]=QString();
|
||||
if (!m_profile->contains(STR_UI_Language)) (*m_profile)[STR_UI_Language]="English";
|
||||
if (!m_profile->contains(STR_UI_DST)) (*m_profile)[STR_UI_DST]=false;
|
||||
|
||||
}
|
||||
~UserInfo() {}
|
||||
|
||||
void setProfile(Profile *p) { m_profile=p; }
|
||||
|
||||
QDate DOB() { return (*m_profile)[UI_STR_DOB].toDate(); }
|
||||
const QString firstName() { return (*m_profile)[UI_STR_FirstName].toString(); }
|
||||
const QString lastName() { return (*m_profile)[UI_STR_LastName].toString(); }
|
||||
const QString userName() { return (*m_profile)[UI_STR_UserName].toString(); }
|
||||
const QString address() { return (*m_profile)[UI_STR_Address].toString(); }
|
||||
const QString phone() { return (*m_profile)[UI_STR_Phone].toString(); }
|
||||
const QString email() { return (*m_profile)[UI_STR_EmailAddress].toString(); }
|
||||
double height() { return (*m_profile)[UI_STR_Height].toDouble(); }
|
||||
const QString country() { return (*m_profile)[UI_STR_Country].toString(); }
|
||||
Gender gender() { return (Gender)(*m_profile)[UI_STR_Gender].toInt(); }
|
||||
const QString timeZone() { return (*m_profile)[UI_STR_TimeZone].toString(); }
|
||||
const QString language() { return (*m_profile)[UI_STR_Language].toString(); }
|
||||
bool daylightSaving() { return (*m_profile)[UI_STR_DST].toBool(); }
|
||||
QDate DOB() { return (*m_profile)[STR_UI_DOB].toDate(); }
|
||||
const QString firstName() { return (*m_profile)[STR_UI_FirstName].toString(); }
|
||||
const QString lastName() { return (*m_profile)[STR_UI_LastName].toString(); }
|
||||
const QString userName() { return (*m_profile)[STR_UI_UserName].toString(); }
|
||||
const QString address() { return (*m_profile)[STR_UI_Address].toString(); }
|
||||
const QString phone() { return (*m_profile)[STR_UI_Phone].toString(); }
|
||||
const QString email() { return (*m_profile)[STR_UI_EmailAddress].toString(); }
|
||||
double height() { return (*m_profile)[STR_UI_Height].toDouble(); }
|
||||
const QString country() { return (*m_profile)[STR_UI_Country].toString(); }
|
||||
Gender gender() { return (Gender)(*m_profile)[STR_UI_Gender].toInt(); }
|
||||
const QString timeZone() { return (*m_profile)[STR_UI_TimeZone].toString(); }
|
||||
const QString language() { return (*m_profile)[STR_UI_Language].toString(); }
|
||||
bool daylightSaving() { return (*m_profile)[STR_UI_DST].toBool(); }
|
||||
|
||||
void setDOB(QDate date) { (*m_profile)[UI_STR_DOB]=date; }
|
||||
void setFirstName(QString name) { (*m_profile)[UI_STR_FirstName]=name; }
|
||||
void setLastName(QString name) { (*m_profile)[UI_STR_LastName]=name; }
|
||||
void setUserName(QString username) { (*m_profile)[UI_STR_UserName]=username; }
|
||||
void setAddress(QString address) { (*m_profile)[UI_STR_Address]=address; }
|
||||
void setPhone(QString phone) { (*m_profile)[UI_STR_Phone]=phone; }
|
||||
void setEmail(QString email) { (*m_profile)[UI_STR_EmailAddress]=email; }
|
||||
void setHeight(double height) { (*m_profile)[UI_STR_Height]=height; }
|
||||
void setCountry(QString country) { (*m_profile)[UI_STR_Country]=country; }
|
||||
void setGender(Gender g) { (*m_profile)[UI_STR_Gender]=(int)g; }
|
||||
void setTimeZone(QString tz) { (*m_profile)[UI_STR_TimeZone]=tz; }
|
||||
void setLanguage(QString language) { (*m_profile)[UI_STR_Language]=language; }
|
||||
void setDaylightSaving(bool ds) { (*m_profile)[UI_STR_DST]=ds; }
|
||||
void setDOB(QDate date) { (*m_profile)[STR_UI_DOB]=date; }
|
||||
void setFirstName(QString name) { (*m_profile)[STR_UI_FirstName]=name; }
|
||||
void setLastName(QString name) { (*m_profile)[STR_UI_LastName]=name; }
|
||||
void setUserName(QString username) { (*m_profile)[STR_UI_UserName]=username; }
|
||||
void setAddress(QString address) { (*m_profile)[STR_UI_Address]=address; }
|
||||
void setPhone(QString phone) { (*m_profile)[STR_UI_Phone]=phone; }
|
||||
void setEmail(QString email) { (*m_profile)[STR_UI_EmailAddress]=email; }
|
||||
void setHeight(double height) { (*m_profile)[STR_UI_Height]=height; }
|
||||
void setCountry(QString country) { (*m_profile)[STR_UI_Country]=country; }
|
||||
void setGender(Gender g) { (*m_profile)[STR_UI_Gender]=(int)g; }
|
||||
void setTimeZone(QString tz) { (*m_profile)[STR_UI_TimeZone]=tz; }
|
||||
void setLanguage(QString language) { (*m_profile)[STR_UI_Language]=language; }
|
||||
void setDaylightSaving(bool ds) { (*m_profile)[STR_UI_DST]=ds; }
|
||||
|
||||
bool hasPassword() {
|
||||
return !((*m_profile)[UI_STR_Password].toString().isEmpty());
|
||||
return !((*m_profile)[STR_UI_Password].toString().isEmpty());
|
||||
}
|
||||
bool checkPassword(QString password) {
|
||||
QByteArray ba=password.toUtf8();
|
||||
return ((*m_profile)[UI_STR_Password].toString()==QString(QCryptographicHash::hash(ba,QCryptographicHash::Sha1).toHex()));
|
||||
return ((*m_profile)[STR_UI_Password].toString()==QString(QCryptographicHash::hash(ba,QCryptographicHash::Sha1).toHex()));
|
||||
}
|
||||
void setPassword(QString password) {
|
||||
QByteArray ba=password.toUtf8();
|
||||
// Hash me.
|
||||
(*m_profile)[UI_STR_Password]=QString(QCryptographicHash::hash(ba,QCryptographicHash::Sha1).toHex());
|
||||
(*m_profile)[STR_UI_Password]=QString(QCryptographicHash::hash(ba,QCryptographicHash::Sha1).toHex());
|
||||
}
|
||||
|
||||
Profile *m_profile;
|
||||
};
|
||||
|
||||
extern const char * OS_STR_EnableOximetry;
|
||||
extern const char * OS_STR_SyncOximetry;
|
||||
extern const char * OS_STR_OximeterType;
|
||||
extern const char * OS_STR_OxiDiscardThreshold;
|
||||
extern const char * OS_STR_SPO2DropDuration;
|
||||
extern const char * OS_STR_SPO2DropPercentage;
|
||||
extern const char * OS_STR_PulseChangeDuration;
|
||||
extern const char * OS_STR_PulseChangeBPM;
|
||||
extern const char * STR_OS_EnableOximetry;
|
||||
extern const char * STR_OS_SyncOximetry;
|
||||
extern const char * STR_OS_OximeterType;
|
||||
extern const char * STR_OS_OxiDiscardThreshold;
|
||||
extern const char * STR_OS_SPO2DropDuration;
|
||||
extern const char * STR_OS_SPO2DropPercentage;
|
||||
extern const char * STR_OS_PulseChangeDuration;
|
||||
extern const char * STR_OS_PulseChangeBPM;
|
||||
|
||||
/*! \class OxiSettings
|
||||
\brief Profile Options relating to the Oximetry settings
|
||||
@ -296,52 +296,52 @@ public:
|
||||
//! \brief Create OxiSettings object given Profile *p, and initialize the defaults
|
||||
OxiSettings(Profile *p) :m_profile(p)
|
||||
{
|
||||
if (!m_profile->contains(OS_STR_EnableOximetry)) (*m_profile)[OS_STR_EnableOximetry]=false;
|
||||
if (!m_profile->contains(OS_STR_SyncOximetry)) (*m_profile)[OS_STR_SyncOximetry]=true;
|
||||
if (!m_profile->contains(OS_STR_OximeterType)) (*m_profile)[OS_STR_OximeterType]="CMS50";
|
||||
if (!m_profile->contains(OS_STR_OxiDiscardThreshold)) (*m_profile)[OS_STR_OxiDiscardThreshold]=0.0;
|
||||
if (!m_profile->contains(OS_STR_SPO2DropDuration)) (*m_profile)[OS_STR_SPO2DropDuration]=8.0;
|
||||
if (!m_profile->contains(OS_STR_SPO2DropPercentage)) (*m_profile)[OS_STR_SPO2DropPercentage]=3.0;
|
||||
if (!m_profile->contains(OS_STR_PulseChangeDuration)) (*m_profile)[OS_STR_PulseChangeDuration]=8.0;
|
||||
if (!m_profile->contains(OS_STR_PulseChangeBPM)) (*m_profile)[OS_STR_PulseChangeBPM]=5.0;
|
||||
if (!m_profile->contains(STR_OS_EnableOximetry)) (*m_profile)[STR_OS_EnableOximetry]=false;
|
||||
if (!m_profile->contains(STR_OS_SyncOximetry)) (*m_profile)[STR_OS_SyncOximetry]=true;
|
||||
if (!m_profile->contains(STR_OS_OximeterType)) (*m_profile)[STR_OS_OximeterType]="CMS50";
|
||||
if (!m_profile->contains(STR_OS_OxiDiscardThreshold)) (*m_profile)[STR_OS_OxiDiscardThreshold]=0.0;
|
||||
if (!m_profile->contains(STR_OS_SPO2DropDuration)) (*m_profile)[STR_OS_SPO2DropDuration]=8.0;
|
||||
if (!m_profile->contains(STR_OS_SPO2DropPercentage)) (*m_profile)[STR_OS_SPO2DropPercentage]=3.0;
|
||||
if (!m_profile->contains(STR_OS_PulseChangeDuration)) (*m_profile)[STR_OS_PulseChangeDuration]=8.0;
|
||||
if (!m_profile->contains(STR_OS_PulseChangeBPM)) (*m_profile)[STR_OS_PulseChangeBPM]=5.0;
|
||||
}
|
||||
~OxiSettings() {}
|
||||
|
||||
void setProfile(Profile *p) { m_profile=p; }
|
||||
|
||||
bool oximetryEnabled() { return (*m_profile)[OS_STR_EnableOximetry].toBool(); }
|
||||
bool syncOximetry() { return (*m_profile)[OS_STR_SyncOximetry].toBool(); }
|
||||
QString oximeterType() { return (*m_profile)[OS_STR_OximeterType].toString(); }
|
||||
double oxiDiscardThreshold() { return (*m_profile)[OS_STR_OxiDiscardThreshold].toDouble(); }
|
||||
double spO2DropDuration() { return (*m_profile)[OS_STR_SPO2DropDuration].toDouble(); }
|
||||
double spO2DropPercentage() { return (*m_profile)[OS_STR_SPO2DropPercentage].toDouble(); }
|
||||
double pulseChangeDuration() { return (*m_profile)[OS_STR_PulseChangeDuration].toDouble(); }
|
||||
double pulseChangeBPM() { return (*m_profile)[OS_STR_PulseChangeBPM].toDouble(); }
|
||||
bool oximetryEnabled() { return (*m_profile)[STR_OS_EnableOximetry].toBool(); }
|
||||
bool syncOximetry() { return (*m_profile)[STR_OS_SyncOximetry].toBool(); }
|
||||
QString oximeterType() { return (*m_profile)[STR_OS_OximeterType].toString(); }
|
||||
double oxiDiscardThreshold() { return (*m_profile)[STR_OS_OxiDiscardThreshold].toDouble(); }
|
||||
double spO2DropDuration() { return (*m_profile)[STR_OS_SPO2DropDuration].toDouble(); }
|
||||
double spO2DropPercentage() { return (*m_profile)[STR_OS_SPO2DropPercentage].toDouble(); }
|
||||
double pulseChangeDuration() { return (*m_profile)[STR_OS_PulseChangeDuration].toDouble(); }
|
||||
double pulseChangeBPM() { return (*m_profile)[STR_OS_PulseChangeBPM].toDouble(); }
|
||||
|
||||
void setOximetryEnabled(bool enabled) { (*m_profile)[OS_STR_EnableOximetry]=enabled; }
|
||||
void setSyncOximetry(bool synced) { (*m_profile)[OS_STR_SyncOximetry]=synced; }
|
||||
void setOximeterType(QString oxitype) { (*m_profile)[OS_STR_OximeterType]=oxitype; }
|
||||
void setOxiDiscardThreshold(double thresh) { (*m_profile)[OS_STR_OxiDiscardThreshold]=thresh; }
|
||||
void setSpO2DropDuration(double duration) { (*m_profile)[OS_STR_SPO2DropDuration]=duration; }
|
||||
void setSpO2DropPercentage(double percentage) { (*m_profile)[OS_STR_SPO2DropPercentage]=percentage; }
|
||||
void setPulseChangeDuration(double duration) { (*m_profile)[OS_STR_PulseChangeDuration]=duration; }
|
||||
void setPulseChangeBPM(double bpm) { (*m_profile)[OS_STR_PulseChangeBPM]=bpm; }
|
||||
void setOximetryEnabled(bool enabled) { (*m_profile)[STR_OS_EnableOximetry]=enabled; }
|
||||
void setSyncOximetry(bool synced) { (*m_profile)[STR_OS_SyncOximetry]=synced; }
|
||||
void setOximeterType(QString oxitype) { (*m_profile)[STR_OS_OximeterType]=oxitype; }
|
||||
void setOxiDiscardThreshold(double thresh) { (*m_profile)[STR_OS_OxiDiscardThreshold]=thresh; }
|
||||
void setSpO2DropDuration(double duration) { (*m_profile)[STR_OS_SPO2DropDuration]=duration; }
|
||||
void setSpO2DropPercentage(double percentage) { (*m_profile)[STR_OS_SPO2DropPercentage]=percentage; }
|
||||
void setPulseChangeDuration(double duration) { (*m_profile)[STR_OS_PulseChangeDuration]=duration; }
|
||||
void setPulseChangeBPM(double bpm) { (*m_profile)[STR_OS_PulseChangeBPM]=bpm; }
|
||||
|
||||
Profile *m_profile;
|
||||
};
|
||||
|
||||
extern const char * CS_STR_ComplianceHours;
|
||||
extern const char * CS_STR_ShowCompliance;
|
||||
extern const char * CS_STR_ShowLeaksMode;
|
||||
extern const char * CS_STR_MaskStartDate;
|
||||
extern const char * CS_STR_MaskDescription;
|
||||
extern const char * CS_STR_MaskType;
|
||||
extern const char * CS_STR_PrescribedMode;
|
||||
extern const char * CS_STR_PrescribedMinPressure;
|
||||
extern const char * CS_STR_PrescribedMaxPressure;
|
||||
extern const char * CS_STR_UntreatedAHI;
|
||||
extern const char * CS_STR_Notes;
|
||||
extern const char * CS_STR_DateDiagnosed;
|
||||
extern const char * STR_CS_ComplianceHours;
|
||||
extern const char * STR_CS_ShowCompliance;
|
||||
extern const char * STR_CS_ShowLeaksMode;
|
||||
extern const char * STR_CS_MaskStartDate;
|
||||
extern const char * STR_CS_MaskDescription;
|
||||
extern const char * STR_CS_MaskType;
|
||||
extern const char * STR_CS_PrescribedMode;
|
||||
extern const char * STR_CS_PrescribedMinPressure;
|
||||
extern const char * STR_CS_PrescribedMaxPressure;
|
||||
extern const char * STR_CS_UntreatedAHI;
|
||||
extern const char * STR_CS_Notes;
|
||||
extern const char * STR_CS_DateDiagnosed;
|
||||
|
||||
/*! \class CPAPSettings
|
||||
\brief Profile Options relating to the CPAP settings
|
||||
@ -352,19 +352,19 @@ public:
|
||||
//! \brief Create CPAPSettings object given Profile *p, and initialize the defaults
|
||||
CPAPSettings(Profile *p) :m_profile(p)
|
||||
{
|
||||
if (!m_profile->contains(CS_STR_ComplianceHours)) (*m_profile)[CS_STR_ComplianceHours]=4;
|
||||
if (!m_profile->contains(CS_STR_ShowCompliance)) (*m_profile)[CS_STR_ShowCompliance]=true;
|
||||
if (!m_profile->contains(CS_STR_ShowLeaksMode)) (*m_profile)[CS_STR_ShowLeaksMode]=0;
|
||||
if (!m_profile->contains(STR_CS_ComplianceHours)) (*m_profile)[STR_CS_ComplianceHours]=4;
|
||||
if (!m_profile->contains(STR_CS_ShowCompliance)) (*m_profile)[STR_CS_ShowCompliance]=true;
|
||||
if (!m_profile->contains(STR_CS_ShowLeaksMode)) (*m_profile)[STR_CS_ShowLeaksMode]=0;
|
||||
// TODO: Check if this date is initiliazed yet
|
||||
if (!m_profile->contains(CS_STR_MaskStartDate)) (*m_profile)[CS_STR_MaskStartDate]=QDate();
|
||||
if (!m_profile->contains(CS_STR_MaskDescription)) (*m_profile)[CS_STR_MaskDescription]=QString();
|
||||
if (!m_profile->contains(CS_STR_MaskType)) (*m_profile)[CS_STR_MaskType]=Mask_Unknown;
|
||||
if (!m_profile->contains(CS_STR_PrescribedMode)) (*m_profile)[CS_STR_PrescribedMode]=MODE_UNKNOWN;
|
||||
if (!m_profile->contains(CS_STR_PrescribedMinPressure)) (*m_profile)[CS_STR_PrescribedMinPressure]=0.0;
|
||||
if (!m_profile->contains(CS_STR_PrescribedMaxPressure)) (*m_profile)[CS_STR_PrescribedMaxPressure]=0.0;
|
||||
if (!m_profile->contains(CS_STR_UntreatedAHI)) (*m_profile)[CS_STR_UntreatedAHI]=0.0;
|
||||
if (!m_profile->contains(CS_STR_Notes)) (*m_profile)[CS_STR_Notes]=QString();
|
||||
if (!m_profile->contains(CS_STR_DateDiagnosed)) (*m_profile)[CS_STR_DateDiagnosed]=QDate();
|
||||
if (!m_profile->contains(STR_CS_MaskStartDate)) (*m_profile)[STR_CS_MaskStartDate]=QDate();
|
||||
if (!m_profile->contains(STR_CS_MaskDescription)) (*m_profile)[STR_CS_MaskDescription]=QString();
|
||||
if (!m_profile->contains(STR_CS_MaskType)) (*m_profile)[STR_CS_MaskType]=Mask_Unknown;
|
||||
if (!m_profile->contains(STR_CS_PrescribedMode)) (*m_profile)[STR_CS_PrescribedMode]=MODE_UNKNOWN;
|
||||
if (!m_profile->contains(STR_CS_PrescribedMinPressure)) (*m_profile)[STR_CS_PrescribedMinPressure]=0.0;
|
||||
if (!m_profile->contains(STR_CS_PrescribedMaxPressure)) (*m_profile)[STR_CS_PrescribedMaxPressure]=0.0;
|
||||
if (!m_profile->contains(STR_CS_UntreatedAHI)) (*m_profile)[STR_CS_UntreatedAHI]=0.0;
|
||||
if (!m_profile->contains(STR_CS_Notes)) (*m_profile)[STR_CS_Notes]=QString();
|
||||
if (!m_profile->contains(STR_CS_DateDiagnosed)) (*m_profile)[STR_CS_DateDiagnosed]=QDate();
|
||||
}
|
||||
|
||||
~CPAPSettings() {}
|
||||
@ -372,43 +372,44 @@ public:
|
||||
void setProfile(Profile *p) { m_profile=p; }
|
||||
|
||||
//Getters
|
||||
double complianceHours() { return (*m_profile)[CS_STR_ComplianceHours].toDouble(); }
|
||||
bool showComplianceInfo() { return (*m_profile)[CS_STR_ShowCompliance].toBool(); }
|
||||
int leakMode() { return (*m_profile)[CS_STR_ShowLeaksMode].toInt(); }
|
||||
QDate maskStartDate() { return (*m_profile)[CS_STR_MaskStartDate].toDate(); }
|
||||
QString maskDescription() { return (*m_profile)[CS_STR_MaskDescription].toString(); }
|
||||
MaskType maskType() { return (MaskType)(*m_profile)[CS_STR_MaskType].toInt(); }
|
||||
CPAPMode mode() { return CPAPMode((*m_profile)[CS_STR_PrescribedMode].toInt()); }
|
||||
double minPressure() { return (*m_profile)[CS_STR_PrescribedMinPressure].toDouble(); }
|
||||
double maxPressure() { return (*m_profile)[CS_STR_PrescribedMaxPressure].toDouble(); }
|
||||
double untreatedAHI() { return (*m_profile)[CS_STR_UntreatedAHI].toDouble(); }
|
||||
const QString notes() { return (*m_profile)[CS_STR_Notes].toString(); }
|
||||
QDate dateDiagnosed() { return (*m_profile)[CS_STR_DateDiagnosed].toDate(); }
|
||||
double complianceHours() { return (*m_profile)[STR_CS_ComplianceHours].toDouble(); }
|
||||
bool showComplianceInfo() { return (*m_profile)[STR_CS_ShowCompliance].toBool(); }
|
||||
int leakMode() { return (*m_profile)[STR_CS_ShowLeaksMode].toInt(); }
|
||||
QDate maskStartDate() { return (*m_profile)[STR_CS_MaskStartDate].toDate(); }
|
||||
QString maskDescription() { return (*m_profile)[STR_CS_MaskDescription].toString(); }
|
||||
MaskType maskType() { return (MaskType)(*m_profile)[STR_CS_MaskType].toInt(); }
|
||||
CPAPMode mode() { return CPAPMode((*m_profile)[STR_CS_PrescribedMode].toInt()); }
|
||||
double minPressure() { return (*m_profile)[STR_CS_PrescribedMinPressure].toDouble(); }
|
||||
double maxPressure() { return (*m_profile)[STR_CS_PrescribedMaxPressure].toDouble(); }
|
||||
double untreatedAHI() { return (*m_profile)[STR_CS_UntreatedAHI].toDouble(); }
|
||||
const QString notes() { return (*m_profile)[STR_CS_Notes].toString(); }
|
||||
QDate dateDiagnosed() { return (*m_profile)[STR_CS_DateDiagnosed].toDate(); }
|
||||
|
||||
//Setters
|
||||
void setMode(CPAPMode mode) { (*m_profile)[CS_STR_PrescribedMode]=(int)mode; }
|
||||
void setMinPressure(double pressure) { (*m_profile)[CS_STR_PrescribedMinPressure]=pressure; }
|
||||
void setMaxPressure(double pressure) { (*m_profile)[CS_STR_PrescribedMaxPressure]=pressure; }
|
||||
void setUntreatedAHI(double ahi) { (*m_profile)[CS_STR_UntreatedAHI]=ahi; }
|
||||
void setNotes(QString notes) { (*m_profile)[CS_STR_Notes]=notes; }
|
||||
void setDateDiagnosed(QDate date) { (*m_profile)[CS_STR_DateDiagnosed]=date; }
|
||||
void setComplianceHours(double hours) { (*m_profile)[CS_STR_ComplianceHours]=hours; }
|
||||
void setShowComplianceInfo(bool b) { (*m_profile)[CS_STR_ShowCompliance]=b; }
|
||||
void setLeakMode(int leakmode) { (*m_profile)[CS_STR_ShowLeaksMode]=(int)leakmode; }
|
||||
void setMaskStartDate(QDate date) { (*m_profile)[CS_STR_MaskStartDate]=date; }
|
||||
void setMaskDescription(QString description) { (*m_profile)[CS_STR_MaskDescription]=description; }
|
||||
void setMaskType(MaskType masktype) { (*m_profile)[CS_STR_MaskType]=(int)masktype; }
|
||||
void setMode(CPAPMode mode) { (*m_profile)[STR_CS_PrescribedMode]=(int)mode; }
|
||||
void setMinPressure(double pressure) { (*m_profile)[STR_CS_PrescribedMinPressure]=pressure; }
|
||||
void setMaxPressure(double pressure) { (*m_profile)[STR_CS_PrescribedMaxPressure]=pressure; }
|
||||
void setUntreatedAHI(double ahi) { (*m_profile)[STR_CS_UntreatedAHI]=ahi; }
|
||||
void setNotes(QString notes) { (*m_profile)[STR_CS_Notes]=notes; }
|
||||
void setDateDiagnosed(QDate date) { (*m_profile)[STR_CS_DateDiagnosed]=date; }
|
||||
void setComplianceHours(double hours) { (*m_profile)[STR_CS_ComplianceHours]=hours; }
|
||||
void setShowComplianceInfo(bool b) { (*m_profile)[STR_CS_ShowCompliance]=b; }
|
||||
void setLeakMode(int leakmode) { (*m_profile)[STR_CS_ShowLeaksMode]=(int)leakmode; }
|
||||
void setMaskStartDate(QDate date) { (*m_profile)[STR_CS_MaskStartDate]=date; }
|
||||
void setMaskDescription(QString description) { (*m_profile)[STR_CS_MaskDescription]=description; }
|
||||
void setMaskType(MaskType masktype) { (*m_profile)[STR_CS_MaskType]=(int)masktype; }
|
||||
|
||||
Profile *m_profile;
|
||||
};
|
||||
|
||||
extern const char * IS_STR_DaySplitTime;
|
||||
extern const char * IS_STR_CacheSessions;
|
||||
extern const char * IS_STR_CombineCloseSessions;
|
||||
extern const char * IS_STR_IgnoreShorterSessions;
|
||||
extern const char * IS_STR_Multithreading;
|
||||
extern const char * IS_STR_TrashDayCache;
|
||||
extern const char * IS_STR_ShowSerialNumbers;
|
||||
extern const char * STR_IS_DaySplitTime;
|
||||
extern const char * STR_IS_CacheSessions;
|
||||
extern const char * STR_IS_CombineCloseSessions;
|
||||
extern const char * STR_IS_IgnoreShorterSessions;
|
||||
extern const char * STR_IS_Multithreading;
|
||||
extern const char * STR_IS_BackupCardData;
|
||||
extern const char * STR_IS_CompressBackupData;
|
||||
extern const char * STR_IS_CompressSessionData;
|
||||
|
||||
/*! \class ImportSettings
|
||||
\brief Profile Options relating to the Import process
|
||||
@ -419,43 +420,46 @@ public:
|
||||
//! \brief Create ImportSettings object given Profile *p, and initialize the defaults
|
||||
SessionSettings(Profile *p) :m_profile(p)
|
||||
{
|
||||
if (!m_profile->contains(IS_STR_DaySplitTime)) (*m_profile)[IS_STR_DaySplitTime]=QTime(12,0,0);
|
||||
if (!m_profile->contains(IS_STR_CacheSessions)) (*m_profile)[IS_STR_CacheSessions]=false;
|
||||
if (!m_profile->contains(IS_STR_CombineCloseSessions)) (*m_profile)[IS_STR_CombineCloseSessions]=240;
|
||||
if (!m_profile->contains(IS_STR_IgnoreShorterSessions)) (*m_profile)[IS_STR_IgnoreShorterSessions]=5;
|
||||
if (!m_profile->contains(IS_STR_Multithreading)) (*m_profile)[IS_STR_Multithreading]=QThread::idealThreadCount() > 1;
|
||||
if (!m_profile->contains(IS_STR_TrashDayCache)) (*m_profile)[IS_STR_TrashDayCache]=false; // can't remember..
|
||||
if (!m_profile->contains(IS_STR_ShowSerialNumbers)) (*m_profile)[IS_STR_ShowSerialNumbers]=false;
|
||||
if (!m_profile->contains(STR_IS_DaySplitTime)) (*m_profile)[STR_IS_DaySplitTime]=QTime(12,0,0);
|
||||
if (!m_profile->contains(STR_IS_CacheSessions)) (*m_profile)[STR_IS_CacheSessions]=false;
|
||||
if (!m_profile->contains(STR_IS_CombineCloseSessions)) (*m_profile)[STR_IS_CombineCloseSessions]=240;
|
||||
if (!m_profile->contains(STR_IS_IgnoreShorterSessions)) (*m_profile)[STR_IS_IgnoreShorterSessions]=5;
|
||||
if (!m_profile->contains(STR_IS_Multithreading)) (*m_profile)[STR_IS_Multithreading]=QThread::idealThreadCount() > 1;
|
||||
if (!m_profile->contains(STR_IS_BackupCardData)) (*m_profile)[STR_IS_BackupCardData]=true;
|
||||
if (!m_profile->contains(STR_IS_CompressBackupData)) (*m_profile)[STR_IS_CompressBackupData]=false;
|
||||
if (!m_profile->contains(STR_IS_CompressSessionData)) (*m_profile)[STR_IS_CompressSessionData]=false;
|
||||
}
|
||||
~SessionSettings() {}
|
||||
|
||||
void setProfile(Profile *p) { m_profile=p; }
|
||||
|
||||
QTime daySplitTime() { return (*m_profile)[IS_STR_DaySplitTime].toTime(); }
|
||||
bool cacheSessions() { return (*m_profile)[IS_STR_CacheSessions].toBool(); }
|
||||
double combineCloseSessions() { return (*m_profile)[IS_STR_CombineCloseSessions].toDouble(); }
|
||||
double ignoreShortSessions() { return (*m_profile)[IS_STR_IgnoreShorterSessions].toDouble(); }
|
||||
bool multithreading() { return (*m_profile)[IS_STR_Multithreading].toBool(); }
|
||||
bool trashDayCache() { return (*m_profile)[IS_STR_TrashDayCache].toBool(); }
|
||||
bool showSerialNumbers() { return (*m_profile)[IS_STR_ShowSerialNumbers].toBool(); }
|
||||
QTime daySplitTime() { return (*m_profile)[STR_IS_DaySplitTime].toTime(); }
|
||||
bool cacheSessions() { return (*m_profile)[STR_IS_CacheSessions].toBool(); }
|
||||
double combineCloseSessions() { return (*m_profile)[STR_IS_CombineCloseSessions].toDouble(); }
|
||||
double ignoreShortSessions() { return (*m_profile)[STR_IS_IgnoreShorterSessions].toDouble(); }
|
||||
bool multithreading() { return (*m_profile)[STR_IS_Multithreading].toBool(); }
|
||||
bool compressSessionData() { return (*m_profile)[STR_IS_CompressSessionData].toBool(); }
|
||||
bool compressBackupData() { return (*m_profile)[STR_IS_CompressBackupData].toBool(); }
|
||||
bool backupCardData() { return (*m_profile)[STR_IS_BackupCardData].toBool(); }
|
||||
|
||||
void setDaySplitTime(QTime time) { (*m_profile)[IS_STR_DaySplitTime]=time; }
|
||||
void setCacheSessions(bool c) { (*m_profile)[IS_STR_CacheSessions]=c; }
|
||||
void setCombineCloseSessions(double val) { (*m_profile)[IS_STR_CombineCloseSessions]=val; }
|
||||
void setIgnoreShortSessions(double val) { (*m_profile)[IS_STR_IgnoreShorterSessions]=val; }
|
||||
void setMultithreading(bool enabled) { (*m_profile)[IS_STR_Multithreading]=enabled; }
|
||||
void setTrashDayCache(bool trash) { (*m_profile)[IS_STR_TrashDayCache]=trash; }
|
||||
void setShowSerialNumbers(bool trash) { (*m_profile)[IS_STR_ShowSerialNumbers]=trash; }
|
||||
void setDaySplitTime(QTime time) { (*m_profile)[STR_IS_DaySplitTime]=time; }
|
||||
void setCacheSessions(bool c) { (*m_profile)[STR_IS_CacheSessions]=c; }
|
||||
void setCombineCloseSessions(double val) { (*m_profile)[STR_IS_CombineCloseSessions]=val; }
|
||||
void setIgnoreShortSessions(double val) { (*m_profile)[STR_IS_IgnoreShorterSessions]=val; }
|
||||
void setMultithreading(bool enabled) { (*m_profile)[STR_IS_Multithreading]=enabled; }
|
||||
void setBackupCardData(bool enabled) { (*m_profile)[STR_IS_BackupCardData]=enabled; }
|
||||
void setCompressBackupData(bool enabled) { (*m_profile)[STR_IS_CompressBackupData]=enabled; }
|
||||
void setCompressSessionData(bool enabled) { (*m_profile)[STR_IS_CompressSessionData]=enabled; }
|
||||
|
||||
Profile *m_profile;
|
||||
};
|
||||
|
||||
extern const char * AS_STR_GraphHeight;
|
||||
extern const char * AS_STR_AntiAliasing;
|
||||
extern const char * AS_STR_GraphSnapshots;
|
||||
extern const char * AS_STR_Animations;
|
||||
extern const char * AS_STR_SquareWave;
|
||||
extern const char * AS_STR_OverlayType;
|
||||
extern const char * STR_AS_GraphHeight;
|
||||
extern const char * STR_AS_AntiAliasing;
|
||||
extern const char * STR_AS_GraphSnapshots;
|
||||
extern const char * STR_AS_Animations;
|
||||
extern const char * STR_AS_SquareWave;
|
||||
extern const char * STR_AS_OverlayType;
|
||||
|
||||
/*! \class AppearanceSettings
|
||||
\brief Profile Options relating to Visual Appearance
|
||||
@ -466,54 +470,54 @@ public:
|
||||
//! \brief Create AppearanceSettings object given Profile *p, and initialize the defaults
|
||||
AppearanceSettings(Profile *p) :m_profile(p)
|
||||
{
|
||||
if (!m_profile->contains(AS_STR_GraphHeight)) (*m_profile)[AS_STR_GraphHeight]=180.0;
|
||||
if (!m_profile->contains(AS_STR_AntiAliasing)) (*m_profile)[AS_STR_AntiAliasing]=false; // i think it's ugly
|
||||
if (!m_profile->contains(AS_STR_GraphSnapshots)) (*m_profile)[AS_STR_GraphSnapshots]=true;
|
||||
if (!m_profile->contains(AS_STR_Animations)) (*m_profile)[AS_STR_Animations]=true;
|
||||
if (!m_profile->contains(AS_STR_SquareWave)) (*m_profile)[AS_STR_SquareWave]=false;
|
||||
if (!m_profile->contains(AS_STR_OverlayType)) (*m_profile)[AS_STR_OverlayType]=ODT_Bars;
|
||||
if (!m_profile->contains(STR_AS_GraphHeight)) (*m_profile)[STR_AS_GraphHeight]=180.0;
|
||||
if (!m_profile->contains(STR_AS_AntiAliasing)) (*m_profile)[STR_AS_AntiAliasing]=false; // i think it's ugly
|
||||
if (!m_profile->contains(STR_AS_GraphSnapshots)) (*m_profile)[STR_AS_GraphSnapshots]=true;
|
||||
if (!m_profile->contains(STR_AS_Animations)) (*m_profile)[STR_AS_Animations]=true;
|
||||
if (!m_profile->contains(STR_AS_SquareWave)) (*m_profile)[STR_AS_SquareWave]=false;
|
||||
if (!m_profile->contains(STR_AS_OverlayType)) (*m_profile)[STR_AS_OverlayType]=ODT_Bars;
|
||||
}
|
||||
~AppearanceSettings() {}
|
||||
|
||||
void setProfile(Profile *p) { m_profile=p; }
|
||||
|
||||
//! \brief Returns the normal (unscaled) height of a graph
|
||||
int graphHeight() { return (*m_profile)[AS_STR_GraphHeight].toInt(); }
|
||||
int graphHeight() { return (*m_profile)[STR_AS_GraphHeight].toInt(); }
|
||||
//! \brief Returns true if AntiAliasing (the graphical smoothing method) is enabled
|
||||
bool antiAliasing() { return (*m_profile)[AS_STR_AntiAliasing].toBool(); }
|
||||
bool antiAliasing() { return (*m_profile)[STR_AS_AntiAliasing].toBool(); }
|
||||
//! \brief Returns true if renderPixmap function is in use, which takes snapshots of graphs
|
||||
bool graphSnapshots() { return (*m_profile)[AS_STR_GraphSnapshots].toBool(); }
|
||||
bool graphSnapshots() { return (*m_profile)[STR_AS_GraphSnapshots].toBool(); }
|
||||
//! \brief Returns true if Graphical animations & Transitions will be drawn
|
||||
bool animations() { return (*m_profile)[AS_STR_Animations].toBool(); }
|
||||
bool animations() { return (*m_profile)[STR_AS_Animations].toBool(); }
|
||||
//! \brief Returns true if Square Wave plots are preferred (where possible)
|
||||
bool squareWavePlots() { return (*m_profile)[AS_STR_SquareWave].toBool(); }
|
||||
bool squareWavePlots() { return (*m_profile)[STR_AS_SquareWave].toBool(); }
|
||||
//! \brief Returns the type of overlay flags (which are displayed over the Flow Waveform)
|
||||
OverlayDisplayType overlayType() { return (OverlayDisplayType )(*m_profile)[AS_STR_OverlayType].toInt(); }
|
||||
OverlayDisplayType overlayType() { return (OverlayDisplayType )(*m_profile)[STR_AS_OverlayType].toInt(); }
|
||||
|
||||
//! \brief Set the normal (unscaled) height of a graph.
|
||||
void setGraphHeight(int height) { (*m_profile)[AS_STR_GraphHeight]=height; }
|
||||
void setGraphHeight(int height) { (*m_profile)[STR_AS_GraphHeight]=height; }
|
||||
//! \brief Set to true to turn on AntiAliasing (the graphical smoothing method)
|
||||
void setAntiAliasing(bool aa) { (*m_profile)[AS_STR_AntiAliasing]=aa; }
|
||||
void setAntiAliasing(bool aa) { (*m_profile)[STR_AS_AntiAliasing]=aa; }
|
||||
//! \brief Set to true if renderPixmap functions are in use, which takes snapshots of graphs.
|
||||
void setGraphSnapshots(bool gs) { (*m_profile)[AS_STR_GraphSnapshots]=gs; }
|
||||
void setGraphSnapshots(bool gs) { (*m_profile)[STR_AS_GraphSnapshots]=gs; }
|
||||
//! \brief Set to true if Graphical animations & Transitions will be drawn
|
||||
void setAnimations(bool anim) { (*m_profile)[AS_STR_Animations]=anim; }
|
||||
void setAnimations(bool anim) { (*m_profile)[STR_AS_Animations]=anim; }
|
||||
//! \brief Set whether or not to useSquare Wave plots (where possible)
|
||||
void setSquareWavePlots(bool sw) { (*m_profile)[AS_STR_SquareWave]=sw; }
|
||||
void setSquareWavePlots(bool sw) { (*m_profile)[STR_AS_SquareWave]=sw; }
|
||||
//! \brief Sets the type of overlay flags (which are displayed over the Flow Waveform)
|
||||
void setOverlayType(OverlayDisplayType od) { (*m_profile)[AS_STR_OverlayType]=(int)od; }
|
||||
void setOverlayType(OverlayDisplayType od) { (*m_profile)[STR_AS_OverlayType]=(int)od; }
|
||||
|
||||
Profile *m_profile;
|
||||
};
|
||||
|
||||
|
||||
extern const char * US_STR_UnitSystem;
|
||||
extern const char * US_STR_EventWindowSize;
|
||||
extern const char * US_STR_SkipEmptyDays;
|
||||
extern const char * US_STR_RebuildCache;
|
||||
extern const char * US_STR_ShowDebug;
|
||||
extern const char * US_STR_LinkGroups;
|
||||
extern const char * US_STR_CalculateRDI;
|
||||
extern const char * STR_US_UnitSystem;
|
||||
extern const char * STR_US_EventWindowSize;
|
||||
extern const char * STR_US_SkipEmptyDays;
|
||||
extern const char * STR_US_RebuildCache;
|
||||
extern const char * STR_US_ShowDebug;
|
||||
extern const char * STR_US_LinkGroups;
|
||||
extern const char * STR_US_CalculateRDI;
|
||||
extern const char * STR_US_ShowSerialNumbers;
|
||||
|
||||
/*! \class UserSettings
|
||||
\brief Profile Options relating to General User Settings
|
||||
@ -524,34 +528,36 @@ public:
|
||||
//! \brief Create UserSettings object given Profile *p, and initialize the defaults
|
||||
UserSettings(Profile *p) :m_profile(p)
|
||||
{
|
||||
if (!m_profile->contains(US_STR_UnitSystem)) (*m_profile)[US_STR_UnitSystem]=US_Metric;
|
||||
if (!m_profile->contains(US_STR_EventWindowSize)) (*m_profile)[US_STR_EventWindowSize]=4.0;
|
||||
if (!m_profile->contains(US_STR_SkipEmptyDays)) (*m_profile)[US_STR_SkipEmptyDays]=true;
|
||||
if (!m_profile->contains(US_STR_RebuildCache)) (*m_profile)[US_STR_RebuildCache]=false; // can't remember..
|
||||
if (!m_profile->contains(US_STR_ShowDebug)) (*m_profile)[US_STR_ShowDebug]=false;
|
||||
if (!m_profile->contains(US_STR_LinkGroups)) (*m_profile)[US_STR_LinkGroups]=true; // can't remember..
|
||||
if (!m_profile->contains(US_STR_CalculateRDI)) (*m_profile)[US_STR_CalculateRDI]=false;
|
||||
if (!m_profile->contains(STR_US_UnitSystem)) (*m_profile)[STR_US_UnitSystem]=US_Metric;
|
||||
if (!m_profile->contains(STR_US_EventWindowSize)) (*m_profile)[STR_US_EventWindowSize]=4.0;
|
||||
if (!m_profile->contains(STR_US_SkipEmptyDays)) (*m_profile)[STR_US_SkipEmptyDays]=true;
|
||||
if (!m_profile->contains(STR_US_RebuildCache)) (*m_profile)[STR_US_RebuildCache]=false; // can't remember..
|
||||
if (!m_profile->contains(STR_US_ShowDebug)) (*m_profile)[STR_US_ShowDebug]=false;
|
||||
if (!m_profile->contains(STR_US_LinkGroups)) (*m_profile)[STR_US_LinkGroups]=true; // can't remember..
|
||||
if (!m_profile->contains(STR_US_CalculateRDI)) (*m_profile)[STR_US_CalculateRDI]=false;
|
||||
if (!m_profile->contains(STR_US_ShowSerialNumbers)) (*m_profile)[STR_US_ShowSerialNumbers]=false;
|
||||
}
|
||||
~UserSettings() {}
|
||||
|
||||
void setProfile(Profile *p) { m_profile=p; }
|
||||
|
||||
UnitSystem unitSystem() { return (UnitSystem)(*m_profile)[US_STR_UnitSystem].toInt(); }
|
||||
double eventWindowSize() { return (*m_profile)[US_STR_EventWindowSize].toDouble(); }
|
||||
bool skipEmptyDays() { return (*m_profile)[US_STR_SkipEmptyDays].toBool(); }
|
||||
bool rebuildCache() { return (*m_profile)[US_STR_RebuildCache].toBool(); }
|
||||
bool showDebug() { return (*m_profile)[US_STR_ShowDebug].toBool(); }
|
||||
bool linkGroups() { return (*m_profile)[US_STR_LinkGroups].toBool(); }
|
||||
bool calculateRDI() { return (*m_profile)[US_STR_CalculateRDI].toBool(); }
|
||||
|
||||
void setUnitSystem(UnitSystem us) { (*m_profile)[US_STR_UnitSystem]=(int)us; }
|
||||
void setEventWindowSize(double size) { (*m_profile)[US_STR_EventWindowSize]=size; }
|
||||
void setSkipEmptyDays(bool skip) { (*m_profile)[US_STR_SkipEmptyDays]=skip; }
|
||||
void setRebuildCache(bool rebuild) { (*m_profile)[US_STR_RebuildCache]=rebuild; }
|
||||
void setShowDebug(bool b) { (*m_profile)[US_STR_ShowDebug]=b; }
|
||||
void setLinkGroups(bool link) { (*m_profile)[US_STR_LinkGroups]=link; }
|
||||
void setCalculateRDI(bool rdi) { (*m_profile)[US_STR_CalculateRDI]=rdi; }
|
||||
UnitSystem unitSystem() { return (UnitSystem)(*m_profile)[STR_US_UnitSystem].toInt(); }
|
||||
double eventWindowSize() { return (*m_profile)[STR_US_EventWindowSize].toDouble(); }
|
||||
bool skipEmptyDays() { return (*m_profile)[STR_US_SkipEmptyDays].toBool(); }
|
||||
bool rebuildCache() { return (*m_profile)[STR_US_RebuildCache].toBool(); }
|
||||
bool showDebug() { return (*m_profile)[STR_US_ShowDebug].toBool(); }
|
||||
bool linkGroups() { return (*m_profile)[STR_US_LinkGroups].toBool(); }
|
||||
bool calculateRDI() { return (*m_profile)[STR_US_CalculateRDI].toBool(); }
|
||||
bool showSerialNumbers() { return (*m_profile)[STR_US_ShowSerialNumbers].toBool(); }
|
||||
|
||||
void setUnitSystem(UnitSystem us) { (*m_profile)[STR_US_UnitSystem]=(int)us; }
|
||||
void setEventWindowSize(double size) { (*m_profile)[STR_US_EventWindowSize]=size; }
|
||||
void setSkipEmptyDays(bool skip) { (*m_profile)[STR_US_SkipEmptyDays]=skip; }
|
||||
void setRebuildCache(bool rebuild) { (*m_profile)[STR_US_RebuildCache]=rebuild; }
|
||||
void setShowDebug(bool b) { (*m_profile)[STR_US_ShowDebug]=b; }
|
||||
void setLinkGroups(bool link) { (*m_profile)[STR_US_LinkGroups]=link; }
|
||||
void setCalculateRDI(bool rdi) { (*m_profile)[STR_US_CalculateRDI]=rdi; }
|
||||
void setShowSerialNumbers(bool enabled) { (*m_profile)[STR_US_ShowSerialNumbers]=enabled; }
|
||||
|
||||
Profile *m_profile;
|
||||
};
|
||||
|
@ -12,7 +12,9 @@
|
||||
#include <QMessageBox>
|
||||
#include <QMetaType>
|
||||
#include <algorithm>
|
||||
#include <SleepLib/calcs.h>
|
||||
|
||||
#include "SleepLib/calcs.h"
|
||||
#include "SleepLib/profiles.h"
|
||||
|
||||
using namespace std;
|
||||
|
||||
@ -23,7 +25,7 @@ const quint16 filetype_data=1;
|
||||
// This is the uber important database version for SleepyHeads internal storage
|
||||
// Increment this after stuffing with Session's save & load code.
|
||||
const quint16 summary_version=11;
|
||||
const quint16 events_version=8;
|
||||
const quint16 events_version=9;
|
||||
|
||||
Session::Session(Machine * m,SessionID session)
|
||||
{
|
||||
@ -337,27 +339,42 @@ bool Session::LoadSummary(QString filename)
|
||||
return true;
|
||||
}
|
||||
|
||||
const quint16 compress_method=1;
|
||||
|
||||
bool Session::StoreEvents(QString filename)
|
||||
{
|
||||
QFile file(filename);
|
||||
file.open(QIODevice::WriteOnly);
|
||||
|
||||
QDataStream out(&file);
|
||||
QByteArray headerbytes;
|
||||
QDataStream header(&headerbytes,QIODevice::WriteOnly);
|
||||
header.setVersion(QDataStream::Qt_4_6);
|
||||
header.setByteOrder(QDataStream::LittleEndian);
|
||||
|
||||
header << (quint32)magic; // New Magic Number
|
||||
header << (quint16)events_version; // File Version
|
||||
header << (quint16)filetype_data; // File type 1 == Event
|
||||
header << (quint32)s_machine->id();// Machine Type
|
||||
header << (quint32)s_session; // This session's ID
|
||||
header << s_first;
|
||||
header << s_last;
|
||||
|
||||
quint16 compress=0;
|
||||
|
||||
if (p_profile->session->compressSessionData())
|
||||
compress=compress_method;
|
||||
|
||||
header << (quint16)compress;
|
||||
|
||||
header << (quint16)s_machine->GetType();// Machine Type
|
||||
|
||||
QByteArray databytes;
|
||||
QDataStream out(&databytes,QIODevice::WriteOnly);
|
||||
out.setVersion(QDataStream::Qt_4_6);
|
||||
out.setByteOrder(QDataStream::LittleEndian);
|
||||
|
||||
out << (quint32)magic; // Magic Number
|
||||
out << (quint16)events_version; // File Version
|
||||
out << (quint16)filetype_data; // File type 1 == Event
|
||||
out << (quint32)s_machine->id();// Machine ID
|
||||
|
||||
out << (quint32)s_session; // This session's ID
|
||||
out << s_first;
|
||||
out << s_last;
|
||||
|
||||
out << (qint16)eventlist.size(); // Number of event categories
|
||||
|
||||
|
||||
QHash<ChannelID,QVector<EventList *> >::iterator i;
|
||||
|
||||
for (i=eventlist.begin(); i!=eventlist.end(); i++) {
|
||||
@ -386,26 +403,59 @@ bool Session::StoreEvents(QString filename)
|
||||
for (int j=0;j<i.value().size();j++) {
|
||||
EventList &e=*i.value()[j];
|
||||
|
||||
for (quint32 c=0;c<e.count();c++) {
|
||||
out << e.raw(c);
|
||||
}
|
||||
// Store the raw event list data in EventStoreType (16bit short)
|
||||
EventStoreType *ptr=e.m_data.data();
|
||||
out.writeRawData((char *)ptr,e.count() << 1);
|
||||
// for (quint32 c=0;c<e.count();c++) {
|
||||
// out << *ptr++;//e.raw(c);
|
||||
// }
|
||||
// Store the second field, only if there
|
||||
if (e.hasSecondField()) {
|
||||
for (quint32 c=0;c<e.count();c++) {
|
||||
out << e.raw2(c);
|
||||
}
|
||||
ptr=e.m_data2.data();
|
||||
out.writeRawData((char *)ptr,e.count() << 1);
|
||||
// for (quint32 c=0;c<e.count();c++) {
|
||||
// out << *ptr++; //e.raw2(c);
|
||||
// }
|
||||
}
|
||||
// Store the time delta fields for non-waveform EventLists
|
||||
if (e.type()!=EVL_Waveform) {
|
||||
for (quint32 c=0;c<e.count();c++) {
|
||||
out << e.getTime()[c];
|
||||
}
|
||||
quint32 * tptr=e.m_time.data();
|
||||
out.writeRawData((char *)tptr,e.count() << 2);
|
||||
// for (quint32 c=0;c<e.count();c++) {
|
||||
// out << *tptr++; //e.getTime()[c];
|
||||
// }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
qint32 datasize=databytes.size();
|
||||
|
||||
// Checksum the _uncompressed_ data
|
||||
quint16 chk=qChecksum(databytes.data(),databytes.size());
|
||||
|
||||
header << datasize;
|
||||
header << chk;
|
||||
|
||||
QByteArray data;
|
||||
|
||||
if (compress>0) {
|
||||
data=qCompress(databytes);
|
||||
} else {
|
||||
data=databytes;
|
||||
}
|
||||
|
||||
file.write(headerbytes);
|
||||
file.write(data);
|
||||
file.close();
|
||||
return true;
|
||||
}
|
||||
bool Session::LoadEvents(QString filename)
|
||||
{
|
||||
quint32 t32,magicnum,machid,sessid;
|
||||
quint16 t16,version,type,crc16,machtype,compmethod;
|
||||
quint8 t8;
|
||||
qint32 datasize;
|
||||
|
||||
if (filename.isEmpty()) {
|
||||
qDebug() << "Session::LoadEvents() Filename is empty";
|
||||
return false;
|
||||
@ -416,47 +466,67 @@ bool Session::LoadEvents(QString filename)
|
||||
qDebug() << "Couldn't open file" << filename;
|
||||
return false;
|
||||
}
|
||||
QDataStream in(&file);
|
||||
in.setVersion(QDataStream::Qt_4_6);
|
||||
in.setByteOrder(QDataStream::LittleEndian);
|
||||
|
||||
quint32 t32;
|
||||
quint16 t16;
|
||||
quint8 t8;
|
||||
//qint16 i16;
|
||||
qint16 version;
|
||||
QByteArray headerbytes=file.read(42);
|
||||
|
||||
in >> t32; // Magic Number
|
||||
if (t32!=magic) {
|
||||
qWarning() << "Wrong Magic number in " << filename;
|
||||
return false;
|
||||
}
|
||||
QDataStream header(headerbytes);
|
||||
header.setVersion(QDataStream::Qt_4_6);
|
||||
header.setByteOrder(QDataStream::LittleEndian);
|
||||
|
||||
in >> version; // File Version
|
||||
bool compressed=true;
|
||||
header >> magicnum; // Magic Number (quint32)
|
||||
header >> version; // Version (quint16)
|
||||
header >> type; // File type (quint16)
|
||||
header >> machid; // Machine ID (quint32)
|
||||
header >> sessid; //(quint32)
|
||||
header >> s_first; //(qint64)
|
||||
header >> s_last; //(qint64)
|
||||
|
||||
if (version<6) { // prior to version 6 is too old to deal with
|
||||
qDebug() << "Old File Version";
|
||||
//throw OldDBVersion();
|
||||
//qWarning() << "Old dbversion "<< t16 << "summary file.. Sorry, you need to purge and reimport";
|
||||
return false;
|
||||
}
|
||||
|
||||
in >> t16; // File Type
|
||||
if (t16!=filetype_data) {
|
||||
if (type!=filetype_data) {
|
||||
qDebug() << "Wrong File Type in " << filename;
|
||||
return false;
|
||||
}
|
||||
|
||||
qint32 ts32;
|
||||
in >> ts32; // MachineID
|
||||
if (ts32!=s_machine->id()) {
|
||||
qWarning() << "Machine ID does not match in" << filename << " I will try to load anyway in case you know what your doing.";
|
||||
if (magicnum!=magic) {
|
||||
qWarning() << "Wrong Magic number in " << filename;
|
||||
return false;
|
||||
}
|
||||
|
||||
in >> t32; // Sessionid
|
||||
s_session=t32;
|
||||
in >> s_first;
|
||||
in >> s_last;
|
||||
if (version<6) { // prior to version 6 is too old to deal with
|
||||
qDebug() << "Old File Version, can't open file";
|
||||
return false;
|
||||
}
|
||||
|
||||
if (version<9) {
|
||||
file.seek(32);
|
||||
} else {
|
||||
header >> compmethod; // Compression Method (quint16)
|
||||
header >> machtype; // Machine Type (quint16)
|
||||
header >> datasize; // Size of Uncompressed Data (quint32)
|
||||
header >> crc16; // CRC16 of Uncompressed Data (quint16)
|
||||
}
|
||||
|
||||
QByteArray databytes,temp=file.readAll();
|
||||
file.close();
|
||||
if (compmethod>0) {
|
||||
databytes=qUncompress(temp);
|
||||
} else {
|
||||
databytes=temp;
|
||||
}
|
||||
|
||||
if (databytes.size()!=datasize) {
|
||||
qDebug() << "File" << filename << "has returned wrong datasize";
|
||||
return false;
|
||||
}
|
||||
quint16 crc=qChecksum(databytes.data(),databytes.size());
|
||||
if (crc!=crc16) {
|
||||
qDebug() << "CRC Doesn't match in" << filename;
|
||||
return false;
|
||||
}
|
||||
|
||||
QDataStream in(databytes);
|
||||
in.setVersion(QDataStream::Qt_4_6);
|
||||
in.setByteOrder(QDataStream::LittleEndian);
|
||||
|
||||
qint16 mcsize;
|
||||
in >> mcsize; // number of Machine Code lists
|
||||
@ -523,34 +593,39 @@ bool Session::LoadEvents(QString filename)
|
||||
size2=sizevec[i];
|
||||
for (int j=0;j<size2;j++) {
|
||||
EventList &evec=*eventlist[code][j];
|
||||
evec.m_data.reserve(evec.m_count);
|
||||
for (quint32 c=0;c<evec.m_count;c++) {
|
||||
in >> t;
|
||||
evec.m_data.push_back(t);
|
||||
}
|
||||
evec.m_data.resize(evec.m_count);
|
||||
EventStoreType *ptr=evec.m_data.data();
|
||||
|
||||
in.readRawData((char *)ptr, evec.m_count << 1);
|
||||
// for (quint32 c=0;c<evec.m_count;c++) {
|
||||
// in >> t;
|
||||
// *ptr++=t;
|
||||
// }
|
||||
if (evec.hasSecondField()) {
|
||||
evec.m_data2.reserve(evec.m_count);
|
||||
for (quint32 c=0;c<evec.m_count;c++) {
|
||||
in >> t;
|
||||
evec.m_data2.push_back(t);
|
||||
}
|
||||
evec.m_data2.resize(evec.m_count);
|
||||
ptr=evec.m_data2.data();
|
||||
|
||||
in.readRawData((char *)ptr,evec.m_count << 1);
|
||||
// for (quint32 c=0;c<evec.m_count;c++) {
|
||||
// in >> t;
|
||||
// *ptr++=t;
|
||||
// }
|
||||
}
|
||||
//last=evec.first();
|
||||
if (evec.type()!=EVL_Waveform) {
|
||||
evec.m_time.reserve(evec.m_count);
|
||||
for (quint32 c=0;c<evec.m_count;c++) {
|
||||
in >> x;
|
||||
//last+=x;
|
||||
evec.m_time.push_back(x);
|
||||
//evec.m_time[c]=x;
|
||||
evec.m_time.resize(evec.m_count);
|
||||
quint32 * tptr=evec.m_time.data();
|
||||
in.readRawData((char *)tptr,evec.m_count << 2);
|
||||
// for (quint32 c=0;c<evec.m_count;c++) {
|
||||
// in >> x;
|
||||
// *tptr++=x;
|
||||
// }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
file.close();
|
||||
|
||||
if (version<events_version) {
|
||||
qDebug() << "Upgrading Events file to version" << events_version;
|
||||
qDebug() << "Upgrading Events file" << filename << "to version" << events_version;
|
||||
UpdateSummaries();
|
||||
StoreEvents(filename);
|
||||
}
|
||||
@ -619,7 +694,7 @@ void Session::UpdateSummaries()
|
||||
EventDataType gain=el->gain();
|
||||
m_gain[id]=gain;
|
||||
}
|
||||
if (!((id==CPAP_FlowRate) || (id==CPAP_MaskPressure)))
|
||||
if (!((id==CPAP_FlowRate) || (id==CPAP_MaskPressureHi) || (id==CPAP_RespEvent) || (id==CPAP_MaskPressure)))
|
||||
updateCountSummary(id);
|
||||
|
||||
Min(id);
|
||||
@ -627,7 +702,8 @@ void Session::UpdateSummaries()
|
||||
count(id);
|
||||
last(id);
|
||||
first(id);
|
||||
if ((id==CPAP_FlowRate) || (id==CPAP_MaskPressure)) continue;
|
||||
if (((id==CPAP_FlowRate) || (id==CPAP_MaskPressureHi) || (id==CPAP_RespEvent) || (id==CPAP_MaskPressure)))
|
||||
continue;
|
||||
|
||||
cph(id);
|
||||
sph(id);
|
||||
@ -1183,6 +1259,7 @@ EventList * Session::AddEventList(ChannelID code, EventListType et,EventDataType
|
||||
//return NULL;
|
||||
}
|
||||
EventList * el=new EventList(et,gain,offset,min,max,rate,second_field);
|
||||
|
||||
eventlist[code].push_back(el);
|
||||
//s_machine->registerChannel(chan);
|
||||
return el;
|
||||
|
19
daily.cpp
19
daily.cpp
@ -423,15 +423,17 @@ void Daily::ReloadGraphs()
|
||||
{
|
||||
ui->splitter->setVisible(true);
|
||||
QDate d;
|
||||
|
||||
if (previous_date.isValid()) {
|
||||
d=previous_date;
|
||||
Unload(d);
|
||||
} //else
|
||||
// Unload(d);
|
||||
}
|
||||
d=PROFILE.LastDay();
|
||||
if (!d.isValid()) {
|
||||
d=ui->calendar->selectedDate();
|
||||
}
|
||||
on_calendar_currentPageChanged(d.year(),d.month());
|
||||
// this fires a signal which unloads the old and loads the new
|
||||
ui->calendar->setSelectedDate(d);
|
||||
//Load(d);
|
||||
}
|
||||
@ -582,15 +584,21 @@ void Daily::LoadDate(QDate date)
|
||||
|
||||
void Daily::on_calendar_selectionChanged()
|
||||
{
|
||||
this->setCursor(Qt::BusyCursor);
|
||||
QTime time;
|
||||
time_t unload_time, load_time, other_time;
|
||||
time.start();
|
||||
|
||||
this->setCursor(Qt::BusyCursor);
|
||||
if (previous_date.isValid()) {
|
||||
// GraphView->fadeOut();
|
||||
Unload(previous_date);
|
||||
}
|
||||
unload_time=time.restart();
|
||||
//bool fadedir=previous_date < ui->calendar->selectedDate();
|
||||
ZombieMeterMoved=false;
|
||||
Load(ui->calendar->selectedDate());
|
||||
load_time=time.restart();
|
||||
|
||||
//GraphView->fadeIn(fadedir);
|
||||
GraphView->redraw();
|
||||
ui->calButton->setText(ui->calendar->selectedDate().toString(Qt::TextDate));
|
||||
@ -607,6 +615,9 @@ void Daily::on_calendar_selectionChanged()
|
||||
ui->weightSpinBox->setSuffix(STR_UNIT_KG);
|
||||
}
|
||||
this->setCursor(Qt::ArrowCursor);
|
||||
other_time=time.restart();
|
||||
|
||||
qDebug() << "Page change time (in ms): Unload ="<<unload_time<<"Load =" << load_time << "Other =" << other_time;
|
||||
}
|
||||
void Daily::ResetGraphLayout()
|
||||
{
|
||||
@ -760,7 +771,7 @@ void Daily::Load(QDate date)
|
||||
if (cpap->machine->properties.find(STR_PROP_SubModel)!=cpap->machine->properties.end())
|
||||
submodel=" <br/>"+cpap->machine->properties[STR_PROP_SubModel];
|
||||
html+="<tr><td colspan=4 align=center><b>"+cpap->machine->properties[STR_PROP_Brand]+"</b> <br>"+cpap->machine->properties[STR_PROP_Model]+" "+cpap->machine->properties[STR_PROP_ModelNumber]+submodel+"</td></tr>\n";
|
||||
if (PROFILE.session->showSerialNumbers()) {
|
||||
if (PROFILE.general->showSerialNumbers()) {
|
||||
html+="<tr><td colspan=4 align=center>"+cpap->machine->properties[STR_PROP_Serial]+"</td></tr>\n";
|
||||
}
|
||||
CPAPMode mode=(CPAPMode)(int)cpap->settings_max(CPAP_Mode);
|
||||
|
@ -560,7 +560,7 @@ void MainWindow::on_summaryButton_clicked()
|
||||
|
||||
if (mach.size()==0) {
|
||||
html+="<table cellpadding=2 cellspacing=0 border=0 width=100% height=70%>";
|
||||
html+="<tr><td align=center><h1>Please Import Some Data</h1><br/><i>SleepyHead is pretty much useless without it.</i></td></tr></table>";
|
||||
html+="<tr><td align=center><h1>Please Import Some Data</h1><br/><i>SleepyHead is pretty much useless without it.</i><br/>First import can take a few minutes.</td></tr></table>";
|
||||
html+=htmlFooter();
|
||||
ui->summaryView->setHtml(html);
|
||||
return;
|
||||
@ -813,7 +813,6 @@ void MainWindow::on_summaryButton_clicked()
|
||||
QDate date=lastcpap;
|
||||
Day * day;
|
||||
bool lastchanged=false;
|
||||
int cnt=0;
|
||||
QVector<RXChange> rxchange;
|
||||
do {
|
||||
day=PROFILE.GetGoodDay(date,MT_CPAP);
|
||||
@ -1531,7 +1530,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date)
|
||||
return;
|
||||
}
|
||||
|
||||
QString username=PROFILE.Get(QString("_{")+QString(UI_STR_UserName)+"}_");
|
||||
QString username=PROFILE.Get(QString("_{")+QString(STR_UI_UserName)+"}_");
|
||||
|
||||
bool print_bookmarks=false;
|
||||
if (name==STR_TR_Daily) {
|
||||
@ -1823,7 +1822,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date)
|
||||
if (name==STR_TR_Daily) {
|
||||
if (!print_bookmarks) {
|
||||
for (int i=0;i<gv->size();i++) {
|
||||
gGraph *g=(*gv)[i];
|
||||
g=(*gv)[i];
|
||||
if (g->isEmpty()) continue;
|
||||
if (!g->visible()) continue;
|
||||
|
||||
@ -2251,6 +2250,7 @@ void MainWindow::on_actionAll_Data_for_current_CPAP_machine_triggered()
|
||||
|
||||
void MainWindow::keyPressEvent(QKeyEvent * event)
|
||||
{
|
||||
Q_UNUSED(event)
|
||||
//qDebug() << "Keypress:" << event->key();
|
||||
}
|
||||
|
||||
|
@ -140,7 +140,7 @@ void NewProfile::on_nextButton_clicked()
|
||||
}
|
||||
} else {
|
||||
|
||||
prof.Erase(UI_STR_Password);
|
||||
prof.Erase(STR_UI_Password);
|
||||
}
|
||||
|
||||
profile->user->setGender((Gender)ui->genderCombo->currentIndex());
|
||||
@ -246,7 +246,7 @@ void NewProfile::edit(const QString name)
|
||||
ui->userNameEdit->setReadOnly(true);
|
||||
ui->firstNameEdit->setText(profile->user->firstName());
|
||||
ui->lastNameEdit->setText(profile->user->lastName());
|
||||
if (profile->contains(UI_STR_Password) && !profile->p_preferences[UI_STR_Password].toString().isEmpty()) {
|
||||
if (profile->contains(STR_UI_Password) && !profile->p_preferences[STR_UI_Password].toString().isEmpty()) {
|
||||
// leave the password box blank..
|
||||
QString a="******";
|
||||
ui->passwordEdit1->setText(a);
|
||||
|
@ -430,7 +430,7 @@ void Overview::ResetGraphLayout()
|
||||
}
|
||||
|
||||
|
||||
void Overview::on_printDailyButton_clicked()
|
||||
/*void Overview::on_printDailyButton_clicked()
|
||||
{
|
||||
qint64 st,et;
|
||||
GraphView->GetXBounds(st,et);
|
||||
@ -460,7 +460,7 @@ void Overview::on_printDailyButton_clicked()
|
||||
|
||||
} else mainwin->Notify("If this was implemented yet, You'd be able to print multiple daily reports right now.");
|
||||
|
||||
}
|
||||
}*/
|
||||
|
||||
void Overview::on_rangeCombo_activated(int index)
|
||||
{
|
||||
|
@ -94,7 +94,7 @@ private slots:
|
||||
//! \brief Resets view to currently shown start & end dates
|
||||
void on_toolButton_clicked();
|
||||
|
||||
void on_printDailyButton_clicked();
|
||||
//void on_printDailyButton_clicked();
|
||||
|
||||
void on_rangeCombo_activated(int index);
|
||||
|
||||
|
22
overview.ui
22
overview.ui
@ -42,13 +42,13 @@
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>9</number>
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>9</number>
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>0</number>
|
||||
@ -62,6 +62,12 @@
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QComboBox" name="rangeCombo">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Fixed">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<item>
|
||||
<property name="text">
|
||||
<string>Last Week</string>
|
||||
@ -118,6 +124,12 @@
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QDateEdit" name="dateStart">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Fixed">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="calendarPopup">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
@ -135,6 +147,12 @@
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QDateEdit" name="dateEnd">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Fixed">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="calendarPopup">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
|
@ -168,6 +168,12 @@ PreferencesDialog::PreferencesDialog(QWidget *parent,Profile * _profile) :
|
||||
ui->complianceGroupbox->setChecked(profile->cpap->showComplianceInfo());
|
||||
ui->complianceHours->setValue(profile->cpap->complianceHours());
|
||||
|
||||
bool bcd=profile->session->backupCardData();
|
||||
ui->createSDBackups->setChecked(bcd);
|
||||
ui->compressSDBackups->setEnabled(bcd);
|
||||
ui->compressSDBackups->setChecked(profile->session->compressBackupData());
|
||||
ui->compressSessionData->setChecked(profile->session->compressSessionData());
|
||||
|
||||
ui->graphHeight->setValue(profile->appearance->graphHeight());
|
||||
|
||||
if (!PREF.contains(STR_GEN_UpdatesAutoCheck)) PREF[STR_GEN_UpdatesAutoCheck]=true;
|
||||
@ -288,14 +294,16 @@ void PreferencesDialog::Save()
|
||||
if ((profile->session->daySplitTime()!=ui->timeEdit->time()) ||
|
||||
(profile->session->combineCloseSessions()!=ui->combineSlider->value()) ||
|
||||
(profile->session->ignoreShortSessions()!=ui->IgnoreSlider->value())) {
|
||||
profile->session->setTrashDayCache(true);
|
||||
needs_restart=true;
|
||||
} else profile->session->setTrashDayCache(false);
|
||||
}
|
||||
|
||||
if (profile->general->calculateRDI() != ui->AddRERAtoAHI->isChecked()) {
|
||||
profile->general->setCalculateRDI(ui->AddRERAtoAHI->isChecked());
|
||||
needs_restart=true;
|
||||
}
|
||||
profile->session->setBackupCardData(ui->createSDBackups->isChecked());
|
||||
profile->session->setCompressBackupData(ui->compressSDBackups->isChecked());
|
||||
profile->session->setCompressSessionData(ui->compressSessionData->isChecked());
|
||||
|
||||
profile->session->setCombineCloseSessions(ui->combineSlider->value());
|
||||
profile->session->setIgnoreShortSessions(ui->IgnoreSlider->value());
|
||||
@ -712,3 +720,23 @@ void PreferencesDialog::on_maskTypeCombo_activated(int index)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void PreferencesDialog::on_createSDBackups_toggled(bool checked)
|
||||
{
|
||||
if (profile->session->backupCardData() && !checked) {
|
||||
QList<Machine *> mach=PROFILE.GetMachines(MT_CPAP);
|
||||
bool haveS9=false;
|
||||
for (int i=0;i<mach.size();i++) {
|
||||
if (mach[i]->GetClass()==STR_MACH_ResMed) {
|
||||
haveS9=true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (haveS9 && QMessageBox::question(this,"This may not be a good idea","ResMed S9 machines routinely delete certain data from your SD card older than 7 and 30 days (depending on resolution). If you ever need to reimport this data again (whether in SleepyHead or ResScan) this data won't come back. If you need to conserve disk space, please remember to carry out manual backups. Are you sure you want to disable these backups?",QMessageBox::Yes,QMessageBox::No)==QMessageBox::No) {
|
||||
ui->createSDBackups->setChecked(true);
|
||||
return;
|
||||
}
|
||||
}
|
||||
if (!checked) ui->compressSDBackups->setChecked(false);
|
||||
ui->compressSDBackups->setEnabled(checked);
|
||||
}
|
||||
|
@ -87,6 +87,8 @@ private slots:
|
||||
|
||||
void on_maskTypeCombo_activated(int index);
|
||||
|
||||
void on_createSDBackups_toggled(bool checked);
|
||||
|
||||
private:
|
||||
//! \brief Populates the Graph Model view with data from the Daily, Overview & Oximetry gGraphView objects
|
||||
void resetGraphModel();
|
||||
|
@ -42,7 +42,7 @@
|
||||
<item>
|
||||
<widget class="QTabWidget" name="tabWidget">
|
||||
<property name="currentIndex">
|
||||
<number>6</number>
|
||||
<number>0</number>
|
||||
</property>
|
||||
<widget class="QWidget" name="importTab">
|
||||
<attribute name="title">
|
||||
@ -50,10 +50,10 @@
|
||||
</attribute>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_5">
|
||||
<property name="spacing">
|
||||
<number>6</number>
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="margin">
|
||||
<number>5</number>
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="groupBox_6">
|
||||
@ -62,7 +62,7 @@
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_10">
|
||||
<property name="spacing">
|
||||
<number>0</number>
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>0</number>
|
||||
@ -258,6 +258,16 @@ p, li { white-space: pre-wrap; }
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QCheckBox" name="cacheSessionData">
|
||||
<property name="toolTip">
|
||||
<string>Keep session data in memory to speed up revisiting days.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Cache Session Data (uses more system memory)</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="horizontalSpacer">
|
||||
<property name="orientation">
|
||||
@ -277,198 +287,73 @@ p, li { white-space: pre-wrap; }
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="customEventGroupbox">
|
||||
<widget class="QGroupBox" name="groupBox_7">
|
||||
<property name="title">
|
||||
<string>Custom Event Flagging</string>
|
||||
<string>Session Storage Options</string>
|
||||
</property>
|
||||
<property name="checkable">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<layout class="QGridLayout" name="gridLayout_6">
|
||||
<layout class="QGridLayout" name="gridLayout_9">
|
||||
<property name="leftMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>9</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>0</number>
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>0</number>
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item row="1" column="2">
|
||||
<widget class="QDoubleSpinBox" name="apneaDuration">
|
||||
<property name="suffix">
|
||||
<string>s</string>
|
||||
</property>
|
||||
<property name="minimum">
|
||||
<double>1.000000000000000</double>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>10.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="4">
|
||||
<widget class="QDoubleSpinBox" name="apneaFlowRestriction">
|
||||
<property name="suffix">
|
||||
<string>%</string>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>10.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="0">
|
||||
<widget class="QLabel" name="label_34">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<widget class="QCheckBox" name="createSDBackups">
|
||||
<property name="text">
|
||||
<string>Apnea</string>
|
||||
<string>Create SD Card Backups during Import (especially important for ResMed users)</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="2">
|
||||
<widget class="QLabel" name="label_38">
|
||||
<item row="3" column="0">
|
||||
<widget class="QCheckBox" name="compressSessionData">
|
||||
<property name="text">
|
||||
<string>Duration</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="4">
|
||||
<widget class="QLabel" name="label_35">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Flow Restriction</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="5">
|
||||
<spacer name="horizontalSpacer_4">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item row="2" column="4">
|
||||
<widget class="QDoubleSpinBox" name="hypopneaFlowRestriction">
|
||||
<property name="suffix">
|
||||
<string>%</string>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>40.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="2" column="2">
|
||||
<widget class="QDoubleSpinBox" name="hypopneaDuration">
|
||||
<property name="suffix">
|
||||
<string>s</string>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>10.000000000000000</double>
|
||||
<string>Compress Session Data (slower, but makes SleepyHead data smaller)</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="2" column="0">
|
||||
<widget class="QLabel" name="label_37">
|
||||
<widget class="QCheckBox" name="compressSDBackups">
|
||||
<property name="text">
|
||||
<string>Hypopnea</string>
|
||||
<string>Compress SD Card Backups (slower, but makes backups smaller)</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="ahiGraphGroupbox">
|
||||
<property name="title">
|
||||
<string>AHI/Hour Graph Settings</string>
|
||||
<item row="0" column="0">
|
||||
<widget class="QLabel" name="label_32">
|
||||
<property name="font">
|
||||
<font>
|
||||
<italic>true</italic>
|
||||
</font>
|
||||
</property>
|
||||
<property name="checkable">
|
||||
<property name="text">
|
||||
<string>The following options affect the amount of disk space SleepyHead uses, and all have an effect on how long import takes. This will only really be noticeable on first import.</string>
|
||||
</property>
|
||||
<property name="wordWrap">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<layout class="QGridLayout" name="gridLayout_8">
|
||||
<property name="leftMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="horizontalSpacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="verticalSpacing">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<item row="0" column="0">
|
||||
<widget class="QLabel" name="label_36">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="4" column="0">
|
||||
<widget class="QLabel" name="label_41">
|
||||
<property name="text">
|
||||
<string>Window Size</string>
|
||||
<string><!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd">
|
||||
<html><head><meta name="qrichtext" content="1" /><style type="text/css">
|
||||
p, li { white-space: pre-wrap; }
|
||||
</style></head><body style=" font-family:'Sans'; font-size:10pt; font-weight:400; font-style:normal;">
|
||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Note: <span style=" font-style:italic;">Compression options don't automatically recompress already saved data.</span></p></body></html></string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="1">
|
||||
<widget class="QSpinBox" name="ahiGraphWindowSize">
|
||||
<property name="suffix">
|
||||
<string> minutes</string>
|
||||
</property>
|
||||
<property name="value">
|
||||
<number>60</number>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="2">
|
||||
<widget class="QCheckBox" name="ahiGraphZeroReset">
|
||||
<property name="text">
|
||||
<string>Reset to zero after each window</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="3">
|
||||
<spacer name="horizontalSpacer_6">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
@ -487,112 +372,17 @@ p, li { white-space: pre-wrap; }
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<widget class="QWidget" name="tabOther">
|
||||
<widget class="QWidget" name="tabCPAP">
|
||||
<attribute name="title">
|
||||
<string>Other</string>
|
||||
<string>&CPAP</string>
|
||||
</attribute>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_13">
|
||||
<layout class="QHBoxLayout" name="horizontalLayout_15">
|
||||
<property name="spacing">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<property name="margin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="groupBox_3">
|
||||
<property name="title">
|
||||
<string>Import Locations</string>
|
||||
</property>
|
||||
<layout class="QHBoxLayout" name="horizontalLayout_10">
|
||||
<property name="spacing">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QListView" name="importListWidget">
|
||||
<property name="editTriggers">
|
||||
<set>QAbstractItemView::NoEditTriggers</set>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QWidget" name="widget" native="true">
|
||||
<layout class="QVBoxLayout" name="verticalLayout_9">
|
||||
<item>
|
||||
<widget class="QPushButton" name="addImportLocation">
|
||||
<property name="text">
|
||||
<string>Add</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QPushButton" name="removeImportLocation">
|
||||
<property name="text">
|
||||
<string>Remove</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="verticalSpacer_5">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Vertical</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>20</width>
|
||||
<height>40</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<widget class="QWidget" name="maskTab">
|
||||
<attribute name="title">
|
||||
<string>&Mask</string>
|
||||
</attribute>
|
||||
<layout class="QHBoxLayout" name="horizontalLayout_14">
|
||||
<property name="spacing">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>2</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>2</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>2</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="groupBox_5">
|
||||
<property name="sizePolicy">
|
||||
@ -771,53 +561,258 @@ p, li { white-space: pre-wrap; }
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="groupBox_7">
|
||||
<property name="title">
|
||||
<string/>
|
||||
<widget class="Line" name="line">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Vertical</enum>
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_6">
|
||||
<property name="leftMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>2</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>0</number>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_13">
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QLabel" name="label_32">
|
||||
<property name="frameShape">
|
||||
<enum>QFrame::StyledPanel</enum>
|
||||
<widget class="QCheckBox" name="AddRERAtoAHI">
|
||||
<property name="font">
|
||||
<font>
|
||||
<weight>75</weight>
|
||||
<bold>true</bold>
|
||||
</font>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Hmmm... Empty Space
|
||||
Mask History could go here one day
|
||||
|
||||
For now a reminder for PRS1 Users
|
||||
The lower leak does not yet show unintentional
|
||||
leaks. It currently showns the leak line
|
||||
subtracted from the 0% percentile
|
||||
(aka the minimum value)
|
||||
|
||||
<--- None of this data is being used yet!
|
||||
|
||||
To solve this properly requires some fairly
|
||||
complex forumulae derived from the
|
||||
Mask Leak Profiles
|
||||
(Believe it or not, I am not a maths geek)</string>
|
||||
<string>Use RDI instead of AHI (PRS1 only)</string>
|
||||
</property>
|
||||
<property name="alignment">
|
||||
<set>Qt::AlignCenter</set>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="complianceGroupbox">
|
||||
<property name="title">
|
||||
<string>Show Compliance</string>
|
||||
</property>
|
||||
<property name="checkable">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<layout class="QGridLayout" name="gridLayout_7">
|
||||
<property name="margin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item row="0" column="1">
|
||||
<widget class="QDoubleSpinBox" name="complianceHours">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Fixed">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="toolTip">
|
||||
<string>Regard days with under this usage as "incompliant".</string>
|
||||
</property>
|
||||
<property name="suffix">
|
||||
<string> hours</string>
|
||||
</property>
|
||||
<property name="decimals">
|
||||
<number>1</number>
|
||||
</property>
|
||||
<property name="maximum">
|
||||
<double>8.000000000000000</double>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>4.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="0">
|
||||
<widget class="QLabel" name="label_39">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>as over</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="2">
|
||||
<widget class="QLabel" name="label_40">
|
||||
<property name="text">
|
||||
<string>of usage per night</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="customEventGroupbox">
|
||||
<property name="title">
|
||||
<string>Custom Event Flagging</string>
|
||||
</property>
|
||||
<property name="checkable">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<layout class="QGridLayout" name="gridLayout_6">
|
||||
<item row="0" column="1">
|
||||
<widget class="QLabel" name="label_38">
|
||||
<property name="text">
|
||||
<string>Duration</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="2">
|
||||
<widget class="QLabel" name="label_35">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Flow Restriction</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="0">
|
||||
<widget class="QLabel" name="label_34">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Apnea</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="1">
|
||||
<widget class="QDoubleSpinBox" name="apneaDuration">
|
||||
<property name="suffix">
|
||||
<string>s</string>
|
||||
</property>
|
||||
<property name="minimum">
|
||||
<double>1.000000000000000</double>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>10.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="2">
|
||||
<widget class="QDoubleSpinBox" name="apneaFlowRestriction">
|
||||
<property name="suffix">
|
||||
<string>%</string>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>10.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="3">
|
||||
<spacer name="horizontalSpacer_4">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item row="2" column="0">
|
||||
<widget class="QLabel" name="label_37">
|
||||
<property name="text">
|
||||
<string>Hypopnea</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="2" column="1">
|
||||
<widget class="QDoubleSpinBox" name="hypopneaDuration">
|
||||
<property name="suffix">
|
||||
<string>s</string>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>10.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="2" column="2">
|
||||
<widget class="QDoubleSpinBox" name="hypopneaFlowRestriction">
|
||||
<property name="suffix">
|
||||
<string>%</string>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>40.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="ahiGraphGroupbox">
|
||||
<property name="title">
|
||||
<string>AHI/Hour Graph Settings</string>
|
||||
</property>
|
||||
<property name="checkable">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<layout class="QGridLayout" name="gridLayout_8">
|
||||
<item row="0" column="0">
|
||||
<widget class="QLabel" name="label_36">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Window</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="1">
|
||||
<widget class="QSpinBox" name="ahiGraphWindowSize">
|
||||
<property name="suffix">
|
||||
<string> minutes</string>
|
||||
</property>
|
||||
<property name="value">
|
||||
<number>60</number>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="2">
|
||||
<widget class="QCheckBox" name="ahiGraphZeroReset">
|
||||
<property name="text">
|
||||
<string>Zero Reset</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="verticalSpacer_7">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Vertical</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>20</width>
|
||||
<height>40</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
</layout>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<widget class="QWidget" name="colourTab">
|
||||
@ -825,10 +820,25 @@ Mask Leak Profiles
|
||||
<string>&Events</string>
|
||||
</attribute>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_3">
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QLabel" name="label_6">
|
||||
<property name="text">
|
||||
<string>This doesn't save yet...</string>
|
||||
<string>Not entirely sure if this will get to live or not..</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
@ -926,7 +936,7 @@ Mask Leak Profiles
|
||||
<item>
|
||||
<widget class="QLabel" name="label_9">
|
||||
<property name="text">
|
||||
<string>Please Note: Graph visibility also depends on machine availability.</string>
|
||||
<string>Double click on the (Y-axis) min/max values to edit them</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
@ -940,6 +950,9 @@ Mask Leak Profiles
|
||||
<string>&Oximetry</string>
|
||||
</attribute>
|
||||
<layout class="QHBoxLayout" name="horizontalLayout_12">
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="margin">
|
||||
<number>2</number>
|
||||
</property>
|
||||
@ -1169,6 +1182,12 @@ p, li { white-space: pre-wrap; }
|
||||
<string>&General</string>
|
||||
</attribute>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_7">
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="margin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="groupBox_2">
|
||||
<property name="title">
|
||||
@ -1181,7 +1200,7 @@ p, li { white-space: pre-wrap; }
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item row="0" column="0">
|
||||
<item row="1" column="0">
|
||||
<widget class="QCheckBox" name="skipLoginScreen">
|
||||
<property name="toolTip">
|
||||
<string>Bypass the login screen and load the most recent User Profile</string>
|
||||
@ -1191,115 +1210,20 @@ p, li { white-space: pre-wrap; }
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="0">
|
||||
<widget class="QCheckBox" name="enableMultithreading">
|
||||
<property name="toolTip">
|
||||
<string>Allow use of multiple CPU cores where available to improve performance. May cause problems.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Enable Multithreading</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="4" column="0">
|
||||
<item row="5" column="0">
|
||||
<widget class="QCheckBox" name="skipEmptyDays">
|
||||
<property name="text">
|
||||
<string>Skip over Empty Days</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="1">
|
||||
<widget class="QCheckBox" name="cacheSessionData">
|
||||
<property name="toolTip">
|
||||
<string>Keep session data in memory to speed up revisiting days.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Cache Session Data</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="1">
|
||||
<widget class="QCheckBox" name="enableGraphSnapshots">
|
||||
<widget class="QCheckBox" name="enableMultithreading">
|
||||
<property name="toolTip">
|
||||
<string>Allows graphs to be "screenshotted" for display purposes.
|
||||
The Event Breakdown PIE chart uses this method, as does
|
||||
the printing code.
|
||||
Unfortunately some older computers/versions of Qt can cause
|
||||
this application to be unstable with this feature enabled.</string>
|
||||
<string>Allow use of multiple CPU cores where available to improve performance. May cause problems.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Enable event breakdown pie chart</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="4" column="1">
|
||||
<widget class="QCheckBox" name="AddRERAtoAHI">
|
||||
<property name="text">
|
||||
<string>Use RDI instead of AHI (PRS1 only)</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="complianceGroupbox">
|
||||
<property name="title">
|
||||
<string>Show Compliance</string>
|
||||
</property>
|
||||
<property name="checkable">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<layout class="QGridLayout" name="gridLayout_7">
|
||||
<property name="margin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item row="0" column="1">
|
||||
<widget class="QDoubleSpinBox" name="complianceHours">
|
||||
<property name="toolTip">
|
||||
<string>Regard days with under this usage as "incompliant".</string>
|
||||
</property>
|
||||
<property name="suffix">
|
||||
<string> hours</string>
|
||||
</property>
|
||||
<property name="decimals">
|
||||
<number>1</number>
|
||||
</property>
|
||||
<property name="maximum">
|
||||
<double>8.000000000000000</double>
|
||||
</property>
|
||||
<property name="value">
|
||||
<double>4.000000000000000</double>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="3">
|
||||
<spacer name="horizontalSpacer_7">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item row="0" column="0">
|
||||
<widget class="QLabel" name="label_39">
|
||||
<property name="text">
|
||||
<string>Define compliance as over</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="2">
|
||||
<widget class="QLabel" name="label_40">
|
||||
<property name="text">
|
||||
<string>of usage per night</string>
|
||||
<string>Enable Multithreading</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
@ -1326,6 +1250,70 @@ this application to be unstable with this feature enabled.</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="groupBox_3">
|
||||
<property name="title">
|
||||
<string>Import Locations</string>
|
||||
</property>
|
||||
<layout class="QHBoxLayout" name="horizontalLayout_10">
|
||||
<property name="spacing">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>0</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QListView" name="importListWidget">
|
||||
<property name="editTriggers">
|
||||
<set>QAbstractItemView::NoEditTriggers</set>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QWidget" name="widget" native="true">
|
||||
<layout class="QVBoxLayout" name="verticalLayout_9">
|
||||
<item>
|
||||
<widget class="QPushButton" name="addImportLocation">
|
||||
<property name="text">
|
||||
<string>Add</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QPushButton" name="removeImportLocation">
|
||||
<property name="text">
|
||||
<string>Remove</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="verticalSpacer_5">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Vertical</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>20</width>
|
||||
<height>40</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="automaticallyCheckUpdates">
|
||||
<property name="sizePolicy">
|
||||
@ -1620,6 +1608,20 @@ this application to be unstable with this feature enabled.</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QCheckBox" name="enableGraphSnapshots">
|
||||
<property name="toolTip">
|
||||
<string>Allows graphs to be "screenshotted" for display purposes.
|
||||
The Event Breakdown PIE chart uses this method, as does
|
||||
the printing code.
|
||||
Unfortunately some older computers/versions of Qt can cause
|
||||
this application to be unstable with this feature enabled.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Show event breakdown pie chart</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QCheckBox" name="animationsAndTransitionsCheckbox">
|
||||
<property name="text">
|
||||
|
Loading…
Reference in New Issue
Block a user