OSCAR-code/oximetry.cpp

750 lines
23 KiB
C++
Raw Normal View History

2011-07-30 00:36:31 +00:00
#include <QApplication>
2011-07-13 05:27:27 +00:00
#include <QDebug>
#include <QProgressBar>
2011-07-20 03:43:41 +00:00
#include <QMessageBox>
2011-07-29 14:58:44 +00:00
#include "oximetry.h"
2011-07-13 05:27:27 +00:00
#include "ui_oximetry.h"
#include "qextserialport/qextserialenumerator.h"
#include "SleepLib/loader_plugins/cms50_loader.h"
2011-07-29 14:58:44 +00:00
#include "SleepLib/event.h"
#include "Graphs/gXAxis.h"
2011-09-17 12:39:00 +00:00
#include "Graphs/gSummaryChart.h"
#include "Graphs/gLineChart.h"
#include "Graphs/gYAxis.h"
2011-07-13 05:27:27 +00:00
Oximetry::Oximetry(QWidget *parent,Profile * _profile,gGraphView * shared) :
2011-07-13 05:27:27 +00:00
QWidget(parent),
ui(new Ui::Oximetry),
profile(_profile)
2011-07-13 05:27:27 +00:00
{
2011-08-29 07:13:58 +00:00
m_shared=shared;
2011-07-13 05:27:27 +00:00
ui->setupUi(this);
Q_ASSERT(profile!=NULL);
2011-08-29 07:13:58 +00:00
port=NULL;
2011-07-18 02:33:25 +00:00
portname="";
2011-07-29 14:58:44 +00:00
mach=profile->GetMachine(MT_OXIMETER);
if (!mach) {
CMS50Loader *l=dynamic_cast<CMS50Loader *>(GetLoader("CMS50"));
if (l) {
mach=l->CreateMachine(profile);
}
qDebug() << "Create Oximeter device";
}
2011-07-29 14:58:44 +00:00
// Create dummy day & session for holding eventlists.
day=new Day(mach);
session=new Session(mach,0);
day->AddSession(session);
2011-08-29 07:13:58 +00:00
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);
2011-09-17 12:39:00 +00:00
Layer *cl=new gLineChart(OXI_Plethy);
2011-08-29 07:13:58 +00:00
CONTROL->AddLayer(cl);
cl->SetDay(day);
CONTROL->setBlockZoom(true);
for (int i=0;i<GraphView->size();i++) {
gGraph *g=(*GraphView)[i];
g->AddLayer(new gXAxis(),LayerBottom,0,gXAxis::Margin);
g->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin);
g->AddLayer(new gXGrid());
}
2011-07-29 14:58:44 +00:00
// Create the Event Lists to store / import data
2011-09-17 12:39:00 +00:00
ev_plethy=session->AddEventList(OXI_Plethy,EVL_Waveform,1,0,0,0,1000.0/50.0);
2011-07-18 02:33:25 +00:00
2011-09-17 12:39:00 +00:00
ev_pulse=session->AddEventList(OXI_Pulse,EVL_Event,1);
2011-09-17 12:39:00 +00:00
ev_spo2=session->AddEventList(OXI_SPO2,EVL_Event,1);
2011-09-17 12:39:00 +00:00
plethy=new gLineChart(OXI_Plethy,Qt::black,false,true);
2011-07-29 14:58:44 +00:00
plethy->SetDay(day);
2011-08-29 07:13:58 +00:00
CONTROL->AddLayer(plethy); //new gLineChart(OXI_Plethysomogram));
2011-07-29 14:58:44 +00:00
pulse=new gLineChart(OXI_Pulse,Qt::red,true);
pulse->SetDay(day);
spo2=new gLineChart(OXI_SPO2,Qt::blue,true);
spo2->SetDay(day);
2011-07-29 14:58:44 +00:00
PLETHY->AddLayer(plethy);
2011-08-29 07:13:58 +00:00
2011-07-29 14:58:44 +00:00
PULSE->AddLayer(pulse);
SPO2->AddLayer(spo2);
2011-07-18 10:57:34 +00:00
2011-09-22 00:33:29 +00:00
GraphView->setEmptyText("No Oximetry Data");
2011-08-29 07:13:58 +00:00
GraphView->updateGL();
on_RefreshPortsButton_clicked();
2011-07-13 05:27:27 +00:00
}
Oximetry::~Oximetry()
{
delete ui;
}
void Oximetry::on_RefreshPortsButton_clicked()
{
QList<QextPortInfo> ports = QextSerialEnumerator::getPorts();
ui->SerialPortsCombo->clear();
int z=0;
QString firstport;
bool current_found=false;
2011-07-29 14:58:44 +00:00
// Windows build mixes these up
2011-07-19 16:44:03 +00:00
#ifdef Q_WS_WIN32
2011-07-18 11:56:56 +00:00
#define qesPORTNAME portName
#else
#define qesPORTNAME physName
#endif
2011-07-13 05:27:27 +00:00
for (int i = 0; i < ports.size(); i++) {
2011-08-07 03:22:09 +00:00
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;
2011-07-13 05:27:27 +00:00
}
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()
{
2011-08-29 07:13:58 +00:00
GraphView->updateGL();
}
void Oximetry::on_RunButton_toggled(bool checked)
{
if (checked) {
lasttime=0;
lastpulse=0;
lastspo2=0;
2011-07-29 14:58:44 +00:00
// 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??
2011-07-17 12:10:51 +00:00
starttime=lasttime;
session->SetSessionID(lasttime/1000L);
2011-07-29 14:58:44 +00:00
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);
2011-07-29 14:58:44 +00:00
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))
2011-08-29 07:13:58 +00:00
qDebug() << "check device is turned on";
qDebug() << "listening for data on" << port->portName();
} else {
qDebug() << "device failed to open:" << port->errorString();
}
portmode=PM_LIVE;
2011-08-29 07:13:58 +00:00
//foobar->setVisible(false);
CONTROL->setVisible(false);
} else {
2011-08-29 07:13:58 +00:00
//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);
2011-07-29 14:58:44 +00:00
ev_spo2->setLast(lasttime);
ev_pulse->setLast(lasttime);
ev_plethy->setLast(lasttime);
day->setLast(lasttime);
session->set_last(lasttime);
2011-07-29 14:58:44 +00:00
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();
2011-08-29 07:13:58 +00:00
//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;
2011-08-29 07:13:58 +00:00
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";
2011-08-29 07:13:58 +00:00
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); */
2011-08-29 07:13:58 +00:00
sess->eventlist[OXI_SPO2].push_back(ev_spo2);
sess->eventlist[OXI_Pulse].push_back(ev_pulse);
2011-09-17 12:39:00 +00:00
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->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);
2011-09-17 12:39:00 +00:00
sess->avg(OXI_Plethy);
sess->wavg(OXI_Plethy);
sess->p90(OXI_Plethy);
sess->setMin(OXI_Plethy,ev_plethy->min());
sess->setMax(OXI_Plethy,ev_plethy->max());
2011-08-29 07:13:58 +00:00
2011-09-17 12:39:00 +00:00
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));
2011-09-17 12:39:00 +00:00
sess->updateFirst(sess->first(OXI_Plethy));
sess->updateLast(sess->last(OXI_Plethy));
sess->SetChanged(true);
mach->AddSession(sess,profile);
mach->Save();
2011-09-17 12:39:00 +00:00
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);
2011-09-17 12:39:00 +00:00
session->setCount(OXI_Plethy,0);
2011-08-29 07:13:58 +00:00
session->setCount(OXI_Pulse,0);
session->setCount(OXI_SPO2,0);
//m_shared->ResetBounds();
//m_shared->updateScale();
//m_shared->updateGL();
}
2011-08-29 07:13:58 +00:00
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 (d<ev_plethy->min()) ev_plethy->setMin(d);
if (d>ev_plethy->max()) ev_plethy->setMax(d);
2011-08-29 07:13:58 +00:00
int i=ev_plethy->count()+1;
ev_plethy->setCount(i);
2011-09-17 12:39:00 +00:00
session->setCount(OXI_Plethy,i); // update the cache
//ev_plethy->AddEvent(lasttime,d);
2011-07-29 14:58:44 +00:00
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());
2011-07-29 14:58:44 +00:00
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);
2011-07-29 14:58:44 +00:00
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)
{
2011-07-29 14:58:44 +00:00
ev_pulse->AddEvent(lasttime,pul);
2011-08-29 07:13:58 +00:00
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)
{
2011-07-29 14:58:44 +00:00
ev_spo2->AddEvent(lasttime,sp);
2011-08-29 07:13:58 +00:00
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 (i<bytes.size()) {
if (bytes[i]&0x80) {
EventDataType d=bytes[i+1] & 0x7f;
UpdatePlethy(d);
2011-07-29 14:58:44 +00:00
//qDebug() << d;
i+=3;
} else {
UpdatePulse(bytes[i]);
UpdateSPO2(bytes[i+1]);
i+=2;
}
}
2011-08-29 07:13:58 +00:00
if ((ev_plethy->count()<=2) || (ev_pulse->count()<=2) || (ev_spo2->count()<=2)) {
2011-08-29 07:13:58 +00:00
GraphView->updateScale();
}
GraphView->updateGL();
}
2011-07-29 14:58:44 +00:00
void Oximetry::onDsrChanged(bool status) // Doesn't work for CMS50's
{
if (status)
qDebug() << "device was turned on";
else
qDebug() << "device was turned off";
}
2011-07-29 14:58:44 +00:00
extern QProgressBar *qprogress;
extern QLabel *qstatus;
2011-07-18 10:57:34 +00:00
void DumpBytes(int blocks, unsigned char * b,int len)
{
2011-07-18 10:57:34 +00:00
QString a=QString::number(blocks,16)+": Bytes "+QString::number(len,16)+": ";
for (int i=0;i<len;i++) {
a.append(QString::number(b[i],16)+" ");
}
qDebug() << a;
}
// Move this code to CMS50 Importer??
void Oximetry::on_ImportButton_clicked()
{
2011-07-30 10:40:22 +00:00
ui->ImportButton->setDisabled(true);
2011-07-20 03:43:41 +00:00
QMessageBox msgbox(QMessageBox::Information,"Importing","Please Wait",QMessageBox::NoButton,this);
msgbox.show();
2011-07-30 10:40:22 +00:00
QApplication::processEvents();
2011-07-20 03:43:41 +00:00
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];
2011-07-29 14:58:44 +00:00
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;
2011-07-20 03:43:41 +00:00
int dr;
int ec;
do {
bool fnd=false;
2011-07-20 03:43:41 +00:00
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);
2011-07-18 10:57:34 +00:00
done=false;
if (blocks==0) {
2011-07-20 03:43:41 +00:00
if (res>5)
for (int i=0;i<res-2;i++) {
if ((rb[i]==0xf2) && (rb[i+1]==0x80) && (rb[i+2]==0x00)) {
fnd=true;
2011-07-18 10:57:34 +00:00
startpos=i+9;
break;
}
}
if (!fnd) {
qDebug() << "Retrying..";
fails++;
break; // reissue the F5 and try again
2011-07-18 10:57:34 +00:00
} else {
qDebug() << "Found";
}
2011-07-18 10:57:34 +00:00
// 84 95 7c
// 0x10556 bytes..
// 0x82b blocks
2011-07-20 03:43:41 +00:00
length=0;
while (rb[startpos]!=0xf0) {
length=(length << 7) | (rb[startpos++] & 0x7f);
}
2011-07-27 09:21:53 +00:00
//length=(rb[startpos++] & 0x7f) << 14;
//length|=(rb[startpos++] & 0x7f) << 7;
//length|=(rb[startpos++] & 0x7f);
//if (!(rb[startpos]&0x80)) {
// length <<= 8;
// length|=rb[startpos++];
//}
//length=(rb[startpos] ^ 0x80)<< 7 | (rb[startpos+1]);
//startpos+=2;
//if (!(rb[startpos]&0x80)) {
// length|=(rb[startpos]&0x7f) << 14;
// startpos++;
//} else oneoff=true;
buffer=new unsigned char [length+32];
//qDebug() << length << startpos;
bytes+=res-startpos;
memcpy((char *)buffer,(char *)&rb[startpos],bytes);
2011-07-30 10:40:22 +00:00
qprogress->setValue((75.0/length)*bytes);
QApplication::processEvents();
} else {
2011-07-30 10:40:22 +00:00
qprogress->setValue((75.0/length)*bytes);
2011-07-30 00:36:31 +00:00
QApplication::processEvents();
memcpy((char *)&buffer[bytes],(char *)rb,res);
bytes+=res;
}
blocks++;
2011-07-20 03:43:41 +00:00
if (res<rb_size) {
2011-07-18 10:57:34 +00:00
qDebug() << "Read "<< bytes << " bytes of " << length;
done=true;
break;
}
} while (bytes<length);
if (done) break;
if (fails>4) break;
}
if (done) {
if (oneoff) bytes--; // this is retarded..
QDateTime date=QDateTime::currentDateTime().toUTC();
SessionID sid=date.toTime_t();
2011-07-29 14:58:44 +00:00
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;
2011-07-29 14:58:44 +00:00
session->set_first(tt);
2011-07-30 10:40:22 +00:00
ev_pulse->setMin(999999);
ev_pulse->setMax(-999999);
ev_spo2->setMin(999999);
ev_spo2->setMax(-999999);
2011-07-29 14:58:44 +00:00
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 (i<bytes) {
if (buffer[i++]!=0xf0) {
qDebug() << "Faulty PulseOx data";
continue;
}
pulse=buffer[i++] & 0x7f;
spo2=buffer[i++];
if (pulse!=lastpulse) {
data=pulse;
2011-07-29 14:58:44 +00:00
ev_pulse->AddEvent(tt,data);
//qDebug() << "Pulse: " << int(pulse);
}
if (spo2!=lastspo2) {
data=spo2;
2011-07-29 14:58:44 +00:00
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) min=rb_pulse[j];
if (rb_pulse[j]>min) max=rb_pulse[j];
}
if (min>0 && max>0) {
int drop=max-min;
if (drop>6) {
if (!oxf1) {
2011-09-17 12:39:00 +00:00
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) min=rb_spo2[j];
if (rb_spo2[j]>min) max=rb_spo2[j];
}
if (min>0 && max>0) {
int drop=max-min;
if (drop>4) {
if (!oxf1) {
2011-09-17 12:39:00 +00:00
oxf2=session->AddEventList(OXI_SPO2Drop,EVL_Event);
}
oxf2->AddEvent(tt,drop);
}
}
++rb_pos;
rb_pos=rb_pos % rb_size;
tt+=1000;
2011-07-30 10:40:22 +00:00
qprogress->setValue(75+(25.0/bytes)*i);
QApplication::processEvents();
}
2011-07-29 14:58:44 +00:00
ev_pulse->AddEvent(tt,pulse);
ev_spo2->AddEvent(tt,spo2);
2011-07-30 10:40:22 +00:00
session->set_last(tt);
2011-07-29 14:58:44 +00:00
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);
2011-07-29 14:58:44 +00:00
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);
2011-07-29 14:58:44 +00:00
session->SetChanged(true);
mach->AddSession(session,profile);
mach->Save();
// Output Pulse & SPO2 here..
delete [] buffer;
port->write((char *)b2,3);
2011-07-29 14:58:44 +00:00
// 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
2011-09-17 12:39:00 +00:00
ev_plethy=session->AddEventList(OXI_Plethy,EVL_Waveform,1,0,0,0,1000.0/50.0);
2011-07-29 14:58:44 +00:00
2011-09-17 12:39:00 +00:00
ev_pulse=session->AddEventList(OXI_Pulse,EVL_Event,1);
2011-07-29 14:58:44 +00:00
2011-09-17 12:39:00 +00:00
ev_spo2=session->AddEventList(OXI_SPO2,EVL_Event,1);
}
delete port;
port=NULL;
2011-07-20 03:43:41 +00:00
msgbox.hide();
qprogress->hide();
ui->SerialPortsCombo->setEnabled(true);
qstatus->setText("Ready");
2011-07-30 10:40:22 +00:00
ui->ImportButton->setDisabled(false);
2011-07-13 05:27:27 +00:00
}
2011-07-29 14:58:44 +00:00