#include #include #include #include #include #include #include #include #include #include #include "oximetry.h" #include "ui_oximetry.h" #include "common_gui.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())); import_mode=false; m_mode=SO_WAIT; } 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_callbacks=0; 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::addPulse(qint64 time, EventDataType pr) { //EventDataType min=0,max=0; if (pr>0) { if (lastpr==0) { if (pulse->count()==0) { pulse->setFirst(time); if (session->eventlist[OXI_Pulse].size()<=1) { session->setFirst(OXI_Pulse,time); if (session->first()==0) session->set_first(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); session->set_last(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); } } } lastpr=pr; emit(updatePulse(pr)); } void SerialOximeter::addSpO2(qint64 time, EventDataType o2) { //EventDataType min=0,max=0; if (o2>0) { if (lasto2==0) { if (spo2->count()==0) { spo2->setFirst(time); if (session->eventlist[OXI_SPO2].size()<=1) { session->setFirst(OXI_SPO2,time); if (session->first()==0) session->set_first(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); session->set_last(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); } } } 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(time); 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()<2) return; EventList nel(EVL_Waveform); EventDataType t,lastt=0; //el->data(0); qint64 ti;//=el->time(0); //nel.AddEvent(ti,lastt); bool f; qint64 lasttime=0; EventDataType min=999,max=0; for (quint32 i=0;icount();i++) { t=el->data(i); ti=el->time(i); f=false; if (t!=0) { if (t!=lastt) { if (!lasttime) { nel.setFirst(ti); } nel.AddEvent(ti,t); if (t < min) min=t; if (t > max) max=t; lasttime=ti; f=true; } } else { if (lastt!=0) { nel.AddEvent(ti,lastt); lasttime=ti; f=true; } } lastt=t; } if (!f) { if (t!=0) { nel.AddEvent(ti,t); lasttime=ti; } } el->setFirst(nel.first()); el->setLast(nel.last()); el->setMin(min); el->setMax(max); el->getData().clear(); el->getTime().clear(); el->setCount(nel.count()); el->getData()=nel.getData(); el->getTime()=nel.getTime(); } void SerialOximeter::compactAll() { if (!session) return; QHash >::iterator i; qint64 tminx=0,tmaxx=0,minx,maxx; EventDataType min,max; for (i=session->eventlist.begin();i!=session->eventlist.end();i++) { const QString & code=i.key(); min=999,max=0; minx=maxx=0; for (int j=0;j e->Min()) min=e->Min(); if (max < e->Max()) max=e->Max(); if (!minx || (minx > e->first())) minx=e->first(); if (!maxx || (maxx < e->last())) maxx=e->last(); } if ((code==OXI_SPO2) || (code==OXI_Pulse) || (code==OXI_Plethy)) { session->setMin(code,min); session->setMax(code,max); if (minx!=0) { session->setFirst(code,minx); if (!tminx || tminx > minx) tminx=minx; } if (maxx!=0){ session->setLast(code,maxx); if (!tmaxx || tmaxx < max) tmaxx=maxx; } } } if (tminx>0) session->really_set_first(tminx); if (tmaxx>0) session->really_set_last(tmaxx); } Session *SerialOximeter::createSession(QDateTime date) { if (session) { delete session; } int sid=date.toTime_t(); lasttime=qint64(sid)*1000L; lasto2=lastpr=0; 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); m_callbacks=0; emit(sessionCreated(session)); return session; } bool SerialOximeter::startLive() { import_mode=false; m_mode=SO_LIVE; lastpr=lasto2=0; if (!m_opened && !Open(QextSerialPort::EventDriven)) return false; createSession(); return true; } void SerialOximeter::stopLive() { if (timer->isActive()) timer->stop(); m_mode=SO_WAIT; 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) { } CMS50Serial::~CMS50Serial() { } void CMS50Serial::import_process() { if (!session) { qDebug() << "User pushing import too many times in a row?"; return; } qDebug() << "CMS50 import complete. Processing" << data.size() << "bytes"; unsigned short 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++); // low bits are supposedly the high bits of the heart rate pl=(data.at(i++) & 0x7f) | ((a & 3) << 7); 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++; } } else { if (lastpl!=0) { pulse->AddEvent(lasttime,pl); lastpltime=lasttime; 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++; } } else { if (lasto2!=0) { spo2->AddEvent(lasttime,o2); lasto2time=lasttime; 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++; }*/ qint64 rlasttime=qMax(lastpltime,lasto2time); session->set_last(rlasttime); 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)); disconnect(this,SIGNAL(importProcess()),this,SLOT(import_process())); } void CMS50Serial::ReadyRead() { QByteArray bytes; int a = m_port->bytesAvailable(); bytes.resize(a); m_port->read(bytes.data(), bytes.size()); m_callbacks++; if (m_mode==SO_WAIT) return; 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 short c; unsigned short pl,o2; if (!import_mode) { QString data="Read: "; #ifdef SERIAL_DEBUG 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 { static unsigned short hb=0; if (bytes[i]&0x80) { // 0x80 == sync bit EventDataType d=bytes[i+1] & 0x7f; hb=(bytes[i+2] & 0x40) << 1; addPlethy(lasttime,d); lasttime+=20; i+=3; } else { pl=(bytes[i] & 0x7f) | hb; o2=bytes[i+1]; addPulse(lasttime,pl); addSpO2(lasttime,o2); i+=2; } } } if (import_mode && waitf6 && (cntf6==0)) { failcnt++; if (failcnt>4) { // Device missed the 0xf5 code sequence somehow.. m_mode=SO_WAIT; emit(importAborted()); return; } } if (!import_mode) emit(dataChanged()); else if (done_import) { qDebug() << "End"; resetDevice(); m_mode=SO_WAIT; emit(importProcess()); } } void CMS50Serial::resetDevice() { m_port->flush(); static unsigned char b1[3]={0xf6,0xf6,0xf6}; if (m_port->write((char *)b1,3)==-1) { qDebug() << "Couldn't write closing bytes to CMS50"; } } void CMS50Serial::requestData() { static unsigned char b1[2]={0xf5,0xf5}; if (m_port->write((char *)b1,2)==-1) { qDebug() << "Couldn't write data request bytes to CMS50"; } } bool CMS50Serial::startImport() { m_mode=SO_WAIT; if (!m_opened && !Open(QextSerialPort::EventDriven)) return false; m_callbacks=0; import_fails=0; QTimer::singleShot(250,this,SLOT(startImportTimeout())); //make sure there is a data stream first.. createSession(); return true; } void CMS50Serial::startImportTimeout() { if (m_callbacks>0) { connect(this,SIGNAL(importProcess()),this,SLOT(import_process())); import_mode=true; waitf6=true; done_import=false; cntf6=0; failcnt=0; m_mode=SO_IMPORT; requestData(); } else { import_fails++; if (import_fails<5) { resetDevice(); QTimer::singleShot(250,this,SLOT(startImportTimeout())); } else { qDebug() << "No oximeter signal!!!!!!!!!!!!!!!!"; emit(importAborted()); } } } 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,schema::channel[OXI_Plethy].label(),schema::channel[OXI_Plethy].units(),120); CONTROL=new gGraph(GraphView,tr("Control"),"",75); PULSE=new gGraph(GraphView,schema::channel[OXI_Pulse].label(),schema::channel[OXI_Pulse].units(),120); SPO2=new gGraph(GraphView,schema::channel[OXI_SPO2].label(),schema::channel[OXI_SPO2].units(),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(tr("No Oximetry Data")); GraphView->updateGL(); on_RefreshPortsButton_clicked(); ui->RunButton->setChecked(false); ui->saveButton->setEnabled(false); GraphView->LoadSettings("Oximetry"); QLocale locale=QLocale::system(); QString shortformat=locale.dateFormat(QLocale::ShortFormat); if (!shortformat.toLower().contains("yyyy")) { shortformat.replace("yy","yyyy"); } ui->dateEdit->setDisplayFormat(shortformat+" HH:mm:ss"); //Qt::DayOfWeek dow=firstDayOfWeekFromLocale(); //ui->dateEdit->calendarWidget()->setFirstDayOfWeek(dow); // Stop both calendar drop downs highlighting weekends in red //QTextCharFormat format = ui->dateEdit->calendarWidget()->weekdayTextFormat(Qt::Saturday); //format.setForeground(QBrush(Qt::black, Qt::SolidPattern)); //ui->dateEdit->calendarWidget()->setWeekdayTextFormat(Qt::Saturday, format); //ui->dateEdit->calendarWidget()->setWeekdayTextFormat(Qt::Sunday, format); dont_update_date=false; } Oximetry::~Oximetry() { delete oximeter; 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) { Q_UNUSED(session); mainwin->Notify(tr("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(tr("&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); if (oximeter->getSession()) saved_starttime=oximeter->getSession()->first(); //CONTROL->setVisible(true); } else { if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { int res=QMessageBox::question(this,tr("Save Session?"),tr("Creating a new oximetry session will destroy the old one.\nWould you like to save it first?"),tr("Save"),tr("Destroy It"),tr("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(tr("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(tr("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(tr("&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)); // translation fix? } 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(tr("Oximeter Error\n\nThe device has not responded.. Make sure it's switched on2")); on_ImportButton_clicked(); return; } } if (oximeter->callbacks()==0) { mainwin->Notify(tr("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(tr("Ready")); ui->ImportButton->setEnabled(true); ui->RunButton->setChecked(false); ui->saveButton->setEnabled(false); GraphView->setEmptyText(tr("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(tr("Oximeter Error\n\nThe device did not respond.. Make sure it's switched on.")); 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))); //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(tr("&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(tr("Ready")); ui->ImportButton->setDisabled(false); ui->saveButton->setEnabled(true); if (qprogress) { qprogress->setValue(100); qprogress->hide(); } } void Oximetry::import_aborted() { oximeter->disconnect(oximeter,SIGNAL(importProcess()),0,0); day->getSessions().clear(); //QMessageBox::warning(mainwin,tr("Oximeter Error"),tr("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(tr("Please make sure your oximeter is switched on, and in the right mode to transmit data."),tr("Oximeter Error!"),5000); //qDebug() << "Oximetry import failed"; import_finished(); } void Oximetry::import_complete(Session * session) { qDebug() << "Oximetry import complete"; import_finished(); if (!session) { qDebug() << "Shouldn't happen"; return; } //calcSPO2Drop(session); //calcPulseChange(session); //PLETHY->setVisible(false); CONTROL->setVisible(false); saved_starttime=session->first(); dont_update_date=true; ui->dateEdit->setDateTime(QDateTime::fromTime_t(saved_starttime/1000L)); dont_update_date=false; updateGraphs(); } 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(); PROFILE.RemoveSession(session); Machine *m=oximeter->getMachine(); if (m->SessionExists(session->session())) { m->sessionlist.erase(m->sessionlist.find(session->session())); } // Forgetting to reset the session ID sucks, as it will delete sessions you don't want to delete.. session->SetSessionID(qint64(session->first())/1000L); m->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(); } } bool Oximetry::openSPOFile(QString filename) { QFile f(filename); if (!f.open(QFile::ReadOnly)) return false; QByteArray data; data=f.readAll(); long size=data.size(); int pos=((unsigned char)data.at(1) << 8) | (unsigned char)data.at(0); char dchr[20]; int j=0; for (int i=0;i<18*2;i+=2) { dchr[j++]=data.at(8+i); } dchr[j]=0; QString dstr(dchr); QDateTime date=QDateTime::fromString(dstr,"MM/dd/yy HH:mm:ss"); if (date.date().year()<2000) date=date.addYears(100); //ui->dateEdit->setDateTime(date); day->getSessions().clear(); oximeter->createSession(date); Session *session=oximeter->getSession(); day->AddSession(session); session->set_first(0); firstPulseUpdate=true; firstSPO2Update=true; secondPulseUpdate=true; secondSPO2Update=true; unsigned char o2,pr; quint16 pl; qint64 tt=qint64(date.toTime_t())*1000L; for (int i=pos;isetLastTime(tt); oximeter->addPulse(tt,pr); oximeter->addSpO2(tt,o2); pl=(unsigned char)(data.at(i+1)); //oximeter->addPlethy(tt,pl); //pl=(unsigned char)(data.at(i+1)); //oximeter->addPlethy(tt,pl); //pl=(unsigned char)(data.at(i+2)); //oximeter->addPlethy(tt,pl); i+=5; //data_changed(); tt+=1000; } qint64 t1=session->first(OXI_Pulse); qint64 t2=session->first(OXI_SPO2); qint64 t3=qMin(t1,t2); session->set_first(t3); day->setFirst(t3); int zi=t3/1000L; session->SetSessionID(zi); date.fromTime_t(zi); dont_update_date=true; ui->dateEdit->setDateTime(date); dont_update_date=false; t1=session->last(OXI_Pulse); t2=session->last(OXI_SPO2); t3=qMax(t1,t2); session->set_last(t3); day->setLast(t3); CONTROL->setVisible(false); updateGraphs(); f.close(); ui->saveButton->setEnabled(true); return true; } bool Oximetry::openSPORFile(QString filename) { //GraphView->setEmptyText("Please Wait"); //GraphView->updateGL(); QFile f(filename); if (!f.open(QFile::ReadOnly)) return false; QByteArray data; data=f.readAll(); long size=data.size(); int pos=((unsigned char)data.at(1) << 8) | (unsigned char)data.at(0); char dchr[20]; int j=0; for (int i=0;i<18*2;i+=2) { dchr[j++]=data.at(8+i); } dchr[j]=0; QString dstr(dchr); QDateTime date=QDateTime::fromString(dstr,"MM/dd/yy HH:mm:ss"); if (date.date().year()<2000) date=date.addYears(100); day->getSessions().clear(); oximeter->createSession(date); Session *session=oximeter->getSession(); day->AddSession(session); session->set_first(0); firstPulseUpdate=true; firstSPO2Update=true; secondPulseUpdate=true; secondSPO2Update=true; unsigned char o2,pr; quint16 pl; qint64 tt=qint64(date.toTime_t())*1000L; for (int i=pos;iaddPulse(tt,pr); oximeter->addSpO2(tt,o2); pl=(unsigned char)(data.at(i+1)); i+=2; tt+=1000; } qint64 t1=session->first(OXI_Pulse); qint64 t2=session->first(OXI_SPO2); qint64 t3=qMin(t1,t2); session->set_first(t3); day->setFirst(t3); int zi=t3/1000L; session->SetSessionID(zi); date.fromTime_t(zi); dont_update_date=true; ui->dateEdit->setDateTime(date); dont_update_date=false; t1=session->last(OXI_Pulse); t2=session->last(OXI_SPO2); t3=qMax(t1,t2); session->set_last(t3); day->setLast(t3); //PLETHY->setVisible(false); CONTROL->setVisible(false); updateGraphs(); f.close(); ui->saveButton->setEnabled(true); return true; } void Oximetry::on_openButton_clicked() { if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { int res=QMessageBox::question(this,"Save Session?","Opening this oximetry file will destroy the current session.\nWould you like to keep it?","Save","Destroy It","Cancel",0,2); if (res==0) { on_saveButton_clicked(); return; } else if (res==2) { return; } } // else it's already saved. QString dir=""; QFileDialog fd(this,"Select an oximetry file",dir,"Oximetry Files (*.spo *.spoR)"); fd.setAcceptMode(QFileDialog::AcceptOpen); fd.setFileMode(QFileDialog::ExistingFile); if (fd.exec()!=QFileDialog::Accepted) return; QStringList filenames=fd.selectedFiles(); if (filenames.size()>1) { qDebug() << "Can only open one oximetry file at a time"; } QString filename=filenames[0]; bool r=false; if (filename.toLower().endsWith(".spo")) r=openSPOFile(filename); else if (filename.toLower().endsWith(".spor")) r=openSPORFile(filename); if (!r) { mainwin->Notify("Couldn't open oximetry file \""+filename+"\"."); } qDebug() << "opening" << filename; } void Oximetry::on_dateEdit_dateTimeChanged(const QDateTime &date) { Session *session=oximeter->getSession(); if (!session) return; if (dont_update_date) return; qint64 first=session->first(); //qint64 last=session->last(); qint64 tt=qint64(date.toTime_t())*1000L; qint64 offset=tt-first; if (offset!=0) { session->SetChanged(true); ui->saveButton->setEnabled(true); } session->offsetSession(offset); updateGraphs(); } void Oximetry::openSession(Session * session) { if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { int res=QMessageBox::question(this,"Save Session?","Opening this oximetry session will destroy the unsavedsession in the oximetry tab.\nWould you like to store it first?","Save","Destroy It","Cancel",0,2); if (res==0) { on_saveButton_clicked(); return; } else if (res==2) { return; } } // else it's already saved. day->getSessions().clear(); day->AddSession(session); oximeter->setSession(session); saved_starttime=session->first(); oximeter->compactAll(); QDateTime date=QDateTime::fromTime_t(saved_starttime/1000L); dont_update_date=true; ui->dateEdit->setDateTime(date); dont_update_date=false; updateGraphs(); } void Oximetry::updateGraphs() { Session * session=oximeter->getSession(); if (!session) return; qint64 first=session->first(); qint64 last=session->last(); 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); day->setFirst(first); day->setLast(last); 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->setMinX(first); pulse->setMinX(first); spo2->setMinX(first); PLETHY->SetMinX(first); CONTROL->SetMinX(first); PULSE->SetMinX(first); SPO2->SetMinX(first); plethy->setMaxX(last); pulse->setMaxX(last); spo2->setMaxX(last); PLETHY->SetMaxX(last); CONTROL->SetMaxX(last); PULSE->SetMaxX(last); SPO2->SetMaxX(last); PULSE->setDay(day); SPO2->setDay(day); 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("%02i:%02i:%02i",h,m,s)); } GraphView->updateScale(); GraphView->updateGL(); } void Oximetry::on_resetTimeButton_clicked() //revert to original session time { if (oximeter->getSession()) { //qint64 tt=ui->dateEdit->dateTime().toTime_t()*1000; //qint64 offset=saved_starttime-tt; //oximeter->getSession()->offsetSession(offset); dont_update_date=false; //ui->dateEdit->setDateTime(QDateTime::fromTime_t(saved_starttime/1000L)); ui->dateEdit->setDateTime(QDateTime::fromTime_t(saved_starttime/1000L)); dont_update_date=false; } }