#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 "Graphs/gXAxis.h" #include "Graphs/gBarChart.h" #include "Graphs/gLineChart.h" #include "Graphs/gYAxis.h" Oximetry::Oximetry(QWidget *parent,gGraphView * shared) : QWidget(parent), ui(new Ui::Oximetry) { m_shared=shared; ui->setupUi(this); port=NULL; portname=""; QString prof=pref["Profile"].toString(); profile=Profiles::Get(prof); if (!profile) { qWarning("Couldn't get profile.. Have to abort!"); exit(-1); } mach=profile->GetMachine(MT_OXIMETER); if (!mach) { CMS50Loader *l=dynamic_cast(GetLoader("CMS50")); if (l) { mach=l->CreateMachine(profile); } qDebug() << "Create Oximeter device"; } // Create dummy day & session for holding eventlists. day=new Day(mach); session=new Session(mach,0); day->AddSession(session); 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_Plethysomogram); 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()); } // Create the Event Lists to store / import data ev_plethy=new EventList(OXI_Plethysomogram,EVL_Waveform,1,0,0,0,1000.0/50.0); session->eventlist[OXI_Plethysomogram].push_back(ev_plethy); ev_pulse=new EventList(OXI_Pulse,EVL_Event,1); session->eventlist[OXI_Pulse].push_back(ev_pulse); ev_spo2=new EventList(OXI_SPO2,EVL_Event,1); session->eventlist[OXI_SPO2].push_back(ev_spo2); plethy=new gLineChart(OXI_Plethysomogram,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(pulse); SPO2->AddLayer(spo2); GraphView->setEmptyText(""); GraphView->updateGL(); on_RefreshPortsButton_clicked(); } Oximetry::~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=""; } } 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=qint64(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_Plethysomogram].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->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->avg(OXI_Plethysomogram); sess->wavg(OXI_Plethysomogram); sess->p90(OXI_Plethysomogram); sess->setMin(OXI_Plethysomogram,ev_plethy->min()); sess->setMax(OXI_Plethysomogram,ev_plethy->max()); sess->setFirst(OXI_Plethysomogram,ev_plethy->first()); sess->setLast(OXI_Plethysomogram,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_Plethysomogram)); sess->updateLast(sess->last(OXI_Plethysomogram)); sess->SetChanged(true); mach->AddSession(sess,profile); mach->Save(); ev_plethy=new EventList(OXI_Plethysomogram,EVL_Waveform,1,0,0,0,1000.0/50.0); session->eventlist[OXI_Plethysomogram].push_back(ev_plethy); ev_pulse=new EventList(OXI_Pulse,EVL_Event,1); session->eventlist[OXI_Pulse].push_back(ev_pulse); ev_spo2=new EventList(OXI_SPO2,EVL_Event,1); session->eventlist[OXI_SPO2].push_back(ev_spo2); session->setCount(OXI_Plethysomogram,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_Plethysomogram,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; } void Oximetry::onReadyRead() { QByteArray bytes; int a = port->bytesAvailable(); bytes.resize(a); port->read(bytes.data(), bytes.size()); int i=0; while (icount()<=2) || (ev_pulse->count()<=2) || (ev_spo2->count()<=2)) { GraphView->updateScale(); } GraphView->updateGL(); } void Oximetry::onDsrChanged(bool status) // Doesn't work for CMS50's { if (status) qDebug() << "device was turned on"; else qDebug() << "device was turned off"; } 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;iImportButton->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]; unsigned char * buffer=NULL; 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.. } 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=new EventList(OXI_PulseChange,EVL_Event); session->eventlist[OXI_PulseChange].push_back(oxf1); } 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=new EventList(OXI_SPO2Drop,EVL_Event); session->eventlist[OXI_SPO2Drop].push_back(oxf2); } 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->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->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->SetChanged(true); mach->AddSession(session,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=new EventList(OXI_Plethysomogram,EVL_Waveform,1,0,0,0,1000.0/50.0); session->eventlist[OXI_Plethysomogram].push_back(ev_plethy); ev_pulse=new EventList(OXI_Pulse,EVL_Event,1); session->eventlist[OXI_Pulse].push_back(ev_pulse); ev_spo2=new EventList(OXI_SPO2,EVL_Event,1); session->eventlist[OXI_SPO2].push_back(ev_spo2); } delete port; port=NULL; msgbox.hide(); qprogress->hide(); ui->SerialPortsCombo->setEnabled(true); qstatus->setText("Ready"); ui->ImportButton->setDisabled(false); }