From 088d4ea48ae060d94ee0754932013072ee7fae74 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Thu, 24 Nov 2011 22:47:25 +1000 Subject: [PATCH] Oximetry Tab & CMS50 Serial module Complete Rewrite --- Graphs/gFlagsLine.cpp | 2 +- Graphs/gGraphView.cpp | 2 +- Graphs/gLineChart.cpp | 20 +- Graphs/gLineOverlay.cpp | 2 +- Graphs/gSegmentChart.cpp | 2 +- Resources.qrc | 1 + SleepLib/event.cpp | 6 +- SleepLib/event.h | 10 +- SleepLib/loader_plugins/intellipap_loader.cpp | 38 +- SleepLib/machine_common.h | 3 + SleepLib/session.cpp | 14 +- daily.cpp | 12 +- daily.h | 1 - docs/channels.xml | 5 +- exportcsv.cpp | 2 +- oximetry.cpp | 1266 +++++++++-------- oximetry.h | 139 +- oximetry.ui | 131 ++ 18 files changed, 1034 insertions(+), 622 deletions(-) diff --git a/Graphs/gFlagsLine.cpp b/Graphs/gFlagsLine.cpp index 82e4bbf4..2017a303 100644 --- a/Graphs/gFlagsLine.cpp +++ b/Graphs/gFlagsLine.cpp @@ -136,7 +136,7 @@ void gFlagsLine::paint(gGraph & w,int left, int top, int width, int height) EventList & el=*((*s)->eventlist[m_code][0]); - for (int i=0;i::iterator i=m_layers.begin();i!=m_layers.end();i++) { (*i)->keyPressEvent(event); } - qDebug() << m_title << "Key Pressed.. implement me" << event->key(); + //qDebug() << m_title << "Key Pressed.. implement me" << event->key(); } void gGraph::ZoomX(double mult,int origin_px) diff --git a/Graphs/gLineChart.cpp b/Graphs/gLineChart.cpp index 1f5d5065..55bb60b0 100644 --- a/Graphs/gLineChart.cpp +++ b/Graphs/gLineChart.cpp @@ -532,14 +532,19 @@ void AHIChart::SetDay(Day *d) for (qint64 ti=first;tibegin();s!=d->end();s++) { Session *sess=*s; if ((tifirst()) || (f>sess->last())) continue; + // Drop off suddenly outside of sessions + //if (ti>sess->last()) continue; + + fnd=true; if (sess->eventlist.contains(CPAP_Obstructive)) el[0]=sess->eventlist[CPAP_Obstructive][0]; else el[0]=NULL; @@ -552,18 +557,25 @@ void AHIChart::SetDay(Day *d) if (sess->eventlist.contains(CPAP_ClearAirway)) el[3]=sess->eventlist[CPAP_ClearAirway][0]; else el[3]=NULL; + if (sess->eventlist.contains(CPAP_NRI)) + el[4]=sess->eventlist[CPAP_NRI][0]; + else el[4]=NULL; + /*if (sess->eventlist.contains(CPAP_ExP)) + el[5]=sess->eventlist[CPAP_ExP][0]; + else el[5]=NULL;*/ qint64 t; - for (int i=0;i<4;i++) { + for (int i=0;i<5;i++) { if (!el[i]) continue; - for (int j=0;jcount();j++) { + for (quint32 j=0;jcount();j++) { t=el[i]->time(j); - if ((t>=f) && (t<=ti)) { + if ((t>=f) && (t0) ahi=cnt/g; diff --git a/Graphs/gLineOverlay.cpp b/Graphs/gLineOverlay.cpp index a0bb85a1..608b860a 100644 --- a/Graphs/gLineOverlay.cpp +++ b/Graphs/gLineOverlay.cpp @@ -62,7 +62,7 @@ void gLineOverlayBar::paint(gGraph & w, int left, int topp, int width, int heigh EventList & el=*cei.value()[0]; - for (int i=0;iicons/toggle-off-us.svg icons/toggle-on-us.svg docs/release_notes.html + icons/save.png diff --git a/SleepLib/event.cpp b/SleepLib/event.cpp index cb081358..e468ee21 100644 --- a/SleepLib/event.cpp +++ b/SleepLib/event.cpp @@ -26,7 +26,7 @@ EventList::EventList(EventListType et,EventDataType gain, EventDataType offset, EventList::~EventList() { } -qint64 EventList::time(int i) +qint64 EventList::time(quint32 i) { if (m_type==EVL_Event) { return m_first+qint64(m_time[i]); @@ -35,7 +35,7 @@ qint64 EventList::time(int i) return m_first+qint64((EventDataType(i)*m_rate)); } -EventDataType EventList::data(int i) +EventDataType EventList::data(quint32 i) { return EventDataType(m_data[i])*m_gain; } @@ -59,7 +59,7 @@ void EventList::AddEvent(qint64 time, EventStoreType data) // This really shouldn't happen. qint64 t=(m_first-time); - for (int i=0;i m_data; //ChannelID m_code; EventListType m_type; - int m_count; + quint32 m_count; EventDataType m_gain; EventDataType m_offset; diff --git a/SleepLib/loader_plugins/intellipap_loader.cpp b/SleepLib/loader_plugins/intellipap_loader.cpp index 79de32d1..595bad24 100644 --- a/SleepLib/loader_plugins/intellipap_loader.cpp +++ b/SleepLib/loader_plugins/intellipap_loader.cpp @@ -75,8 +75,8 @@ int IntellipapLoader::Open(QString & path,Profile *profile) lookup["Pl"]="MinPressure"; //lookup["Ds"]="Ds"; //lookup["Pc"]="Pc"; - lookup["Pd"]="RampPressure"; - lookup["Dt"]="RampTime"; + lookup["Pd"]="RampPressure"; // Delay Pressure + lookup["Dt"]="RampTime"; // Delay Time //lookup["Ld"]="Ld"; //lookup["Lh"]="Lh"; //lookup["FC"]="FC"; @@ -88,8 +88,8 @@ int IntellipapLoader::Open(QString & path,Profile *profile) lookup["Hd"]="HypopneaDuration"; //lookup["Pi"]="Pi"; //080 //lookup["Pe"]="Pe"; //WF - //lookup["Ri"]="SmartFlexIRnd"; //1 - //lookup["Re"]="SmartFlexERnd"; //2 + lookup["Ri"]="SmartFlexIRnd"; // Inhale Rounding (0-5) + lookup["Re"]="SmartFlexERnd"; // Exhale Rounding (0-5) //lookup["Bu"]="Bu"; //WF //lookup["Ie"]="Ie"; //20 //lookup["Se"]="Se"; //05 @@ -102,7 +102,7 @@ int IntellipapLoader::Open(QString & path,Profile *profile) //lookup["Hp"]="Hp"; //1 //lookup["Hs"]="Hs"; //02 //lookup["Lu"]="LowUseThreshold"; // defaults to 0 (4 hours) - //lookup["Sf"]="SmartFlex"; + lookup["Sf"]="SmartFlex"; //lookup["Sm"]="SmartFlexMode"; lookup["Ks=s"]="Ks_s"; lookup["Ks=i"]="Ks_i"; @@ -254,13 +254,22 @@ int IntellipapLoader::Open(QString & path,Profile *profile) sess->eventlist[CPAP_Te][0]->AddEvent(time,m_buffer[pos+0xf]); sess->eventlist[CPAP_Ti][0]->AddEvent(time,m_buffer[pos+0xc]); - sess->eventlist[CPAP_Snore][0]->AddEvent(time,m_buffer[pos+0x5]); //4/5?? + sess->eventlist[CPAP_Snore][0]->AddEvent(time,m_buffer[pos+0x4]); //4/5?? - if (m_buffer[pos+0x5]>0) { - if (!sess->eventlist.contains(CPAP_VSnore)) { - sess->AddEventList(CPAP_VSnore,EVL_Event); + // 0x0f == Leak Event + // 0x04 == Snore? + if (m_buffer[pos+0xf]>0) { // Leak Event + if (!sess->eventlist.contains(CPAP_LeakFlag)) { + sess->AddEventList(CPAP_LeakFlag,EVL_Event); } - sess->eventlist[CPAP_VSnore][0]->AddEvent(time,m_buffer[pos+0x5]); + sess->eventlist[CPAP_LeakFlag][0]->AddEvent(time,m_buffer[pos+0xf]); + } + + if (m_buffer[pos+0x5]>4) { // This matches Exhale Puff.. not sure why 4 + if (!sess->eventlist.contains(CPAP_ExP)) { + sess->AddEventList(CPAP_ExP,EVL_Event); + } + sess->eventlist[CPAP_ExP][0]->AddEvent(time,m_buffer[pos+0x5]); } if (m_buffer[pos+0x10]>0) { @@ -273,13 +282,14 @@ int IntellipapLoader::Open(QString & path,Profile *profile) if (!sess->eventlist.contains(CPAP_Hypopnea)) { sess->AddEventList(CPAP_Hypopnea,EVL_Event); } + //for (int i=0;ieventlist[CPAP_Hypopnea][0]->AddEvent(time,m_buffer[pos+0x11]); } - if (m_buffer[pos+0x12]>0) { - if (!sess->eventlist.contains(CPAP_Apnea)) { - sess->AddEventList(CPAP_Apnea,EVL_Event); + if (m_buffer[pos+0x12]>0) { // NRI // is this == to RERA?? CA?? + if (!sess->eventlist.contains(CPAP_NRI)) { + sess->AddEventList(CPAP_NRI,EVL_Event); } - sess->eventlist[CPAP_Apnea][0]->AddEvent(time,m_buffer[pos+0x12]); + sess->eventlist[CPAP_NRI][0]->AddEvent(time,m_buffer[pos+0x12]); } quint16 tv=(m_buffer[pos+0x8] << 8) | m_buffer[pos+0x9]; // correct sess->eventlist[CPAP_TidalVolume][0]->AddEvent(time,tv); diff --git a/SleepLib/machine_common.h b/SleepLib/machine_common.h index b05e3538..a1326ca9 100644 --- a/SleepLib/machine_common.h +++ b/SleepLib/machine_common.h @@ -72,6 +72,9 @@ const QString CPAP_Hypopnea="Hypopnea"; const QString CPAP_ClearAirway="ClearAirway"; const QString CPAP_Apnea="Apnea"; const QString CPAP_CSR="CSR"; +const QString CPAP_LeakFlag="LeakFlag"; +const QString CPAP_ExP="ExP"; +const QString CPAP_NRI="NRI"; const QString CPAP_VSnore="VSnore"; const QString CPAP_VSnore2="VSnore2"; const QString CPAP_RERA="RERA"; diff --git a/SleepLib/session.cpp b/SleepLib/session.cpp index 0eb2d455..16a721f7 100644 --- a/SleepLib/session.cpp +++ b/SleepLib/session.cpp @@ -274,11 +274,11 @@ bool Session::StoreEvents(QString filename) for (int j=0;j> t; //evec.m_data[c]=t; evec.m_data.push_back(t); @@ -392,7 +392,7 @@ bool Session::LoadEvents(QString filename) //last=evec.first(); if (evec.type()!=EVL_Waveform) { evec.m_time.reserve(evec.m_count); - for (int c=0;c> x; //last+=x; evec.m_time.push_back(x); @@ -591,7 +591,7 @@ double Session::sum(ChannelID id) double sum=0; for (int i=0;icount();j++) { + for (quint32 j=0;jcount();j++) { sum+=evec[i]->data(j); } } @@ -615,7 +615,7 @@ EventDataType Session::avg(ChannelID id) double val=0; int cnt=0; for (int i=0;icount();j++) { + for (quint32 j=0;jcount();j++) { val+=evec[i]->data(j); cnt++; } @@ -734,7 +734,7 @@ EventDataType Session::wavg(ChannelID id) if (!evec[i]->count()) continue; lastval=evec[i]->raw(0); lasttime=evec[i]->time(0); - for (int j=1;jcount();j++) { + for (quint32 j=1;jcount();j++) { val=evec[i]->raw(j); time=evec[i]->time(j); td=(time-lasttime); diff --git a/daily.cpp b/daily.cpp index 003cda89..519ca26c 100644 --- a/daily.cpp +++ b/daily.cpp @@ -112,10 +112,13 @@ Daily::Daily(QWidget *parent,gGraphView * shared, MainWindow *mw) gFlagsGroup *fg=new gFlagsGroup(); fg->AddLayer((new gFlagsLine(CPAP_CSR,QColor("light green"),"CSR",false,FT_Span))); - fg->AddLayer((new gFlagsLine(CPAP_ClearAirway,QColor("purple"),"CA",true))); + fg->AddLayer((new gFlagsLine(CPAP_ClearAirway,QColor("purple"),"CA",false))); fg->AddLayer((new gFlagsLine(CPAP_Obstructive,QColor("#40c0ff"),"OA",true))); fg->AddLayer((new gFlagsLine(CPAP_Apnea,QColor("dark green"),"A"))); fg->AddLayer((new gFlagsLine(CPAP_Hypopnea,QColor("blue"),"H",true))); + fg->AddLayer((new gFlagsLine(CPAP_ExP,QColor("dark cyan"),"E",false))); + fg->AddLayer((new gFlagsLine(CPAP_LeakFlag,QColor("dark blue"),"L",false))); + fg->AddLayer((new gFlagsLine(CPAP_NRI,QColor("dark magenta"),"NRI",false))); fg->AddLayer((new gFlagsLine(CPAP_FlowLimit,QColor("black"),"FL"))); fg->AddLayer((new gFlagsLine(CPAP_RERA,QColor("gold"),"RE"))); fg->AddLayer((new gFlagsLine(CPAP_VSnore,QColor("red"),"VS"))); @@ -305,7 +308,7 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day) mcr=mcroot[code]; } for (int z=0;zcount();o++) { + for (quint32 o=0;ocount();o++) { qint64 t=m.value()[z]->time(o); if (code==CPAP_CSR) { @@ -469,7 +472,7 @@ void Daily::Load(QDate date) float csr=(100.0/cpap->hours())*(cpap->sum(CPAP_CSR)/3600.0); float uai=cpap->count(CPAP_Apnea)/cpap->hours(); float oai=cpap->count(CPAP_Obstructive)/cpap->hours(); - float hi=cpap->count(CPAP_Hypopnea)/cpap->hours(); + float hi=(cpap->count(CPAP_ExP)+cpap->count(CPAP_Hypopnea))/cpap->hours(); float cai=cpap->count(CPAP_ClearAirway)/cpap->hours(); float rei=cpap->count(CPAP_RERA)/cpap->hours(); float vsi=cpap->count(CPAP_VSnore)/cpap->hours(); @@ -558,7 +561,7 @@ void Daily::Load(QDate date) CPAP_Pressure,CPAP_EPAP,CPAP_IPAP,CPAP_PS,CPAP_PTB, CPAP_MinuteVent,CPAP_RespRate,CPAP_RespEvent,CPAP_FLG, CPAP_Leak,CPAP_Snore,CPAP_IE,CPAP_Ti,CPAP_Te, CPAP_TgMV, - CPAP_TidalVolume, OXI_Pulse,OXI_SPO2 + CPAP_TidalVolume, OXI_Pulse, OXI_SPO2 }; int numchans=sizeof(chans)/sizeof(ChannelID); int suboffset; @@ -916,3 +919,4 @@ void Daily::on_todayButton_clicked() if (d > PROFILE.LastDay()) d=PROFILE.LastDay(); LoadDate(d); } + diff --git a/daily.h b/daily.h index 1919b434..46c9a86a 100644 --- a/daily.h +++ b/daily.h @@ -64,7 +64,6 @@ private slots: void on_calButton_toggled(bool checked); void on_todayButton_clicked(); - protected: private: diff --git a/docs/channels.xml b/docs/channels.xml index b2d3d050..b8ce6693 100644 --- a/docs/channels.xml +++ b/docs/channels.xml @@ -46,6 +46,9 @@ One id code per item + + + @@ -69,7 +72,7 @@ One id code per item - + diff --git a/exportcsv.cpp b/exportcsv.cpp index 6e1ef414..b19b27bf 100644 --- a/exportcsv.cpp +++ b/exportcsv.cpp @@ -245,7 +245,7 @@ void ExportCSV::on_exportButton_clicked() //header="DateTime"+sep+"Session"+sep+"Event"+sep+"Data/Duration"; for (int e=0;ecount();q++) { + for (quint32 q=0;qcount();q++) { data=QDateTime::fromTime_t(ev->time(q)/1000L).toString(Qt::ISODate); data+=sep+QString::number(sess->session()); data+=sep+key; diff --git a/oximetry.cpp b/oximetry.cpp index 70a3bc30..31d96795 100644 --- a/oximetry.cpp +++ b/oximetry.cpp @@ -2,6 +2,7 @@ #include #include #include +#include #include "oximetry.h" #include "ui_oximetry.h" @@ -13,6 +14,458 @@ #include "Graphs/gLineChart.h" #include "Graphs/gYAxis.h" +extern QLabel * qstatus2; + +int lastpulse; +SerialOximeter::SerialOximeter(QObject * parent,QString oxiname, QString portname, BaudRateType baud, FlowType flow, ParityType parity, DataBitsType databits, StopBitsType stopbits) : + QObject(parent), + session(NULL),pulse(NULL),spo2(NULL),plethy(NULL),m_port(NULL), + m_opened(false), + m_oxiname(oxiname), + m_portname(portname), + m_baud(baud), + m_flow(flow), + m_parity(parity), + m_databits(databits), + m_stopbits(stopbits) +{ + machine=PROFILE.GetMachine(MT_OXIMETER); + if (!machine) { + // Create generic Serial Oximeter object.. + CMS50Loader *l=dynamic_cast(GetLoader("CMS50")); + if (l) { + machine=l->CreateMachine(p_profile); + } + qDebug() << "Create Oximeter device"; + } +} + +SerialOximeter::~SerialOximeter() +{ + if (m_opened) { + if (m_port) m_port->close(); + } + // free up?? +} + +bool SerialOximeter::Open(QextSerialPort::QueryMode mode) +{ + if (m_portname.isEmpty()) { + qDebug() << "Tried to open with empty portname"; + return false; + } + + qDebug() << "Opening serial port" << m_portname << "in mode" << mode; + + if (m_opened) { // Open already? + // Just close it + if (m_port) m_port->close(); + } + + m_mode=mode; + + m_port=new QextSerialPort(m_portname,m_mode); + m_port->setBaudRate(m_baud); + m_port->setFlowControl(m_flow); + m_port->setParity(m_parity); + m_port->setDataBits(m_databits); + m_port->setStopBits(m_stopbits); + + if (m_port->open(QIODevice::ReadWrite) == true) { + // if (m_mode==QextSerialPort::EventDriven) + connect(m_port, SIGNAL(readyRead()), this, SLOT(onReadyRead())); + //connect(port, SIGNAL(dsrChanged(bool)), this, SLOT(onDsrChanged(bool))); + if (!(m_port->lineStatus() & LS_DSR)) + qDebug() << "check device is turned on"; + qDebug() << "listening for data on" << m_port->portName(); + return m_opened=true; + } else { + qDebug() << "device failed to open:" << m_port->errorString(); + return m_opened=false; + } +} + +void SerialOximeter::Close() +{ + qDebug() << "Closing serial port" << m_portname; + if (!m_opened) return; + + if (m_port) m_port->close(); + if (m_mode==QextSerialPort::EventDriven) + disconnect(m_port, SIGNAL(readyRead()), this, SLOT(onReadyRead())); + m_opened=false; +} + +void SerialOximeter::setPortName(QString portname) +{ + if (m_opened) { + qDebug() << "Can't change serial PortName settings while port is open!"; + return; + } + m_portname=portname; +} + +void SerialOximeter::setBaudRate(BaudRateType baud) +{ + if (m_opened) { + qDebug() << "Can't change serial BaudRate settings while port is open!"; + return; + } + m_baud=baud; +} + +void SerialOximeter::setFlowControl(FlowType flow) +{ + if (m_opened) { + qDebug() << "Can't change serial FlowControl settings while port is open!"; + return; + } + m_flow=flow; +} + +void SerialOximeter::setParity(ParityType parity) +{ + if (m_opened) { + qDebug() << "Can't change serial Parity settings while port is open!"; + return; + } + m_parity=parity; +} + +void SerialOximeter::setDataBits(DataBitsType databits) +{ + if (m_opened) { + qDebug() << "Can't change serial DataBit settings while port is open!"; + return; + } + m_databits=databits; +} + +void SerialOximeter::setStopBits(StopBitsType stopbits) +{ + if (m_opened) { + qDebug() << "Can't change serial StopBit settings while port is open!"; + return; + } + m_stopbits=stopbits; +} + +void SerialOximeter::onReadyRead() +{ + int i=5; + qDebug() << "Foo" << i; +} + +void SerialOximeter::addPulse(qint64 time, EventDataType pr) +{ + pulse->AddEvent(time,pr); + session->setCount(OXI_Pulse,pulse->count()); // update the cache + session->setMin(OXI_Pulse,pulse->min()); + session->setMax(OXI_Pulse,pulse->max()); + session->setLast(OXI_Pulse,time); + pulse->setLast(time); + session->set_last(lasttime); + emit(updatePulse(pr)); +} + +void SerialOximeter::addSpO2(qint64 time, EventDataType o2) +{ + spo2->AddEvent(time,o2); + session->setCount(OXI_SPO2,spo2->count()); // update the cache + session->setMin(OXI_SPO2,spo2->min()); + session->setMax(OXI_SPO2,spo2->max()); + session->setLast(OXI_SPO2,time); + session->set_last(lasttime); + spo2->setLast(time); + emit(updateSpO2(o2)); +} + +void SerialOximeter::addPlethy(qint64 time, EventDataType pleth) +{ + plethy->AddEvent(time,pleth); + session->setCount(OXI_Plethy,plethy->count()); // update the cache + session->setMin(OXI_Plethy,plethy->min()); + session->setMax(OXI_Plethy,plethy->max()); + session->setLast(OXI_Plethy,time); + session->set_last(lasttime); + plethy->setLast(time); +} + +/*void SerialOximeter::addEvents(EventDataType pr,EventDataType o2, EventDataType pleth) +{ + lasttime=qint64(QDateTime::currentDateTime().toTime_t())*1000L; + addPulse(lasttime,pr); + addSpO2(lasttime,o2); + addPlethy(lasttime,pleth); + session->set_last(lasttime); + //emit(dataChanged()); +}*/ + +Session *SerialOximeter::createSession() +{ + if (session) { + delete session; + } + int sid=QDateTime::currentDateTime().toTime_t(); + lasttime=qint64(sid)*1000L; + + session=new Session(machine,sid); + session->SetChanged(true); + + session->set_first(lasttime); + pulse=new EventList(EVL_Event); + spo2=new EventList(EVL_Event); + plethy=new EventList(EVL_Event); + session->eventlist[OXI_Pulse].push_back(pulse); + session->eventlist[OXI_SPO2].push_back(spo2); + session->eventlist[OXI_Plethy].push_back(plethy); + + session->setFirst(OXI_Pulse,lasttime); + session->setFirst(OXI_SPO2,lasttime); + session->setFirst(OXI_Plethy,lasttime); + + pulse->setFirst(lasttime); + spo2->setFirst(lasttime); + plethy->setFirst(lasttime); + + emit(sessionCreated(session)); + return session; +} + +bool SerialOximeter::startLive() +{ + import_mode=false; + if (Open(QextSerialPort::EventDriven)) { + createSession(); + return true; + } else { + return false; + } +} + +void SerialOximeter::stopLive() +{ + Close(); + emit(liveStopped(session)); +} + +CMS50Serial::CMS50Serial(QObject * parent, QString portname="") : + SerialOximeter(parent,"CMS50", portname, BAUD19200, FLOW_OFF, PAR_ODD, DATA_8, STOP_1) +{ + import_mode=false; +} + +CMS50Serial::~CMS50Serial() +{ +} + +void CMS50Serial::on_import_process() +{ + qDebug() << "CMS50 import complete. Processing" << data.size() << "bytes"; + unsigned char a,pl,o2,lastpl=0,lasto2=0; + int i=0; + int size=data.size(); + EventList * pulse=(session->eventlist[OXI_Pulse][0]); + EventList * spo2=(session->eventlist[OXI_SPO2][0]); + lasttime=f2time[0].toTime_t(); + session->SetSessionID(lasttime); + lasttime*=1000; + + session->set_first(lasttime); + pulse->setFirst(lasttime); + spo2->setFirst(lasttime); + + EventDataType plmin=999,plmax=0; + EventDataType o2min=100,o2max=0; + int plcnt=0,o2cnt=0; + while (i<(size-3)) { + a=data.at(i++); + pl=data.at(i++) ^ 0x80; + o2=data.at(i++); + if (pl==0) { + if (lastpl!=pl) { + pulse->setLast(lasttime); + if (pulse->min()min(); + if (pulse->max()>plmax) plmax=pulse->max(); + plcnt+=pulse->count(); + pulse=new EventList(EVL_Event); + session->eventlist[OXI_Pulse].push_back(pulse); + } + } else pulse->AddEvent(lasttime,pl); + if (o2==0) { + if (lasto2!=o2) { + spo2->setLast(lasttime); + if (spo2->min()min(); + if (spo2->max()>o2max) o2max=spo2->max(); + o2cnt+=spo2->count(); + spo2=new EventList(EVL_Event); + session->eventlist[OXI_SPO2].push_back(spo2); + } + } else spo2->AddEvent(lasttime,o2); + + + lasttime+=1000; + emit(updateProgress(float(i)/float(size))); + + lastpl=pl; + lasto2=o2; + } + pulse->setLast(lasttime); + spo2->setLast(lasttime); + session->set_last(lasttime); + session->setMin(OXI_Pulse,plmin); + session->setMax(OXI_Pulse,plmax); + session->setMin(OXI_SPO2,o2min); + session->setMax(OXI_SPO2,o2max); + session->setCount(OXI_Pulse,plcnt); + session->setCount(OXI_SPO2,o2cnt); + session->UpdateSummaries(); + emit(importComplete(session)); + disconnect(this,SIGNAL(importProcess()),this,SLOT(on_import_process())); +} +void CMS50Serial::onReadyRead() +{ + QByteArray bytes; + int a = m_port->bytesAvailable(); + bytes.resize(a); + m_port->read(bytes.data(), bytes.size()); + + int i=0; + + // Was going out of sync previously.. To fix this unfortunately requires 4.7 + +#if QT_VERSION >= QT_VERSION_CHECK(4,7,0) + qint64 current=QDateTime::currentMSecsSinceEpoch(); //double(QDateTime::currentDateTime().toTime_t())*1000L; + //qint64 since=current-lasttime; + //if (since>25) + lasttime=current; +#endif + // else (don't bother - we can work some magic at the end of recording.) + + int size=bytes.size(); + // Process all incoming serial data packets + unsigned char c; + while (ifirst()/1000L; + + d=QDateTime::fromTime_t(ti); + qDebug() << "Guessing session starting from CPAP data" << d; + } else { + qDebug() << "Can't guess start time, defaulting to 6pm yesterday" << d; + d=QDateTime::currentDateTime(); + d.setTime(QTime(18,0,0)); + d.addDays(-1); + } + f2time.push_back(d); + } + i+=2; + cntf6++; + } else continue; + } else { + if (cntf6>0) { + qDebug() << "Got Acknowledge Sequence" << cntf6; + i--; + if ((i+3)=datasize) || (((received_bytes/datasize)>0.7) && (size<250))) { + qDebug() << "End"; + static unsigned char b1[3]={0xf6,0xf6,0xf6}; + if (m_port->write((char *)b1,2)==-1) { + qDebug() << "Couldn't write closing bytes to CMS50"; + } + Close(); + emit(importProcess()); + } + break; + //read data blocks.. + } + /*if (size<200) && (received_bytes>=datasize) { + + } */ + } else { + if (bytes[i]&0x80) { // 0x80 == sync bit + EventDataType d=bytes[i+1] & 0x7f; + addPlethy(lasttime,d); + lasttime+=20; + i+=3; + } else { + addPulse(lasttime,bytes[i]); + addSpO2(lasttime,bytes[i+1]); + i+=2; + } + } + emit(dataChanged()); + } +} + +bool CMS50Serial::startImport() +{ + import_mode=true; + waitf6=true; + cntf6=0; + //QMessageBox::information(0,"!!!Important Notice!!!","This Oximetry import method does NOT allow syncing of Oximetry and CPAP data.\nIf you really wish to record your oximetry data and have sync, you have to use the Live View mode (click Start) with the Oximeter connected to a computer via USB cable all night..\nEven then it will be out a bit because of your CPAP machines realtime clock drifts.",QMessageBox::Ok); + if (!Open(QextSerialPort::EventDriven)) return false; + connect(this,SIGNAL(importProcess()),this,SLOT(on_import_process())); + + createSession(); + + static unsigned char b1[2]={0xf5,0xf5}; + + if (m_port->write((char *)b1,2)==-1) { + qDebug() << "Couldn't write data request bytes to CMS50"; + } + + return true; +} + + Oximetry::Oximetry(QWidget *parent,gGraphView * shared) : QWidget(parent), ui(new Ui::Oximetry) @@ -23,19 +476,9 @@ Oximetry::Oximetry(QWidget *parent,gGraphView * shared) : port=NULL; portname=""; - mach=p_profile->GetMachine(MT_OXIMETER); - if (!mach) { - CMS50Loader *l=dynamic_cast(GetLoader("CMS50")); - if (l) { - mach=l->CreateMachine(p_profile); - } - qDebug() << "Create Oximeter device"; - } + oximeter=new CMS50Serial(this); - // Create dummy day & session for holding eventlists. - day=new Day(mach); - session=new Session(mach,0); - day->AddSession(session); + day=new Day(oximeter->getMachine()); layout=new QHBoxLayout(ui->graphArea); layout->setSpacing(0); @@ -76,13 +519,6 @@ Oximetry::Oximetry(QWidget *parent,gGraphView * shared) : g->AddLayer(new gXGrid()); } - // Create the Event Lists to store / import data - ev_plethy=session->AddEventList(OXI_Plethy,EVL_Waveform,1,0,0,0,1000.0/50.0); - - ev_pulse=session->AddEventList(OXI_Pulse,EVL_Event,1); - - ev_spo2=session->AddEventList(OXI_SPO2,EVL_Event,1); - plethy=new gLineChart(OXI_Plethy,Qt::black,false,true); plethy->SetDay(day); @@ -104,7 +540,9 @@ Oximetry::Oximetry(QWidget *parent,gGraphView * shared) : GraphView->updateGL(); on_RefreshPortsButton_clicked(); + ui->RunButton->setChecked(false); + ui->saveButton->setEnabled(false); } Oximetry::~Oximetry() @@ -158,305 +596,132 @@ void Oximetry::on_RefreshPortsButton_clicked() ui->ImportButton->setEnabled(false); portname=""; } + oximeter->setPortName(portname); } void Oximetry::RedrawGraphs() { GraphView->updateGL(); } - -void Oximetry::on_RunButton_toggled(bool checked) -{ - if (checked) { - lasttime=0; - lastpulse=0; - lastspo2=0; - - // Wipe any current data - ev_plethy->getData().clear(); - ev_plethy->getTime().clear(); - ev_plethy->setCount(0); - ev_pulse->getData().clear(); - ev_pulse->getTime().clear(); - ev_pulse->setCount(0); - ev_spo2->getData().clear(); - ev_spo2->getTime().clear(); - ev_spo2->setCount(0); - - lasttime=QDateTime::currentDateTime().toTime_t()*1000L; // utc?? - starttime=lasttime; - - session->SetSessionID(lasttime/1000L); - - day->setFirst(lasttime); - day->setLast(lasttime+30000); - session->set_first(lasttime); - session->set_last(lasttime+30000); - - ev_plethy->setFirst(lasttime); - ev_plethy->setLast(lasttime+3600000); - PLETHY->SetMinX(lasttime); - PLETHY->SetMaxX(lasttime+30000); - CONTROL->SetMinX(lasttime); - CONTROL->SetMaxX(lasttime+30000); - - ev_pulse->setFirst(lasttime); - ev_pulse->setLast(lasttime+3600000); - PULSE->SetMinX(lasttime); - PULSE->SetMaxX(lasttime+30000); - - ev_spo2->setFirst(lasttime); - ev_spo2->setLast(lasttime+3600000); - SPO2->SetMinX(lasttime); - SPO2->SetMaxX(lasttime+30000); - - ui->RunButton->setText("&Stop"); - ui->SerialPortsCombo->setEnabled(false); - // Disconnect?? - port=new QextSerialPort(portname,QextSerialPort::EventDriven); - port->setBaudRate(BAUD19200); - port->setFlowControl(FLOW_OFF); - port->setParity(PAR_ODD); - port->setDataBits(DATA_8); - port->setStopBits(STOP_1); - if (port->open(QIODevice::ReadWrite) == true) { - connect(port, SIGNAL(readyRead()), this, SLOT(onReadyRead())); - connect(port, SIGNAL(dsrChanged(bool)), this, SLOT(onDsrChanged(bool))); - if (!(port->lineStatus() & LS_DSR)) - qDebug() << "check device is turned on"; - qDebug() << "listening for data on" << port->portName(); - } else { - qDebug() << "device failed to open:" << port->errorString(); - } - portmode=PM_LIVE; - //foobar->setVisible(false); - CONTROL->setVisible(false); - } else { - //foobar->setVisible(true); - ui->RunButton->setText("&Start"); - ui->SerialPortsCombo->setEnabled(true); - delete port; - port=NULL; - - ev_pulse->AddEvent(lasttime,lastpulse); - ev_spo2->AddEvent(lasttime,lastspo2); - ev_spo2->setLast(lasttime); - ev_pulse->setLast(lasttime); - ev_plethy->setLast(lasttime); - day->setLast(lasttime); - session->set_last(lasttime); - - - SPO2->SetMinX(ev_spo2->first()); - SPO2->SetMaxX(lasttime); - PULSE->SetMinX(ev_pulse->first()); - PULSE->SetMaxX(lasttime); - PLETHY->SetMinX(ev_plethy->first()); - PLETHY->SetMaxX(lasttime); - SPO2->MinX(); - SPO2->MaxX(); - PULSE->MinX(); - PULSE->MaxX(); - PLETHY->MinX(); - PLETHY->MaxX(); - //GraphView->ResetBounds(); - //CONTROL->SetMinX(PLETHY->MinX()); - //CONTROL->SetMaxX(PLETHY->MaxX()); - //CONTROL->SetMinY(ev_plethy->min()); - //CONTROL->SetMaxY(ev_plethy->max()); - CONTROL->MinX(); - CONTROL->MaxX(); - - //CONTROL->ResetBounds(); - - // qint64 d=session->length(); - // if (d<=30000) - // return; - if (ev_pulse->count()>1 && (ev_spo2->count()>1)) - if (QMessageBox::question(this,"Keep This Recording?","Would you like to keep this oximeter recording?",QMessageBox::Yes|QMessageBox::No)==QMessageBox::Yes) { - qDebug() << "Saving oximeter session data"; - - session->eventlist.clear(); - - Session *sess=new Session(mach,starttime/1000L); - - /*ev_spo2->setCode(CPAP_SPO2); - ev_pulse->setCode(CPAP_Pulse); - ev_plethy->setCode(CPAP_Plethy); */ - - sess->eventlist[OXI_SPO2].push_back(ev_spo2); - sess->eventlist[OXI_Pulse].push_back(ev_pulse); - sess->eventlist[OXI_Plethy].push_back(ev_plethy); - //Session *sess=session; - sess->SetSessionID(starttime/1000L); - - sess->setMin(OXI_Pulse,ev_pulse->min()); - sess->setMax(OXI_Pulse,ev_pulse->max()); - sess->setFirst(OXI_Pulse,ev_pulse->first()); - sess->setLast(OXI_Pulse,ev_pulse->last()); - sess->avg(OXI_Pulse); - sess->wavg(OXI_Pulse); - sess->p90(OXI_Pulse); - sess->count(OXI_Pulse); - - sess->setMin(OXI_SPO2,ev_spo2->min()); - sess->setMax(OXI_SPO2,ev_spo2->max()); - sess->setFirst(OXI_SPO2,ev_spo2->first()); - sess->setLast(OXI_SPO2,ev_spo2->last()); - sess->avg(OXI_SPO2); - sess->wavg(OXI_SPO2); - sess->p90(OXI_SPO2); - sess->count(OXI_SPO2); - - - sess->avg(OXI_Plethy); - sess->wavg(OXI_Plethy); - sess->p90(OXI_Plethy); - sess->count(OXI_Plethy); - - sess->setMin(OXI_Plethy,ev_plethy->min()); - sess->setMax(OXI_Plethy,ev_plethy->max()); - - sess->setFirst(OXI_Plethy,ev_plethy->first()); - sess->setLast(OXI_Plethy,ev_plethy->last()); - - sess->updateFirst(sess->first(OXI_Pulse)); - sess->updateLast(sess->last(OXI_Pulse)); - sess->updateFirst(sess->first(OXI_SPO2)); - sess->updateLast(sess->last(OXI_SPO2)); - sess->updateFirst(sess->first(OXI_Plethy)); - sess->updateLast(sess->last(OXI_Plethy)); - - sess->SetChanged(true); - mach->AddSession(sess,p_profile); - mach->Save(); - - ev_plethy=session->AddEventList(OXI_Plethy,EVL_Waveform,1,0,0,0,1000.0/50.0); - ev_pulse=session->AddEventList(OXI_Pulse,EVL_Event,1); - ev_spo2=session->AddEventList(OXI_SPO2,EVL_Event,1); - - session->setCount(OXI_Plethy,0); - session->setCount(OXI_Pulse,0); - session->setCount(OXI_SPO2,0); - - //m_shared->ResetBounds(); - //m_shared->updateScale(); - //m_shared->updateGL(); - - } - - CONTROL->setVisible(true); - GraphView->updateScale(); - //CONTROL->ResetBounds(); - GraphView->updateGL(); - } -} - void Oximetry::on_SerialPortsCombo_activated(const QString &arg1) { portname=arg1; -} -void Oximetry::UpdatePlethy(qint8 d) -{ - ev_plethy->getData().push_back(d); - if (dmin()) ev_plethy->setMin(d); - if (d>ev_plethy->max()) ev_plethy->setMax(d); - int i=ev_plethy->count()+1; - ev_plethy->setCount(i); - session->setCount(OXI_Plethy,i); // update the cache - //ev_plethy->AddEvent(lasttime,d); - lasttime+=20; // 50 samples per second - PLETHY->SetMinY(ev_plethy->min()); - PLETHY->SetMaxY(ev_plethy->max()); - CONTROL->SetMinY(ev_plethy->min()); - CONTROL->SetMaxY(ev_plethy->max()); - PULSE->SetMinY(ev_pulse->min()); - PULSE->SetMaxY(ev_pulse->max()); - SPO2->SetMinY(ev_spo2->min()); - SPO2->SetMaxY(ev_spo2->max()); - //PLETHY->MaxY(); - PLETHY->SetMaxX(lasttime); - PLETHY->SetMinX(lasttime-30000); - PULSE->SetMaxX(lasttime); - PULSE->SetMinX(lasttime-30000); - SPO2->SetMaxX(lasttime); - SPO2->SetMinX(lasttime-30000); - CONTROL->SetMaxX(lasttime); - CONTROL->SetMinX(lasttime-30000); - session->set_last(lasttime); - day->setLast(lasttime); - PLETHY->MinX(); - PLETHY->MaxX(); - CONTROL->MinX(); - CONTROL->MaxX(); -} -bool Oximetry::UpdatePulse(qint8 pul) -{ - bool ret=false; - - // Don't block zeros.. If the data is used, it's needed - // Can make the graph can skip them. - if (lastpulse!=pul) - { - ev_pulse->AddEvent(lasttime,pul); - session->setCount(OXI_Pulse,ev_pulse->count()); // update the cache - - ret=true; - //qDebug() << "Pulse=" << int(bytes[0]); - } - lastpulse=pul; - return ret; -} -bool Oximetry::UpdateSPO2(qint8 sp) -{ - bool ret=false; - - if (lastspo2!=sp) - { - ev_spo2->AddEvent(lasttime,sp); - session->setCount(OXI_SPO2,ev_spo2->count()); // update the cache - ret=true; - //qDebug() << "SpO2=" << int(bytes[1]); - } - - lastspo2=sp; - return ret; + oximeter->setPortName(arg1); } -void Oximetry::onReadyRead() +void Oximetry::on_RunButton_toggled(bool checked) { - QByteArray bytes; - int a = port->bytesAvailable(); - bytes.resize(a); - port->read(bytes.data(), bytes.size()); + if (!checked) { + oximeter->stopLive(); + ui->RunButton->setText("&Start"); + ui->SerialPortsCombo->setEnabled(true); + disconnect(oximeter,SIGNAL(dataChanged()),this,SLOT(onDataChanged())); + disconnect(oximeter,SIGNAL(updatePulse(float)),this,SLOT(onPulseChanged(float))); + disconnect(oximeter,SIGNAL(updateSpO2(float)),this,SLOT(onSpO2Changed(float))); + ui->saveButton->setEnabled(true); - int i=0; - while (isetVisible(true); + } else { + if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { + int res=QMessageBox::question(this,"Save Session?","Creating a new oximetry session will destroy the old one.\nWould you like to save it first?","Save","Destroy It","Cancel",0,2); + if (res==0) { + ui->RunButton->setChecked(false); + on_saveButton_clicked(); + return; + } else if (res==2) { + ui->RunButton->setChecked(false); + return; + } + } // else it's already saved. + + if (!oximeter->startLive()) { + QMessageBox::warning(this,"Error","Something is wrong with the device connection.",QMessageBox::Ok); + return; } - } + ui->saveButton->setEnabled(false); + day->getSessions().clear(); + day->AddSession(oximeter->getSession()); - if ((ev_plethy->count()<=2) || (ev_pulse->count()<=2) || (ev_spo2->count()<=2)) { - GraphView->updateScale(); + firstPulseUpdate=true; + firstSPO2Update=true; + secondPulseUpdate=true; + secondSPO2Update=true; + + qint64 f=oximeter->getSession()->first(); + day->setFirst(f); + plethy->setMinX(f); + pulse->setMinX(f); + spo2->setMinX(f); + PLETHY->SetMinX(f); + CONTROL->SetMinX(f); + PULSE->SetMinX(f); + SPO2->SetMinX(f); + + PLETHY->forceMinY(0); + PLETHY->forceMaxY(128); + PULSE->forceMinY(30); + PULSE->forceMaxY(180); + SPO2->forceMinY(50); + SPO2->forceMaxY(100); + + connect(oximeter,SIGNAL(dataChanged()),this,SLOT(onDataChanged())); + connect(oximeter,SIGNAL(updatePulse(float)),this,SLOT(onPulseChanged(float))); + connect(oximeter,SIGNAL(updateSpO2(float)),this,SLOT(onSpO2Changed(float))); + CONTROL->setVisible(false); + // connect. + ui->RunButton->setText("&Stop"); + ui->SerialPortsCombo->setEnabled(false); } - GraphView->updateGL(); } -void Oximetry::onDsrChanged(bool status) // Doesn't work for CMS50's +void Oximetry::onDataChanged() { - if (status) - qDebug() << "device was turned on"; - else - qDebug() << "device was turned off"; + + qint64 last=oximeter->lastTime(); + qint64 first=last-30000L; + day->setLast(last); + + plethy->setMinX(first); + plethy->setMaxX(last); + pulse->setMinX(first); + pulse->setMaxX(last); + spo2->setMinX(first); + spo2->setMaxX(last); + + plethy->setMinY(0); + plethy->setMaxY(128); + pulse->setMinY(0); + pulse->setMaxY(120); + spo2->setMinY(0); + spo2->setMaxY(100); + + PLETHY->MinY(); + PLETHY->MaxY(); + PULSE->MinY(); + PULSE->MaxY(); + SPO2->MinY(); + SPO2->MaxY(); + + PLETHY->SetMaxX(last); + PULSE->SetMaxX(last); + CONTROL->SetMaxX(last); + SPO2->SetMaxX(last); + + for (int i=0;isize();i++) { + (*GraphView)[i]->SetXBounds(first,last); + } + + { + int len=(last-first)/1000L; + int h=len/3600; + int m=(len /60) % 60; + int s=(len % 60); + if (qstatus2) qstatus2->setText(QString().sprintf("Rec %02i:%02i:%02i",h,m,s)); + } + + GraphView->updateGL(); } + extern QProgressBar *qprogress; extern QLabel *qstatus; @@ -473,284 +738,151 @@ void DumpBytes(int blocks, unsigned char * b,int len) // Move this code to CMS50 Importer?? void Oximetry::on_ImportButton_clicked() { - ui->ImportButton->setDisabled(true); - QMessageBox msgbox(QMessageBox::Information,"Importing","Please Wait",QMessageBox::NoButton,this); - msgbox.show(); - QApplication::processEvents(); - const int rb_size=0x200; - static unsigned char b1[2]={0xf5,0xf5}; - static unsigned char b2[3]={0xf6,0xf6,0xf6}; - static unsigned char rb[rb_size]; + connect(oximeter,SIGNAL(importComplete(Session*)),this,SLOT(on_import_complete(Session*))); + connect(oximeter,SIGNAL(importAborted()),this,SLOT(on_import_aborted())); + connect(oximeter,SIGNAL(updateProgress(float)),this,SLOT(on_updateProgress(float))); + //connect(oximeter,SIGNAL(dataChanged()),this,SLOT(onDataChanged())); - unsigned char * buffer=NULL; + if (!oximeter->startImport()) { + qDebug() << "Error starting oximetry serial import process"; + return; + } + day->getSessions().clear(); + day->AddSession(oximeter->getSession()); + + if (qprogress) { + qprogress->setValue(0); + qprogress->setMaximum(100); + qprogress->show(); + } + ui->ImportButton->setDisabled(true); ui->SerialPortsCombo->setEnabled(false); ui->RunButton->setText("&Start"); ui->RunButton->setChecked(false); +} - if (port) { - port->close(); - delete port; - } - // Disconnect?? - //qDebug() << "Initiating Polling Mode"; - port=new QextSerialPort(portname,QextSerialPort::Polling); - port->setBaudRate(BAUD19200); - port->setFlowControl(FLOW_OFF); - port->setParity(PAR_ODD); - port->setDataBits(DATA_8); - port->setStopBits(STOP_1); - port->setTimeout(500); - if (port->open(QIODevice::ReadWrite) == true) { - // if (!(port->lineStatus() & LS_DSR)) - // qDebug() << "warning: device is not turned on"; // CMS50 doesn't do this.. +void Oximetry::import_finished() +{ + disconnect(oximeter,SIGNAL(importComplete(Session*)),this,SLOT(on_import_complete(Session*))); + disconnect(oximeter,SIGNAL(importAborted()),this,SLOT(on_import_aborted())); + disconnect(oximeter,SIGNAL(updateProgress(float)),this,SLOT(on_updateProgress(float))); + //disconnect(oximeter,SIGNAL(dataChanged()),this,SLOT(onDataChanged())); - } else { - delete port; - port=NULL; - ui->SerialPortsCombo->setEnabled(true); - - return; - } - bool done=false; - int res; - int blocks=0; - unsigned int bytes=0; - QString aa; - port->flush(); - bool oneoff=false; - - //qprogress->reset(); - qstatus->setText("Importing"); - qprogress->setValue(0); - qprogress->show(); - int fails=0; - while (!done) { - if (port->write((char *)b1,2)==-1) { - qDebug() << "Couldn't write 2 lousy bytes to CMS50"; - } - blocks=0; - int startpos=0; - unsigned int length=0; - int dr; - int ec; - do { - bool fnd=false; - dr=0; - ec=0; - do { - res=port->read((char *)rb,rb_size); - DumpBytes(blocks,rb,res); - if (blocks>0) break; - if (res<=0) ec++; - dr+=res; - } while ((res<=5) && (dr<0x200) && (ec<5)); - - //if (res>5) DumpBytes(blocks,rb,res); - - - done=false; - if (blocks==0) { - if (res>5) - for (int i=0;isetValue((75.0/length)*bytes); - QApplication::processEvents(); - } else { - qprogress->setValue((75.0/length)*bytes); - QApplication::processEvents(); - - memcpy((char *)&buffer[bytes],(char *)rb,res); - bytes+=res; - } - - blocks++; - if (res4) break; - } - if (done) { - if (oneoff) bytes--; // this is retarded.. - - QDateTime date=QDateTime::currentDateTime().toUTC(); - SessionID sid=date.toTime_t(); - session->SetSessionID(sid); - qDebug() << "Read " << bytes << "Bytes"; - qDebug() << "Creating session " << sid; - unsigned short pulse,spo2,lastpulse=0,lastspo2=0; - - qint64 tt=sid-(bytes/3); - tt*=1000; - session->set_first(tt); - ev_pulse->setMin(999999); - ev_pulse->setMax(-999999); - ev_spo2->setMin(999999); - ev_spo2->setMax(-999999); - - ev_pulse->setFirst(tt); - ev_spo2->setFirst(tt); - - EventList *oxf1=NULL,*oxf2=NULL; - EventDataType data; - unsigned i=0; - const int rb_size=60; // last rb_size seconds of data - unsigned rb_pulse[rb_size]={0}; - unsigned rb_spo2[rb_size]={0}; - int rb_pos=0; - while (iAddEvent(tt,data); - //qDebug() << "Pulse: " << int(pulse); - } - if (spo2!=lastspo2) { - data=spo2; - ev_spo2->AddEvent(tt,data); - //qDebug() << "SpO2: " << int(spo2); - } - - lastpulse=pulse; - lastspo2=spo2; - - rb_pulse[rb_pos]=(unsigned)pulse; - rb_spo2[rb_pos]=(unsigned)spo2; - - unsigned int min=255,max=0; - for (int k=rb_pos;k>rb_pos-4;k--) { - int j=abs(k % rb_size); - if (rb_pulse[j]min) max=rb_pulse[j]; - } - if (min>0 && max>0) { - int drop=max-min; - if (drop>6) { - if (!oxf1) { - oxf1=session->AddEventList(OXI_PulseChange,EVL_Event); - } - oxf1->AddEvent(tt,drop); - } - } - - min=255,max=0; - for (int k=rb_pos;k>rb_pos-10;k--) { - int j=abs(k % rb_size); - if (rb_spo2[j]min) max=rb_spo2[j]; - } - if (min>0 && max>0) { - int drop=max-min; - if (drop>4) { - if (!oxf1) { - oxf2=session->AddEventList(OXI_SPO2Drop,EVL_Event); - } - oxf2->AddEvent(tt,drop); - } - } - - ++rb_pos; - rb_pos=rb_pos % rb_size; - - tt+=1000; - qprogress->setValue(75+(25.0/bytes)*i); - QApplication::processEvents(); - } - ev_pulse->AddEvent(tt,pulse); - ev_spo2->AddEvent(tt,spo2); - session->set_last(tt); - - session->m_cnt.clear(); - - session->setMin(OXI_Pulse,ev_pulse->min()); - session->setMax(OXI_Pulse,ev_pulse->max()); - session->avg(OXI_Pulse); - session->p90(OXI_Pulse); - session->cph(OXI_Pulse); - session->wavg(OXI_Pulse); - session->count(OXI_Pulse); - - session->setMin(OXI_SPO2,ev_pulse->min()); - session->setMax(OXI_SPO2,ev_pulse->max()); - session->avg(OXI_SPO2); - session->p90(OXI_SPO2); - session->cph(OXI_SPO2); - session->wavg(OXI_SPO2); - session->count(OXI_SPO2); - - session->SetChanged(true); - mach->AddSession(session,p_profile); - mach->Save(); - // Output Pulse & SPO2 here.. - delete [] buffer; - port->write((char *)b2,3); - - // Need to create a new session as this one got pinched. - session=new Session(mach,0); - day->getSessions().clear(); - day->AddSession(session); - - // As did these - ev_plethy=session->AddEventList(OXI_Plethy,EVL_Waveform,1,0,0,0,1000.0/50.0); - - ev_pulse=session->AddEventList(OXI_Pulse,EVL_Event,1); - - ev_spo2=session->AddEventList(OXI_SPO2,EVL_Event,1); - } - delete port; - port=NULL; - msgbox.hide(); - qprogress->hide(); ui->SerialPortsCombo->setEnabled(true); qstatus->setText("Ready"); ui->ImportButton->setDisabled(false); + ui->saveButton->setEnabled(true); + + if (qprogress) { + qprogress->setValue(100); + QApplication::processEvents(); + qprogress->hide(); + } +} + +void Oximetry::on_import_aborted() +{ + import_finished(); +} + +void Oximetry::on_import_complete(Session * session) +{ + qDebug() << "Oximetry import complete"; + import_finished(); + + PLETHY->setVisible(false); + CONTROL->setVisible(false); + + qint64 f=session->first(); + qint64 l=session->last(); + day->setFirst(f); + day->setLast(l); + + plethy->setMinX(f); + pulse->setMinX(f); + spo2->setMinX(f); + PLETHY->SetMinX(f); + CONTROL->SetMinX(f); + PULSE->SetMinX(f); + SPO2->SetMinX(f); + + plethy->setMaxX(l); + pulse->setMaxX(l); + spo2->setMaxX(l); + PLETHY->SetMaxX(l); + CONTROL->SetMaxX(l); + PULSE->SetMaxX(l); + SPO2->SetMaxX(l); + + + PLETHY->forceMinY(0); + PLETHY->forceMaxY(128); + PULSE->forceMinY(30); + PULSE->forceMaxY(180); + SPO2->forceMinY(50); + SPO2->forceMaxY(100); + + PULSE->setDay(day); + SPO2->setDay(day); + for (int i=0;isize();i++) { + (*GraphView)[i]->SetXBounds(f,l); + } + + { + int len=(l-f)/1000L; + int h=len/3600; + int m=(len /60) % 60; + int s=(len % 60); + if (qstatus2) qstatus2->setText(QString().sprintf("%02i:%02i:%02i",h,m,s)); + } + GraphView->updateScale(); + + GraphView->updateGL(); } +void Oximetry::onPulseChanged(float p) +{ + ui->pulseLCD->display(p); + if (firstPulseUpdate) { + if (secondPulseUpdate) { + secondPulseUpdate=false; + } else { + firstPulseUpdate=false; + GraphView->updateScale(); + } + } +} + +// Only really need to do this once.. +void Oximetry::onSpO2Changed(float o2) +{ + ui->spo2LCD->display(o2); + if (firstSPO2Update) { + if (secondSPO2Update) { + secondSPO2Update=false; + } else { + firstSPO2Update=false; + GraphView->updateScale(); + } + } +} + +void Oximetry::on_saveButton_clicked() +{ + if (QMessageBox::question(this,"Keep This Recording?","A save dialog will go here allowing you to edit parameters for this oximetry session..\nFor now, would you like to save this as is?",QMessageBox::Yes|QMessageBox::No)==QMessageBox::Yes) { + Session *session=oximeter->getSession(); + // Process??? + + session->UpdateSummaries(); + oximeter->getMachine()->AddSession(session,p_profile); + oximeter->getMachine()->Save(); + } +} +void Oximetry::on_updateProgress(float f) +{ + if (qprogress) { + qprogress->setValue(f*100.0); + QApplication::processEvents(); + } +} diff --git a/oximetry.h b/oximetry.h index 77c95c28..f71c0c05 100644 --- a/oximetry.h +++ b/oximetry.h @@ -20,6 +20,111 @@ #include "Graphs/gLineChart.h" #include "Graphs/gFooBar.h" +class SerialOximeter:public QObject +{ + Q_OBJECT +public: + explicit SerialOximeter(QObject * parent,QString oxiname, QString portname="",BaudRateType baud=BAUD19200, FlowType flow=FLOW_OFF, ParityType parity=PAR_ODD, DataBitsType databits=DATA_8, StopBitsType stopbits=STOP_1); + virtual ~SerialOximeter(); + + virtual bool Open(QextSerialPort::QueryMode mode=QextSerialPort::EventDriven); + virtual void Close(); + + virtual bool startImport()=0; + virtual void stopImport() {} // abort, default do nothing. + + virtual bool startLive(); + virtual void stopLive(); + + qint64 lastTime() { return lasttime; } + Machine * getMachine() { return machine; } + + Session *createSession(); + Session * getSession() { return session; } + Session * takeSession() { Session * s=session; session=NULL; return s; } + + void setPortName(QString portname); + void setBaudRate(BaudRateType baud); + void setFlowControl(FlowType flow); + void setParity(ParityType parity); + void setDataBits(DataBitsType databits); + void setStopBits(StopBitsType stopbits); + + QString portName() { return m_portname; } + BaudRateType baudRate() { return m_baud; } + FlowType flowControl() { return m_flow; } + ParityType parity() { return m_parity; } + DataBitsType dataBits() { return m_databits; } + StopBitsType stopBits() { return m_stopbits; } + +signals: + void sessionCreated(Session *); + void dataChanged(); + void importProcess(); + void importComplete(Session *); + void importAborted(); + void updateProgress(float f); // between 0 and 1. + void liveStopped(Session *); + + void updatePulse(float p); + void updateSpO2(float p); + +protected slots: + virtual void onReadyRead(); + virtual void on_import_process()=0; + + +protected: + //virtual void addEvents(EventDataType pr, EventDataType o2, EventDataType pleth=-1000000); + + virtual void addPulse(qint64 time, EventDataType pr); + virtual void addSpO2(qint64 time, EventDataType o2); + virtual void addPlethy(qint64 time, EventDataType pleth); + + + Session * session; + + EventList * pulse; + EventList * spo2; + EventList * plethy; + QextSerialPort *m_port; + + bool m_opened; + QString m_oxiname; + QString m_portname; + BaudRateType m_baud; + FlowType m_flow; + ParityType m_parity; + DataBitsType m_databits; + StopBitsType m_stopbits; + QextSerialPort::QueryMode m_mode; + Machine *machine; + + qint64 lasttime; + bool import_mode; +}; + +class CMS50Serial:public SerialOximeter +{ +public: + explicit CMS50Serial(QObject * parent,QString portname); + virtual ~CMS50Serial(); + virtual bool startImport(); + +protected: + virtual void on_import_process(); + + virtual void onReadyRead(); + bool waitf6; + short cntf6; + + QByteArray data; + QVector f2time; + int datasize; + + int received_bytes; +}; + namespace Ui { class Oximetry; } @@ -42,16 +147,22 @@ private slots: void on_RunButton_toggled(bool checked); void on_SerialPortsCombo_activated(const QString &arg1); - void onReadyRead(); - void onDsrChanged(bool status); + //void onReadyRead(); + //void onDsrChanged(bool status); void on_ImportButton_clicked(); -private: - bool UpdatePulse(qint8 pulse); - bool UpdateSPO2(qint8 spo2); - void UpdatePlethy(qint8 plethy); + void onDataChanged(); + void onPulseChanged(float p); + void onSpO2Changed(float o2); + void on_saveButton_clicked(); + void on_updateProgress(float f); + void on_import_aborted(); + void on_import_complete(Session *session); + +private: + void import_finished(); Ui::Oximetry *ui; gGraphView *GraphView; @@ -69,14 +180,20 @@ private: double lasttime,starttime; int lastpulse, lastspo2; - Machine * mach; Day * day; - Session * session; - EventList * ev_pulse; - EventList * ev_spo2; - EventList * ev_plethy; + //Session * session; + //EventList * ev_pulse; + //EventList * ev_spo2; + //EventList * ev_plethy; Layer * foobar; gGraphView * m_shared; + + SerialOximeter *oximeter; + bool firstSPO2Update; + bool firstPulseUpdate; + bool secondPulseUpdate; + bool secondSPO2Update; + }; #endif // OXIMETRY_H diff --git a/oximetry.ui b/oximetry.ui index 08bad813..d2afb8dc 100644 --- a/oximetry.ui +++ b/oximetry.ui @@ -89,6 +89,137 @@ + + + + ... + + + + :/icons/save.png:/icons/save.png + + + + + + + Pulse + + + + + + + + 0 + 0 + + + + + + + + + 255 + 0 + 0 + + + + + + + + + 255 + 0 + 0 + + + + + + + + + 118 + 118 + 117 + + + + + + + + 3 + + + QLCDNumber::Flat + + + + + + + SpO2 + + + + + + + + 0 + 0 + + + + + + + + + 85 + 0 + 255 + + + + + + + + + 85 + 0 + 255 + + + + + + + + + 118 + 118 + 117 + + + + + + + + 3 + + + QLCDNumber::Flat + + +