/******************************************************************** SleepLib Machine Class Implementation Copyright (c)2011 Mark Watkins License: GPL *********************************************************************/ #include #include #include "binary_file.h" #include "machine.h" #include "profiles.h" #include #include #include extern QProgressBar * qprogress; map MachLastCode; /* ChannelCode RegisterChannel(MachineType type) { if (MachLastCode.find(type)==MachLastCode.end()) { return MachLastCode[type]=0; } return ++(MachLastCode[type]); }; const int CPAP_WHATEVER=RegisterChannel(MT_CPAP); */ //map MachList; /*map MachineTypeString= { {MT_UNKNOWN, wxT("Unknown")}, {MT_CPAP, wxT("CPAP")}, {MT_OXIMETER, wxT("Oximeter")}, {MT_SLEEPSTAGE, wxT("SleepStage")} }; */ /*map MachineTypeLookup= { { MachineTypeString[MT_UNKNOWN].Lower(), MT_UNKNOWN }, { MachineTypeString[MT_CPAP].Lower(), MT_CPAP }, { MachineTypeString[MT_OXIMETER].Lower(), MT_OXIMETER }, { MachineTypeString[MT_SLEEPSTAGE].Lower(), MT_SLEEPSTAGE} }; */ map CPAPModeNames; /*={ {MODE_UNKNOWN,wxT("Undetermined")}, {MODE_CPAP,wxT("CPAP")}, {MODE_APAP,wxT("APAP")}, {MODE_BIPAP,wxT("BIPAP")}, {MODE_ASV,wxT("ASV")} };*/ map PressureReliefNames; /*={ {PR_UNKNOWN,_("Undetermined")}, {PR_NONE,_("None")}, {PR_CFLEX,wxT("C-Flex")}, {PR_CFLEXPLUS,wxT("C-Flex+")}, {PR_AFLEX,wxT("A-Flex")}, {PR_BIFLEX,wxT("Bi-Flex")}, {PR_EPR,wxT("Exhalation Pressure Relief (EPR)")}, {PR_SMARTFLEX,wxT("SmartFlex")} }; */ // Master list. Look up local name table first.. then these if not found. map DefaultMCShortNames; /*= { {CPAP_Obstructive, wxT("OA")}, {CPAP_Hypopnea, wxT("H")}, {CPAP_RERA, wxT("RE")}, {CPAP_ClearAirway, wxT("CA")}, {CPAP_CSR, wxT("CSR")}, {CPAP_VSnore, wxT("VS")}, {CPAP_FlowLimit, wxT("FL")}, {CPAP_Pressure, wxT("P")}, {CPAP_Leak, wxT("LR")}, {CPAP_EAP, wxT("EAP")}, {CPAP_IAP, wxT("IAP")}, {PRS1_VSnore2, wxT("VS")}, {PRS1_PressurePulse,wxT("PP")} }; */ // Master list. Look up local name table first.. then these if not found. map DefaultMCLongNames; /*= { {CPAP_Obstructive, wxT("Obstructive Apnea")}, {CPAP_Hypopnea, wxT("Hypopnea")}, {CPAP_RERA, wxT("Respiratory Effort / Arrousal")}, {CPAP_ClearAirway, wxT("Clear Airway Apnea")}, {CPAP_CSR, wxT("Cheyne Stokes Respiration")}, {CPAP_VSnore, wxT("Vibratory Snore")}, {CPAP_FlowLimit, wxT("Flow Limitation")}, {CPAP_Pressure, wxT("Pressure")}, {CPAP_Leak, wxT("Leak Rate")}, {CPAP_EAP, wxT("BIPAP EPAP")}, {CPAP_IAP, wxT("BIPAP IPAP")}, {PRS1_VSnore2, wxT("Vibratory Snore")}, {PRS1_PressurePulse,wxT("Pressue Pulse")} }; */ inline const QString & _(const QString q) { return q; }; void InitMapsWithoutAwesomeInitializerLists() { CPAPModeNames[MODE_UNKNOWN]=_("Undetermined"); CPAPModeNames[MODE_CPAP]=_("CPAP"); CPAPModeNames[MODE_APAP]=_("Auto"); CPAPModeNames[MODE_BIPAP]=_("BIPAP"); CPAPModeNames[MODE_ASV]=_("ASV"); PressureReliefNames[PR_UNKNOWN]=_("Undetermined"); PressureReliefNames[PR_NONE]=_("None"); PressureReliefNames[PR_CFLEX]=_("C-Flex"); PressureReliefNames[PR_CFLEXPLUS]=_("C-Flex+"); PressureReliefNames[PR_AFLEX]=_("A-Flex"); PressureReliefNames[PR_BIFLEX]=_("Bi-Flex"); PressureReliefNames[PR_EPR]=_("Exhalation Pressure Relief (EPR)"); PressureReliefNames[PR_SMARTFLEX]=_("SmartFlex"); DefaultMCShortNames[CPAP_Obstructive]=_("OA"); DefaultMCShortNames[CPAP_Hypopnea]=_("H"); DefaultMCShortNames[CPAP_RERA]=_("RE"); DefaultMCShortNames[CPAP_ClearAirway]=_("CA"); DefaultMCShortNames[CPAP_CSR]=_("CSR/PB"); DefaultMCShortNames[CPAP_VSnore]=_("VS"); DefaultMCShortNames[CPAP_FlowLimit]=_("FL"); DefaultMCShortNames[CPAP_Pressure]=_("P"); DefaultMCShortNames[CPAP_Leak]=_("LR"); DefaultMCShortNames[CPAP_EAP]=_("EPAP"); DefaultMCShortNames[CPAP_IAP]=_("IPAP"); DefaultMCShortNames[CPAP_Snore]=_("VS2"); DefaultMCShortNames[PRS1_PressurePulse]=_("PP"); DefaultMCLongNames[CPAP_Obstructive]=_("Obstructive Apnea"); DefaultMCLongNames[CPAP_Hypopnea]=_("Hypopnea"); DefaultMCLongNames[CPAP_RERA]=_("RERA"); DefaultMCLongNames[CPAP_ClearAirway]=_("Clear Airway Apnea"); DefaultMCLongNames[CPAP_CSR]=_("Periodic Breathing"); DefaultMCLongNames[CPAP_VSnore]=_("Vibratory Snore"); DefaultMCLongNames[CPAP_FlowLimit]=_("Flow Limitation"); DefaultMCLongNames[CPAP_Pressure]=_("Pressure"); DefaultMCLongNames[CPAP_Leak]=_("Leak Rate"); DefaultMCLongNames[CPAP_EAP]=_("BIPAP EPAP"); DefaultMCLongNames[CPAP_IAP]=_("BIPAP IPAP"); DefaultMCLongNames[CPAP_Snore]=_("Vibratory Snore 2"); DefaultMCLongNames[PRS1_PressurePulse]=_("Pressure Pulse"); DefaultMCLongNames[PRS1_Unknown0E]=_("Unknown 0E"); DefaultMCLongNames[PRS1_Unknown00]=_("Unknown 00"); DefaultMCLongNames[PRS1_Unknown01]=_("Unknown 01"); DefaultMCLongNames[PRS1_Unknown0B]=_("Unknown 0B"); DefaultMCLongNames[PRS1_Unknown10]=_("Unknown 10"); } // This is technically gui related.. however I have something in mind for it. /*const map DefaultFlagTypes= { {CPAP_Obstructive, FT_BAR}, {CPAP_Hypopnea, FT_BAR}, {CPAP_RERA, FT_BAR}, {CPAP_VSnore, FT_BAR}, {PRS1_VSnore2, FT_BAR}, {CPAP_FlowLimit, FT_BAR}, {CPAP_ClearAirway, FT_BAR}, {CPAP_CSR, FT_SPAN}, {PRS1_PressurePulse,FT_DOT}, {CPAP_Pressure, FT_DOT} }; const unsigned char flagalpha=0x80; const map DefaultFlagColours= { {CPAP_Obstructive, wxColour(0x80,0x80,0xff,flagalpha)}, {CPAP_Hypopnea, wxColour(0x00,0x00,0xff,flagalpha)}, {CPAP_RERA, wxColour(0x40,0x80,0xff,flagalpha)}, {CPAP_VSnore, wxColour(0xff,0x20,0x20,flagalpha)}, {CPAP_FlowLimit, wxColour(0x20,0x20,0x20,flagalpha)}, {CPAP_ClearAirway, wxColour(0xff,0x40,0xff,flagalpha)}, {CPAP_CSR, wxColour(0x40,0xff,0x40,flagalpha)}, {PRS1_VSnore2, wxColour(0xff,0x20,0x20,flagalpha)}, {PRS1_PressurePulse,wxColour(0xff,0x40,0xff,flagalpha)} }; */ ////////////////////////////////////////////////////////////////////////////////////////// // Machine Base-Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// Machine::Machine(Profile *p,MachineID id) { day.clear(); highest_sessionid=0; profile=p; if (!id) { std::tr1::minstd_rand gen; std::tr1::uniform_int unif(1, 0x7fffffff); gen.seed((unsigned int) time(NULL)); MachineID temp; do { temp = unif(gen); //unif(gen) << 32 | } while (profile->machlist.find(temp)!=profile->machlist.end()); m_id=temp; } else m_id=id; qDebug("Create Machine: %lx",m_id); m_type=MT_UNKNOWN; firstsession=true; } Machine::~Machine() { qDebug("Destroy Machine"); map::iterator d; for (d=day.begin();d!=day.end();d++) { delete d->second; } } Session *Machine::SessionExists(SessionID session) { if (sessionlist.find(session)!=sessionlist.end()) { return sessionlist[session]; } else { return NULL; } } Day *Machine::AddSession(Session *s,Profile *p) { double span; assert(s!=NULL); assert(p!=NULL); if (s->session()>highest_sessionid) highest_sessionid=s->session(); QDateTime date=s->first(); date=date.addDays(-1); date.setTime(QTime(0,0)); const int hours_since_last_session=4; const int hours_since_midnight=12; bool previous=false; // Find what day session belongs to. if (day.find(date)!=day.end()) { // Previous day record exists... // Calculate time since end of previous days last session span=(s->first().toTime_t() - day[date]->last().toTime_t())/3600.0; // less than n hours since last session yesterday? if (span < hours_since_last_session) { previous=true; } } if (!previous) { // Calculate Hours since midnight. QDateTime t=s->first(); t.setTime(QTime(0,0)); span=(s->first().toTime_t() - t.toTime_t())/3600.0; // Bedtime was late last night. if (span < hours_since_midnight) { previous=true; } } if (!previous) { // Revert to sessions original day. date=date.addDays(1); } sessionlist[s->session()]=s; if (!firstsession) { if (firstday>date) firstday=date; if (lastdayAddDay(date.date(),day[date],m_type); } day[date]->AddSession(s); return day[date]; } // This functions purpose is murder and mayhem... It deletes all of a machines data. // Therefore this is the most dangerous function in this software.. bool Machine::Purge(int secret) { // Boring api key to stop this function getting called by accident :) if (secret!=3478216) return false; // It would be joyous if this function screwed up.. QString path=profile->Get("DataFolder")+"/"+hexid(); QDir dir(path); if (!dir.exists() || !dir.isReadable()) return false; QString native="Purging "+QDir::toNativeSeparators(path); qDebug(native.toLatin1()); dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks); dir.setSorting(QDir::Name); QFileInfoList list=dir.entryInfoList(); int could_not_kill=0; for (int i=0;i0) { qWarning(("Could not purge path\n"+path+"\n\n%i file(s) remain.. Suggest manually deleting this path\n").toLatin1(),could_not_kill); return false; } return true; } bool Machine::Load() { QString path=profile->Get("DataFolder")+"/"+hexid(); QDir dir(path); qDebug(("Loading "+path).toLatin1()); if (!dir.exists() || !dir.isReadable()) return false; dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks); dir.setSorting(QDir::Name); QFileInfoList list=dir.entryInfoList(); typedef vector StringList; map sessfiles; map::iterator s; QString fullpath,ext_s,sesstr; int ext; SessionID sessid; bool ok; for (int i=0;isetValue((float(cnt)/float(size)*100.0)); Session *sess=new Session(this,s->first); if (sess->LoadSummary(s->second[0])) { sess->SetEventFile(s->second[1]); sess->SetWaveFile(s->second[2]); AddSession(sess,profile); } else { delete sess; } } if (qprogress) qprogress->setValue(100); return true; } bool Machine::SaveSession(Session *sess) { QString path=profile->Get("DataFolder")+"/"+hexid(); if (sess->IsChanged()) sess->Store(path); } bool Machine::Save() { map::iterator d; vector::iterator s; int size=0; int cnt=0; QString path=profile->Get("DataFolder")+"/"+hexid(); // Calculate size for progress bar for (d=day.begin();d!=day.end();d++) size+=d->second->size(); for (d=day.begin();d!=day.end();d++) { for (s=d->second->begin(); s!=d->second->end(); s++) { cnt++; if (qprogress) qprogress->setValue(50+(float(cnt)/float(size)*50.0)); if ((*s)->IsChanged()) (*s)->Store(path); } } return true; } ////////////////////////////////////////////////////////////////////////////////////////// // CPAP implmementation ////////////////////////////////////////////////////////////////////////////////////////// CPAP::CPAP(Profile *p,MachineID id):Machine(p,id) { m_type=MT_CPAP; // FlagColours=DefaultFlagColours; } CPAP::~CPAP() { } ////////////////////////////////////////////////////////////////////////////////////////// // Oximeter Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// Oximeter::Oximeter(Profile *p,MachineID id):Machine(p,id) { m_type=MT_OXIMETER; } Oximeter::~Oximeter() { } ////////////////////////////////////////////////////////////////////////////////////////// // SleepStage Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// SleepStage::SleepStage(Profile *p,MachineID id):Machine(p,id) { m_type=MT_SLEEPSTAGE; } SleepStage::~SleepStage() { }