#include #include #include #include #include #include #include "oximetry.h" #include "ui_oximetry.h" #include "qextserialport/qextserialenumerator.h" #include "SleepLib/loader_plugins/cms50_loader.h" #include "SleepLib/event.h" #include "SleepLib/calcs.h" #include "Graphs/gXAxis.h" #include "Graphs/gSummaryChart.h" #include "Graphs/gLineChart.h" #include "Graphs/gYAxis.h" #include "Graphs/gLineOverlay.h" extern QLabel * qstatus2; #include "mainwindow.h" extern MainWindow * mainwin; 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"; } timer=new QTimer(this); connect(timer,SIGNAL(timeout()),this,SLOT(Timeout())); } SerialOximeter::~SerialOximeter() { if (m_opened) { if (m_port) m_port->close(); } disconnect(timer,SIGNAL(timeout()),this,SLOT(Timeout())); delete timer; } void SerialOximeter::Timeout() { qDebug() << "Timeout!"; if (!import_mode) emit(liveStopped(session)); } 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_portmode=mode; m_port=new QextSerialPort(m_portname,m_portmode); 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(ReadyRead())); //connect(port, SIGNAL(dsrChanged(bool)), this, SLOT(DsrChanged(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_portmode==QextSerialPort::EventDriven) disconnect(m_port, SIGNAL(readyRead()), this, SLOT(ReadyRead())); m_mode=SO_OFF; 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::ReadyRead() { int i=5; qDebug() << "Foo" << i; } void SerialOximeter::addPulse(qint64 time, EventDataType pr) { static EventDataType lastpr=0; //EventDataType min=0,max=0; if (pr>0) { if (lastpr==0) { if (pulse->count()==0) pulse->setFirst(time); else { qDebug() << "Shouldn't happen in addPulse()"; } } pulse->AddEvent(time,pr); session->setCount(OXI_Pulse,session->count(OXI_Pulse)+1); session->setLast(OXI_Pulse,time); } else { if (lastpr!=0) { if (pulse->count() > 0) { pulse->AddEvent(time,lastpr); this->compactToEvent(pulse); session->setLast(OXI_Pulse,time); pulse=session->AddEventList(OXI_Pulse,EVL_Event); } } } session->set_last(time); lastpr=pr; emit(updatePulse(pr)); } void SerialOximeter::addSpO2(qint64 time, EventDataType o2) { static EventDataType lasto2=0; //EventDataType min=0,max=0; if (o2>0) { if (lasto2==0) { if (spo2->count()==0) spo2->setFirst(time); else { qDebug() << "Shouldn't happen in addSpO2()"; } } spo2->AddEvent(time,o2); session->setCount(OXI_SPO2,session->count(OXI_SPO2)+1); session->setLast(OXI_SPO2,time); } else { if (lasto2!=0) { if (spo2->count() > 0) { spo2->AddEvent(time,lasto2); this->compactToEvent(spo2); session->setLast(OXI_SPO2,time); spo2=session->AddEventList(OXI_SPO2,EVL_Event); } } } session->set_last(time); lasto2=o2; 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::compactToWaveform(EventList *el) { double rate=double(el->duration())/double(el->count()); el->setType(EVL_Waveform); el->setRate(rate); el->getTime().clear(); } void SerialOximeter::compactToEvent(EventList *el) { if (el->count()==0) return; EventList nel(EVL_Waveform); EventDataType t,lastt=el->data(0); qint64 ti=el->time(0); nel.AddEvent(ti,lastt); bool f; for (quint32 i=1;icount();i++) { t=el->data(i); f=false; if (t!=lastt) { ti=el->time(i); nel.AddEvent(ti,t); f=true; } lastt=t; } if (!f) { nel.AddEvent(el->last(),t); } el->getData().clear(); el->getTime().clear(); el->setCount(nel.count()); el->getData()=nel.getData(); el->getTime()=nel.getTime(); /*for (int i=0;igetData().push_back(nel.data(i)); el->getTime().push_back(nel.time(i)); } */ /*double rate=double(el->duration())/double(el->count()); el->setType(EVL_Waveform); el->setRate(rate); el->getTime().clear();*/ } void SerialOximeter::compactAll() { if (!session) return; QHash >::iterator i; for (i=session->eventlist.begin();i!=session->eventlist.end();i++) { for (int j=0;jSetChanged(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); m_callbacks=0; emit(sessionCreated(session)); return session; } bool SerialOximeter::startLive() { m_mode=SO_LIVE; import_mode=false; if (Open(QextSerialPort::EventDriven)) { createSession(); return true; } else { return false; } } void SerialOximeter::stopLive() { if (timer->isActive()) timer->stop(); Close(); if (session) { compactAll(); calcSPO2Drop(session); calcPulseChange(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::import_process() { disconnect(this,SIGNAL(importProcess()),this,SLOT(import_process())); if (!session) { qDebug() << "User pushing import too many times in a row?"; return; } 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; //spo2->setFirst(lasttime); EventDataType plmin=999,plmax=0; EventDataType o2min=100,o2max=0; int plcnt=0,o2cnt=0; qint64 lastpltime=0,lasto2time=0; bool first=true; while (i<(size-3)) { a=data.at(i++); Q_UNUSED(a); pl=data.at(i++) ^ 0x80; o2=data.at(i++); if (pl!=0) { if (lastpl!=pl) { if (lastpl==0 || !pulse) { if (first) { session->set_first(lasttime); first=false; } if (plcnt==0) session->setFirst(OXI_Pulse,lasttime); if (pulse && pulse->count()==0) { } else { pulse=new EventList(EVL_Event); session->eventlist[OXI_Pulse].push_back(pulse); } } lastpltime=lasttime; pulse->AddEvent(lasttime,pl); if (pl < plmin) plmin=pl; if (pl > plmax) plmax=pl; plcnt++; } } if (o2!=0) { if (lasto2!=o2) { if (lasto2==0 || !spo2) { if (first) { session->set_first(lasttime); first=false; } if (o2cnt==0) session->setFirst(OXI_SPO2,lasttime); if (spo2 && spo2->count()==0) { } else { spo2=new EventList(EVL_Event); session->eventlist[OXI_SPO2].push_back(spo2); } } lasto2time=lasttime; spo2->AddEvent(lasttime,o2); if (o2 < o2min) o2min=o2; if (o2 > o2max) o2max=o2; o2cnt++; } } lasttime+=1000; //emit(updateProgress(float(i)/float(size))); lastpl=pl; lasto2=o2; } if (pulse && (lastpltime!=lasttime) && (pl!=0)) { // lastpl==pl pulse->AddEvent(lasttime,pl); lastpltime=lastpltime; plcnt++; } if (spo2 && (lasto2time!=lasttime) && (o2!=0)) { spo2->AddEvent(lasttime,o2); lasto2time=lasttime; o2cnt++; } session->set_last(lasttime); session->setLast(OXI_Pulse,lastpltime); session->setLast(OXI_SPO2,lasto2time); 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)); } void CMS50Serial::ReadyRead() { QByteArray bytes; int a = m_port->bytesAvailable(); bytes.resize(a); m_port->read(bytes.data(), bytes.size()); m_callbacks++; 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.) static int lastbytesize=0; int size=bytes.size(); // Process all incoming serial data packets unsigned char c; unsigned char pl,o2; if (!import_mode) { QString data="Read: "; for (int i=0;iisActive()) { timer->stop(); } timer->setSingleShot(true); timer->setInterval(10000); timer->start(); } return; //qDebug() << "Oximeter switched of.. wait for timeout?"; } else { if (timer->isActive()) { timer->stop(); } } } lastbytesize=size; 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) done_import=true; waitf6=false; break; } } } else { qDebug() << "Recieving Block" << size << "(" << received_bytes << "of" << datasize <<")"; for (int z=i;z=datasize) || (((received_bytes/datasize)>0.7) && (size<250))) { done_import=true; } break; //read data blocks.. } } else { if (bytes[i]&0x80) { // 0x80 == sync bit EventDataType d=bytes[i+1] & 0x7f; addPlethy(lasttime,d); lasttime+=20; i+=3; } else { pl=bytes[i]; o2=bytes[i+1]; addPulse(lasttime,pl); addSpO2(lasttime,o2); i+=2; } } } //static unsigned char b1[6]={0xf5,0xf5,0xf5,0xf5,0xf5,0xf5}; if (import_mode && waitf6 && (cntf6==0)) { failcnt++; /*if (failcnt>20) { qDebug() << "Retrying request"; if (m_port->write((char *)b1,6)==-1) { qDebug() << "Couldn't write data request bytes to CMS50"; } } */ if (failcnt>4) { emit(importAborted()); return; } } if (!import_mode) emit(dataChanged()); else if (done_import){ 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()); } } bool CMS50Serial::startImport() { m_mode=SO_IMPORT; import_mode=true; waitf6=true; done_import=false; cntf6=0; failcnt=0; //QMessageBox::information(0,"Get Ready","Please make sure your oximeter is switched on and in the correct mode for data transfer.",QMessageBox::Ok); if (!Open(QextSerialPort::EventDriven)) return false; connect(this,SIGNAL(importProcess()),this,SLOT(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) { m_shared=shared; ui->setupUi(this); port=NULL; portname=""; oximeter=new CMS50Serial(this); day=new Day(oximeter->getMachine()); layout=new QHBoxLayout(ui->graphArea); layout->setSpacing(0); layout->setMargin(0); layout->setContentsMargins(0,0,0,0); ui->graphArea->setLayout(layout); ui->graphArea->setAutoFillBackground(false); GraphView=new gGraphView(ui->graphArea,m_shared); GraphView->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding); scrollbar=new MyScrollBar(ui->graphArea); scrollbar->setOrientation(Qt::Vertical); scrollbar->setSizePolicy(QSizePolicy::Maximum,QSizePolicy::Expanding); scrollbar->setMaximumWidth(20); GraphView->setScrollBar(scrollbar); layout->addWidget(GraphView,1); layout->addWidget(scrollbar,0); layout->layout(); PLETHY=new gGraph(GraphView,tr("Plethy"),120); CONTROL=new gGraph(GraphView,tr("Control"),75); PULSE=new gGraph(GraphView,tr("Pulse Rate"),120); SPO2=new gGraph(GraphView,tr("SPO2"),120); foobar=new gShadowArea(); CONTROL->AddLayer(foobar); Layer *cl=new gLineChart(OXI_Plethy); CONTROL->AddLayer(cl); cl->SetDay(day); CONTROL->setBlockZoom(true); for (int i=0;isize();i++) { gGraph *g=(*GraphView)[i]; g->AddLayer(new gXAxis(),LayerBottom,0,gXAxis::Margin); g->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); g->AddLayer(new gXGrid()); } plethy=new gLineChart(OXI_Plethy,Qt::black,false,true); plethy->SetDay(day); CONTROL->AddLayer(plethy); //new gLineChart(OXI_Plethysomogram)); pulse=new gLineChart(OXI_Pulse,Qt::red,true); //pulse->SetDay(day); spo2=new gLineChart(OXI_SPO2,Qt::blue,true); //spo2->SetDay(day); PLETHY->AddLayer(plethy); PULSE->AddLayer(lo1=new gLineOverlayBar(OXI_PulseChange,QColor("light gray"),"PD",FT_Span)); SPO2->AddLayer(lo2=new gLineOverlayBar(OXI_SPO2Drop,QColor("light blue"),"O2",FT_Span)); PULSE->AddLayer(pulse); SPO2->AddLayer(spo2); PULSE->setDay(day); SPO2->setDay(day); lo1->SetDay(day); lo2->SetDay(day); //go->SetDay(day); GraphView->setEmptyText("No Oximetry Data"); GraphView->updateGL(); on_RefreshPortsButton_clicked(); ui->RunButton->setChecked(false); ui->saveButton->setEnabled(false); GraphView->LoadSettings("Oximetry"); } Oximetry::~Oximetry() { GraphView->SaveSettings("Oximetry"); delete ui; } void Oximetry::on_RefreshPortsButton_clicked() { QList ports = QextSerialEnumerator::getPorts(); ui->SerialPortsCombo->clear(); int z=0; QString firstport; bool current_found=false; // Windows build mixes these up #ifdef Q_WS_WIN32 #define qesPORTNAME portName #else #define qesPORTNAME physName #endif for (int i = 0; i < ports.size(); i++) { if (!ports.at(i).friendName.isEmpty()) { if (ports.at(i).friendName.toUpper().contains("USB")) { if (firstport.isEmpty()) firstport=ports.at(i). qesPORTNAME; if (!portname.isEmpty() && ports.at(i).qesPORTNAME==portname) current_found=true; ui->SerialPortsCombo->addItem(ports.at(i).qesPORTNAME); z++; } } else { // Mac stuff. if (ports.at(i).portName.toUpper().contains("USB") || ports.at(i).portName.toUpper().contains("SPO2")) { if (firstport.isEmpty()) firstport=ports.at(i).portName; if (!portname.isEmpty() && ports.at(i).portName==portname) current_found=true; ui->SerialPortsCombo->addItem(ports.at(i).portName); z++; } } //qDebug() << "Serial Port:" << ports.at(i).qesPORTNAME << ports.at(i).friendName; //qDebug() << "port name:" << ports.at(i).portName; //qDebug() << "phys name:" << ports.at(i).physName; //qDebug() << "friendly name:" << ports.at(i).friendName; //qDebug() << "enumerator name:" << ports.at(i).enumName; } if (z>0) { ui->RunButton->setEnabled(true); ui->ImportButton->setEnabled(true); if (!current_found) portname=firstport; } else { ui->RunButton->setEnabled(false); ui->ImportButton->setEnabled(false); portname=""; } oximeter->setPortName(portname); } void Oximetry::RedrawGraphs() { GraphView->updateGL(); } void Oximetry::on_SerialPortsCombo_activated(const QString &arg1) { portname=arg1; oximeter->setPortName(arg1); } void Oximetry::live_stopped(Session * session) { mainwin->Notify("Oximetry live recording has been terminated due to timeout"); //qDebug () << "Live Stopped"; on_RunButton_toggled(false); } void Oximetry::on_RunButton_toggled(bool checked) { if (!checked) { oximeter->stopLive(); ui->RunButton->setText("&Start"); ui->SerialPortsCombo->setEnabled(true); disconnect(oximeter,SIGNAL(dataChanged()),this,SLOT(data_changed())); disconnect(oximeter,SIGNAL(updatePulse(float)),this,SLOT(pulse_changed(float))); disconnect(oximeter,SIGNAL(updateSpO2(float)),this,SLOT(spo2_changed(float))); disconnect(oximeter,SIGNAL(liveStopped(Session*)),this,SLOT(live_stopped(Session *))); ui->saveButton->setEnabled(true); ui->ImportButton->setEnabled(true); lo2->SetDay(day); lo1->SetDay(day); //CONTROL->setVisible(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. GraphView->setEmptyText("Please Wait"); GraphView->updateGL(); PLETHY->setRecMinY(0); PLETHY->setRecMaxY(128); PULSE->setRecMinY(60); PULSE->setRecMaxY(100); SPO2->setRecMinY(90); SPO2->setRecMaxY(100); day->getSessions().clear(); //QTimer::singleShot(10000,this,SLOT(oximeter_running_check())); if (!oximeter->startLive()) { mainwin->Notify("Oximetry Error!\n\nSomething is wrong with the device connection."); return; } ui->saveButton->setEnabled(false); day->AddSession(oximeter->getSession()); 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); //graphView()->updateScale(); /*PLETHY->setForceMinY(0); PLETHY->setForceMaxY(128); PULSE->setForceMinY(30); PULSE->setForceMaxY(180); SPO2->setForceMinY(50); SPO2->setForceMaxY(100); */ connect(oximeter,SIGNAL(dataChanged()),this,SLOT(data_changed())); connect(oximeter,SIGNAL(updatePulse(float)),this,SLOT(pulse_changed(float))); connect(oximeter,SIGNAL(updateSpO2(float)),this,SLOT(spo2_changed(float))); connect(oximeter,SIGNAL(liveStopped(Session*)),this,SLOT(live_stopped(Session *))); CONTROL->setVisible(false); // connect. ui->RunButton->setText("&Stop"); ui->SerialPortsCombo->setEnabled(false); ui->ImportButton->setEnabled(false); } } void Oximetry::data_changed() { 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((oximeter->Plethy())->min()); plethy->setMaxY((oximeter->Plethy())->max()); pulse->setMinY((oximeter->Pulse())->min()); pulse->setMaxY((oximeter->Pulse())->max()); spo2->setMinY((oximeter->Spo2())->min()); spo2->setMaxY((oximeter->Spo2())->max()); PLETHY->SetMinY((oximeter->Plethy())->min()); PLETHY->SetMaxY((oximeter->Plethy())->max()); PULSE->SetMinY((oximeter->Pulse())->min()); PULSE->SetMaxY((oximeter->Pulse())->max()); SPO2->SetMinY((oximeter->Spo2())->min()); SPO2->SetMaxY((oximeter->Spo2())->max()); /*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->updateScale(); GraphView->updateGL(); } extern QProgressBar *qprogress; extern QLabel *qstatus; void DumpBytes(int blocks, unsigned char * b,int len) { QString a=QString::number(blocks,16)+": Bytes "+QString::number(len,16)+": "; for (int i=0;iisOpen()) { if (oximeter->callbacks()==0) { qDebug() << "Not sure how oximeter_running_check gets called with a closed oximeter.. Restarting import process"; //mainwin->Notify("Oximeter Error\n\nThe device has not responded.. Make sure it's switched on2"); on_ImportButton_clicked(); return; } } if (oximeter->callbacks()==0) { mainwin->Notify("Oximeter Error\n\nThe device has not responded.. Make sure it's switched on."); if (oximeter->mode()==SO_IMPORT) oximeter->stopImport(); if (oximeter->mode()==SO_LIVE) oximeter->stopLive(); oximeter->destroySession(); day->getSessions().clear(); ui->SerialPortsCombo->setEnabled(true); qstatus->setText("Ready"); ui->ImportButton->setEnabled(true); ui->RunButton->setChecked(false); ui->saveButton->setEnabled(false); GraphView->setEmptyText("Check Oximeter is Ready"); GraphView->updateGL(); } } // Move this code to CMS50 Importer?? void Oximetry::on_ImportButton_clicked() { connect(oximeter,SIGNAL(importComplete(Session*)),this,SLOT(import_complete(Session*))); connect(oximeter,SIGNAL(importAborted()),this,SLOT(import_aborted())); connect(oximeter,SIGNAL(updateProgress(float)),this,SLOT(update_progress(float))); if (!oximeter->startImport()) { mainwin->Notify("Oximeter Error\n\nThe device did not respond.. Make sure it's switched on."); //qDebug() << "Error starting oximetry serial import process"; return; } QTimer::singleShot(1000,this,SLOT(oximeter_running_check())); 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); } void Oximetry::import_finished() { disconnect(oximeter,SIGNAL(importComplete(Session*)),this,SLOT(import_complete(Session*))); disconnect(oximeter,SIGNAL(importAborted()),this,SLOT(import_aborted())); disconnect(oximeter,SIGNAL(updateProgress(float)),this,SLOT(update_progress(float))); 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::import_aborted() { //QMessageBox::warning(mainwin,"Oximeter Error","Please make sure your oximeter is switched on, and able to transmit data.\n(You may need to enter the oximeters Settings screen for it to be able to transmit.)",QMessageBox::Ok); mainwin->Notify("Oximeter Error!\n\nPlease make sure your oximeter is switched on, and in the right mode to transmit data."); //qDebug() << "Oximetry import failed"; import_finished(); } void Oximetry::import_complete(Session * session) { qDebug() << "Oximetry import complete"; import_finished(); //calcSPO2Drop(session); //calcPulseChange(session); ui->pulseLCD->display(session->min(OXI_Pulse)); ui->spo2LCD->display(session->min(OXI_SPO2)); pulse->setMinY(session->min(OXI_Pulse)); pulse->setMaxY(session->max(OXI_Pulse)); spo2->setMinY(session->min(OXI_SPO2)); spo2->setMaxY(session->max(OXI_SPO2)); PULSE->setRecMinY(60); PULSE->setRecMaxY(100); SPO2->setRecMinY(90); SPO2->setRecMaxY(100); //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); 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::pulse_changed(float p) { ui->pulseLCD->display(p); return; if (firstPulseUpdate) { if (secondPulseUpdate) { secondPulseUpdate=false; } else { firstPulseUpdate=false; GraphView->updateScale(); } } } // Only really need to do this once.. void Oximetry::spo2_changed(float o2) { ui->spo2LCD->display(o2); return; if (firstSPO2Update) { if (secondSPO2Update) { secondSPO2Update=false; } else { firstSPO2Update=false; GraphView->updateScale(); } } } void Oximetry::on_saveButton_clicked() { if (QMessageBox::question(this,"Keep This Recording?","Would you like to save this oximetery session?",QMessageBox::Yes|QMessageBox::No)==QMessageBox::Yes) { Session *session=oximeter->getSession(); // Process??? //session->UpdateSummaries(); oximeter->getMachine()->AddSession(session,p_profile); oximeter->getMachine()->Save(); day->getSessions().clear(); mainwin->getDaily()->ReloadGraphs(); mainwin->getOverview()->ReloadGraphs(); GraphView->setEmptyText("No Oximetry Data"); GraphView->updateGL(); } } void Oximetry::update_progress(float f) { if (qprogress) { qprogress->setValue(f*100.0); QApplication::processEvents(); } }