Oximetry Tab & CMS50 Serial module Complete Rewrite

This commit is contained in:
Mark Watkins 2011-11-24 22:47:25 +10:00
parent 5b04580384
commit 088d4ea48a
18 changed files with 1034 additions and 622 deletions

View File

@ -136,7 +136,7 @@ void gFlagsLine::paint(gGraph & w,int left, int top, int width, int height)
EventList & el=*((*s)->eventlist[m_code][0]);
for (int i=0;i<el.count();i++) {
for (quint32 i=0;i<el.count();i++) {
X=el.time(i);
Y=X-(el.data(i)*1000);
if (Y < minx) continue;

View File

@ -1407,7 +1407,7 @@ void gGraph::keyPressEvent(QKeyEvent * event)
for (QVector<Layer *>::iterator i=m_layers.begin();i!=m_layers.end();i++) {
(*i)->keyPressEvent(event);
}
qDebug() << m_title << "Key Pressed.. implement me" << event->key();
//qDebug() << m_title << "Key Pressed.. implement me" << event->key();
}
void gGraph::ZoomX(double mult,int origin_px)

View File

@ -532,14 +532,19 @@ void AHIChart::SetDay(Day *d)
for (qint64 ti=first;ti<last;ti+=winsize) {
f=ti-3600000L;
//if (f<first) f=first;
EventList *el[4];
EventList *el[6];
EventDataType ahi=0;
int cnt=0;
bool fnd=false;
for (s=d->begin();s!=d->end();s++) {
Session *sess=*s;
if ((ti<sess->first()) || (f>sess->last())) continue;
// Drop off suddenly outside of sessions
//if (ti>sess->last()) continue;
fnd=true;
if (sess->eventlist.contains(CPAP_Obstructive))
el[0]=sess->eventlist[CPAP_Obstructive][0];
else el[0]=NULL;
@ -552,18 +557,25 @@ void AHIChart::SetDay(Day *d)
if (sess->eventlist.contains(CPAP_ClearAirway))
el[3]=sess->eventlist[CPAP_ClearAirway][0];
else el[3]=NULL;
if (sess->eventlist.contains(CPAP_NRI))
el[4]=sess->eventlist[CPAP_NRI][0];
else el[4]=NULL;
/*if (sess->eventlist.contains(CPAP_ExP))
el[5]=sess->eventlist[CPAP_ExP][0];
else el[5]=NULL;*/
qint64 t;
for (int i=0;i<4;i++) {
for (int i=0;i<5;i++) {
if (!el[i]) continue;
for (int j=0;j<el[i]->count();j++) {
for (quint32 j=0;j<el[i]->count();j++) {
t=el[i]->time(j);
if ((t>=f) && (t<=ti)) {
if ((t>=f) && (t<ti)) {
cnt++;
}
}
}
}
if (!fnd) cnt=0;
double g=double(ti-f)/3600000.0;
if (g>0) ahi=cnt/g;

View File

@ -62,7 +62,7 @@ void gLineOverlayBar::paint(gGraph & w, int left, int topp, int width, int heigh
EventList & el=*cei.value()[0];
for (int i=0;i<el.count();i++) {
for (quint32 i=0;i<el.count();i++) {
X=el.time(i);
if (m_flt==FT_Span) {
Y=X-(qint64(el.raw(i))*1000.0L); // duration

View File

@ -218,7 +218,7 @@ void gTAPGraph::SetDay(Day *d)
}
//first=true;
//changed=false;
for (int i=1;i<el.count();i++) {
for (quint32 i=1;i<el.count();i++) {
data=el.raw(i);
time=el.time(i);
if (lastval!=data) {

View File

@ -22,5 +22,6 @@
<file>icons/toggle-off-us.svg</file>
<file>icons/toggle-on-us.svg</file>
<file>docs/release_notes.html</file>
<file>icons/save.png</file>
</qresource>
</RCC>

View File

@ -26,7 +26,7 @@ EventList::EventList(EventListType et,EventDataType gain, EventDataType offset,
EventList::~EventList()
{
}
qint64 EventList::time(int i)
qint64 EventList::time(quint32 i)
{
if (m_type==EVL_Event) {
return m_first+qint64(m_time[i]);
@ -35,7 +35,7 @@ qint64 EventList::time(int i)
return m_first+qint64((EventDataType(i)*m_rate));
}
EventDataType EventList::data(int i)
EventDataType EventList::data(quint32 i)
{
return EventDataType(m_data[i])*m_gain;
}
@ -59,7 +59,7 @@ void EventList::AddEvent(qint64 time, EventStoreType data)
// This really shouldn't happen.
qint64 t=(m_first-time);
for (int i=0;i<m_count;i++) {
for (quint32 i=0;i<m_count;i++) {
m_time[i]-=t & 0xffffffff;
}
m_first=time;

View File

@ -25,13 +25,13 @@ public:
void AddWaveform(qint64 start, unsigned char * data, int recs, qint64 duration);
void AddWaveform(qint64 start, char * data, int recs, qint64 duration);
inline const int & count() { return m_count; }
void setCount(int count) { m_count=count; }
inline const quint32 & count() { return m_count; }
void setCount(quint32 count) { m_count=count; }
inline EventStoreType raw(int i) { return m_data[i]; }
EventDataType data(int i);
qint64 time(int i);
EventDataType data(quint32 i);
qint64 time(quint32 i);
inline const qint64 & first() { return m_first; }
inline const qint64 & last() { return m_last; }
inline qint64 duration() { return m_last-m_first; }
@ -64,7 +64,7 @@ protected:
QVector<EventStoreType> m_data;
//ChannelID m_code;
EventListType m_type;
int m_count;
quint32 m_count;
EventDataType m_gain;
EventDataType m_offset;

View File

@ -75,8 +75,8 @@ int IntellipapLoader::Open(QString & path,Profile *profile)
lookup["Pl"]="MinPressure";
//lookup["Ds"]="Ds";
//lookup["Pc"]="Pc";
lookup["Pd"]="RampPressure";
lookup["Dt"]="RampTime";
lookup["Pd"]="RampPressure"; // Delay Pressure
lookup["Dt"]="RampTime"; // Delay Time
//lookup["Ld"]="Ld";
//lookup["Lh"]="Lh";
//lookup["FC"]="FC";
@ -88,8 +88,8 @@ int IntellipapLoader::Open(QString & path,Profile *profile)
lookup["Hd"]="HypopneaDuration";
//lookup["Pi"]="Pi"; //080
//lookup["Pe"]="Pe"; //WF
//lookup["Ri"]="SmartFlexIRnd"; //1
//lookup["Re"]="SmartFlexERnd"; //2
lookup["Ri"]="SmartFlexIRnd"; // Inhale Rounding (0-5)
lookup["Re"]="SmartFlexERnd"; // Exhale Rounding (0-5)
//lookup["Bu"]="Bu"; //WF
//lookup["Ie"]="Ie"; //20
//lookup["Se"]="Se"; //05
@ -102,7 +102,7 @@ int IntellipapLoader::Open(QString & path,Profile *profile)
//lookup["Hp"]="Hp"; //1
//lookup["Hs"]="Hs"; //02
//lookup["Lu"]="LowUseThreshold"; // defaults to 0 (4 hours)
//lookup["Sf"]="SmartFlex";
lookup["Sf"]="SmartFlex";
//lookup["Sm"]="SmartFlexMode";
lookup["Ks=s"]="Ks_s";
lookup["Ks=i"]="Ks_i";
@ -254,13 +254,22 @@ int IntellipapLoader::Open(QString & path,Profile *profile)
sess->eventlist[CPAP_Te][0]->AddEvent(time,m_buffer[pos+0xf]);
sess->eventlist[CPAP_Ti][0]->AddEvent(time,m_buffer[pos+0xc]);
sess->eventlist[CPAP_Snore][0]->AddEvent(time,m_buffer[pos+0x5]); //4/5??
sess->eventlist[CPAP_Snore][0]->AddEvent(time,m_buffer[pos+0x4]); //4/5??
if (m_buffer[pos+0x5]>0) {
if (!sess->eventlist.contains(CPAP_VSnore)) {
sess->AddEventList(CPAP_VSnore,EVL_Event);
// 0x0f == Leak Event
// 0x04 == Snore?
if (m_buffer[pos+0xf]>0) { // Leak Event
if (!sess->eventlist.contains(CPAP_LeakFlag)) {
sess->AddEventList(CPAP_LeakFlag,EVL_Event);
}
sess->eventlist[CPAP_VSnore][0]->AddEvent(time,m_buffer[pos+0x5]);
sess->eventlist[CPAP_LeakFlag][0]->AddEvent(time,m_buffer[pos+0xf]);
}
if (m_buffer[pos+0x5]>4) { // This matches Exhale Puff.. not sure why 4
if (!sess->eventlist.contains(CPAP_ExP)) {
sess->AddEventList(CPAP_ExP,EVL_Event);
}
sess->eventlist[CPAP_ExP][0]->AddEvent(time,m_buffer[pos+0x5]);
}
if (m_buffer[pos+0x10]>0) {
@ -273,13 +282,14 @@ int IntellipapLoader::Open(QString & path,Profile *profile)
if (!sess->eventlist.contains(CPAP_Hypopnea)) {
sess->AddEventList(CPAP_Hypopnea,EVL_Event);
}
//for (int i=0;i<m_buffer[pos+0x11];i++)
sess->eventlist[CPAP_Hypopnea][0]->AddEvent(time,m_buffer[pos+0x11]);
}
if (m_buffer[pos+0x12]>0) {
if (!sess->eventlist.contains(CPAP_Apnea)) {
sess->AddEventList(CPAP_Apnea,EVL_Event);
if (m_buffer[pos+0x12]>0) { // NRI // is this == to RERA?? CA??
if (!sess->eventlist.contains(CPAP_NRI)) {
sess->AddEventList(CPAP_NRI,EVL_Event);
}
sess->eventlist[CPAP_Apnea][0]->AddEvent(time,m_buffer[pos+0x12]);
sess->eventlist[CPAP_NRI][0]->AddEvent(time,m_buffer[pos+0x12]);
}
quint16 tv=(m_buffer[pos+0x8] << 8) | m_buffer[pos+0x9]; // correct
sess->eventlist[CPAP_TidalVolume][0]->AddEvent(time,tv);

View File

@ -72,6 +72,9 @@ const QString CPAP_Hypopnea="Hypopnea";
const QString CPAP_ClearAirway="ClearAirway";
const QString CPAP_Apnea="Apnea";
const QString CPAP_CSR="CSR";
const QString CPAP_LeakFlag="LeakFlag";
const QString CPAP_ExP="ExP";
const QString CPAP_NRI="NRI";
const QString CPAP_VSnore="VSnore";
const QString CPAP_VSnore2="VSnore2";
const QString CPAP_RERA="RERA";

View File

@ -274,11 +274,11 @@ bool Session::StoreEvents(QString filename)
for (int j=0;j<i.value().size();j++) {
EventList &e=*i.value()[j];
for (int c=0;c<e.count();c++) {
for (quint32 c=0;c<e.count();c++) {
out << e.raw(c);
}
if (e.type()!=EVL_Waveform) {
for (int c=0;c<e.count();c++) {
for (quint32 c=0;c<e.count();c++) {
out << e.getTime()[c];
}
}
@ -384,7 +384,7 @@ bool Session::LoadEvents(QString filename)
for (int j=0;j<size2;j++) {
EventList &evec=*eventlist[code][j];
evec.m_data.reserve(evec.m_count);
for (int c=0;c<evec.m_count;c++) {
for (quint32 c=0;c<evec.m_count;c++) {
in >> t;
//evec.m_data[c]=t;
evec.m_data.push_back(t);
@ -392,7 +392,7 @@ bool Session::LoadEvents(QString filename)
//last=evec.first();
if (evec.type()!=EVL_Waveform) {
evec.m_time.reserve(evec.m_count);
for (int c=0;c<evec.m_count;c++) {
for (quint32 c=0;c<evec.m_count;c++) {
in >> x;
//last+=x;
evec.m_time.push_back(x);
@ -591,7 +591,7 @@ double Session::sum(ChannelID id)
double sum=0;
for (int i=0;i<evec.size();i++) {
for (int j=0;j<evec[i]->count();j++) {
for (quint32 j=0;j<evec[i]->count();j++) {
sum+=evec[i]->data(j);
}
}
@ -615,7 +615,7 @@ EventDataType Session::avg(ChannelID id)
double val=0;
int cnt=0;
for (int i=0;i<evec.size();i++) {
for (int j=0;j<evec[i]->count();j++) {
for (quint32 j=0;j<evec[i]->count();j++) {
val+=evec[i]->data(j);
cnt++;
}
@ -734,7 +734,7 @@ EventDataType Session::wavg(ChannelID id)
if (!evec[i]->count()) continue;
lastval=evec[i]->raw(0);
lasttime=evec[i]->time(0);
for (int j=1;j<evec[i]->count();j++) {
for (quint32 j=1;j<evec[i]->count();j++) {
val=evec[i]->raw(j);
time=evec[i]->time(j);
td=(time-lasttime);

View File

@ -112,10 +112,13 @@ Daily::Daily(QWidget *parent,gGraphView * shared, MainWindow *mw)
gFlagsGroup *fg=new gFlagsGroup();
fg->AddLayer((new gFlagsLine(CPAP_CSR,QColor("light green"),"CSR",false,FT_Span)));
fg->AddLayer((new gFlagsLine(CPAP_ClearAirway,QColor("purple"),"CA",true)));
fg->AddLayer((new gFlagsLine(CPAP_ClearAirway,QColor("purple"),"CA",false)));
fg->AddLayer((new gFlagsLine(CPAP_Obstructive,QColor("#40c0ff"),"OA",true)));
fg->AddLayer((new gFlagsLine(CPAP_Apnea,QColor("dark green"),"A")));
fg->AddLayer((new gFlagsLine(CPAP_Hypopnea,QColor("blue"),"H",true)));
fg->AddLayer((new gFlagsLine(CPAP_ExP,QColor("dark cyan"),"E",false)));
fg->AddLayer((new gFlagsLine(CPAP_LeakFlag,QColor("dark blue"),"L",false)));
fg->AddLayer((new gFlagsLine(CPAP_NRI,QColor("dark magenta"),"NRI",false)));
fg->AddLayer((new gFlagsLine(CPAP_FlowLimit,QColor("black"),"FL")));
fg->AddLayer((new gFlagsLine(CPAP_RERA,QColor("gold"),"RE")));
fg->AddLayer((new gFlagsLine(CPAP_VSnore,QColor("red"),"VS")));
@ -305,7 +308,7 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day)
mcr=mcroot[code];
}
for (int z=0;z<m.value().size();z++) {
for (int o=0;o<m.value()[z]->count();o++) {
for (quint32 o=0;o<m.value()[z]->count();o++) {
qint64 t=m.value()[z]->time(o);
if (code==CPAP_CSR) {
@ -469,7 +472,7 @@ void Daily::Load(QDate date)
float csr=(100.0/cpap->hours())*(cpap->sum(CPAP_CSR)/3600.0);
float uai=cpap->count(CPAP_Apnea)/cpap->hours();
float oai=cpap->count(CPAP_Obstructive)/cpap->hours();
float hi=cpap->count(CPAP_Hypopnea)/cpap->hours();
float hi=(cpap->count(CPAP_ExP)+cpap->count(CPAP_Hypopnea))/cpap->hours();
float cai=cpap->count(CPAP_ClearAirway)/cpap->hours();
float rei=cpap->count(CPAP_RERA)/cpap->hours();
float vsi=cpap->count(CPAP_VSnore)/cpap->hours();
@ -558,7 +561,7 @@ void Daily::Load(QDate date)
CPAP_Pressure,CPAP_EPAP,CPAP_IPAP,CPAP_PS,CPAP_PTB,
CPAP_MinuteVent,CPAP_RespRate,CPAP_RespEvent,CPAP_FLG,
CPAP_Leak,CPAP_Snore,CPAP_IE,CPAP_Ti,CPAP_Te, CPAP_TgMV,
CPAP_TidalVolume, OXI_Pulse,OXI_SPO2
CPAP_TidalVolume, OXI_Pulse, OXI_SPO2
};
int numchans=sizeof(chans)/sizeof(ChannelID);
int suboffset;
@ -916,3 +919,4 @@ void Daily::on_todayButton_clicked()
if (d > PROFILE.LastDay()) d=PROFILE.LastDay();
LoadDate(d);
}

View File

@ -64,7 +64,6 @@ private slots:
void on_calButton_toggled(bool checked);
void on_todayButton_clicked();
protected:
private:

View File

@ -46,6 +46,9 @@ One id code per item
<channel id="0x1113" class="data" name="FLG" details="Flow Limit Graph" label="Flow Limit" unit="" color="dark grey"/>
<channel id="0x1114" class="data" name="TgMV" details="Target Minute Ventilation" label="Trgt Min Vent." unit="" color="dark cyan"/>
<channel id="0x1115" class="data" name="MaxLeak" details="Maximum Leak" label="MaxLeaks" unit="L/min" color="dark red"/>
<channel id="0x1116" class="data" name="LeakFlag" details="Leak Event" label="L" unit="" color="dark blue"/>
<channel id="0x1117" class="data" name="NRI" details="Non Responding Event" label="NRI" unit="" color="orange"/>
<channel id="0x1118" class="data" name="EP" details="Exhale Puff" label="EP" unit="" color="dark purple"/>
<channel id="0x1150" class="data" name="PRS1_00" details="Unknown 00" label="U00" unit="" color="black"/>
<channel id="0x1151" class="data" name="PRS1_01" details="Unknown 01" label="U01" unit="" color="black"/>
@ -69,7 +72,7 @@ One id code per item
</group>
<group name="OXI">
<channel id="0x1800" class="data" name="Pulse" details="Pulse Rate" label="Pulse" color="red"/>
<channel id="0x1801" class="data" name="SPO2" details="SPO2" label="SPO2" color="blue"/>
<channel id="0x1801" class="data" name="SPO2" details="% SpO2" label="SpO2" color="blue"/>
<channel id="0x1802" class="data" name="Plethy" details="Plethysomogram" label="Plethy" color="black"/>
<channel id="0x1803" class="data" name="PulseChange" details="Pulse Change" label="Pulse Change" color="red"/>
<channel id="0x1804" class="data" name="SPO2Drop" details="SPO2Drop" label="SPO2 Drop" color="blue"/>

View File

@ -245,7 +245,7 @@ void ExportCSV::on_exportButton_clicked()
//header="DateTime"+sep+"Session"+sep+"Event"+sep+"Data/Duration";
for (int e=0;e<fnd.value().size();e++) {
EventList *ev=fnd.value()[e];
for (int q=0;q<ev->count();q++) {
for (quint32 q=0;q<ev->count();q++) {
data=QDateTime::fromTime_t(ev->time(q)/1000L).toString(Qt::ISODate);
data+=sep+QString::number(sess->session());
data+=sep+key;

File diff suppressed because it is too large Load Diff

View File

@ -20,6 +20,111 @@
#include "Graphs/gLineChart.h"
#include "Graphs/gFooBar.h"
class SerialOximeter:public QObject
{
Q_OBJECT
public:
explicit SerialOximeter(QObject * parent,QString oxiname, QString portname="",BaudRateType baud=BAUD19200, FlowType flow=FLOW_OFF, ParityType parity=PAR_ODD, DataBitsType databits=DATA_8, StopBitsType stopbits=STOP_1);
virtual ~SerialOximeter();
virtual bool Open(QextSerialPort::QueryMode mode=QextSerialPort::EventDriven);
virtual void Close();
virtual bool startImport()=0;
virtual void stopImport() {} // abort, default do nothing.
virtual bool startLive();
virtual void stopLive();
qint64 lastTime() { return lasttime; }
Machine * getMachine() { return machine; }
Session *createSession();
Session * getSession() { return session; }
Session * takeSession() { Session * s=session; session=NULL; return s; }
void setPortName(QString portname);
void setBaudRate(BaudRateType baud);
void setFlowControl(FlowType flow);
void setParity(ParityType parity);
void setDataBits(DataBitsType databits);
void setStopBits(StopBitsType stopbits);
QString portName() { return m_portname; }
BaudRateType baudRate() { return m_baud; }
FlowType flowControl() { return m_flow; }
ParityType parity() { return m_parity; }
DataBitsType dataBits() { return m_databits; }
StopBitsType stopBits() { return m_stopbits; }
signals:
void sessionCreated(Session *);
void dataChanged();
void importProcess();
void importComplete(Session *);
void importAborted();
void updateProgress(float f); // between 0 and 1.
void liveStopped(Session *);
void updatePulse(float p);
void updateSpO2(float p);
protected slots:
virtual void onReadyRead();
virtual void on_import_process()=0;
protected:
//virtual void addEvents(EventDataType pr, EventDataType o2, EventDataType pleth=-1000000);
virtual void addPulse(qint64 time, EventDataType pr);
virtual void addSpO2(qint64 time, EventDataType o2);
virtual void addPlethy(qint64 time, EventDataType pleth);
Session * session;
EventList * pulse;
EventList * spo2;
EventList * plethy;
QextSerialPort *m_port;
bool m_opened;
QString m_oxiname;
QString m_portname;
BaudRateType m_baud;
FlowType m_flow;
ParityType m_parity;
DataBitsType m_databits;
StopBitsType m_stopbits;
QextSerialPort::QueryMode m_mode;
Machine *machine;
qint64 lasttime;
bool import_mode;
};
class CMS50Serial:public SerialOximeter
{
public:
explicit CMS50Serial(QObject * parent,QString portname);
virtual ~CMS50Serial();
virtual bool startImport();
protected:
virtual void on_import_process();
virtual void onReadyRead();
bool waitf6;
short cntf6;
QByteArray data;
QVector<QDateTime> f2time;
int datasize;
int received_bytes;
};
namespace Ui {
class Oximetry;
}
@ -42,16 +147,22 @@ private slots:
void on_RunButton_toggled(bool checked);
void on_SerialPortsCombo_activated(const QString &arg1);
void onReadyRead();
void onDsrChanged(bool status);
//void onReadyRead();
//void onDsrChanged(bool status);
void on_ImportButton_clicked();
private:
bool UpdatePulse(qint8 pulse);
bool UpdateSPO2(qint8 spo2);
void UpdatePlethy(qint8 plethy);
void onDataChanged();
void onPulseChanged(float p);
void onSpO2Changed(float o2);
void on_saveButton_clicked();
void on_updateProgress(float f);
void on_import_aborted();
void on_import_complete(Session *session);
private:
void import_finished();
Ui::Oximetry *ui;
gGraphView *GraphView;
@ -69,14 +180,20 @@ private:
double lasttime,starttime;
int lastpulse, lastspo2;
Machine * mach;
Day * day;
Session * session;
EventList * ev_pulse;
EventList * ev_spo2;
EventList * ev_plethy;
//Session * session;
//EventList * ev_pulse;
//EventList * ev_spo2;
//EventList * ev_plethy;
Layer * foobar;
gGraphView * m_shared;
SerialOximeter *oximeter;
bool firstSPO2Update;
bool firstPulseUpdate;
bool secondPulseUpdate;
bool secondSPO2Update;
};
#endif // OXIMETRY_H

View File

@ -89,6 +89,137 @@
</property>
</widget>
</item>
<item>
<widget class="QToolButton" name="saveButton">
<property name="text">
<string>...</string>
</property>
<property name="icon">
<iconset resource="Resources.qrc">
<normaloff>:/icons/save.png</normaloff>:/icons/save.png</iconset>
</property>
</widget>
</item>
<item>
<widget class="QLabel" name="label_2">
<property name="text">
<string>Pulse</string>
</property>
</widget>
</item>
<item>
<widget class="QLCDNumber" name="pulseLCD">
<property name="sizePolicy">
<sizepolicy hsizetype="Fixed" vsizetype="Minimum">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<property name="palette">
<palette>
<active>
<colorrole role="WindowText">
<brush brushstyle="SolidPattern">
<color alpha="255">
<red>255</red>
<green>0</green>
<blue>0</blue>
</color>
</brush>
</colorrole>
</active>
<inactive>
<colorrole role="WindowText">
<brush brushstyle="SolidPattern">
<color alpha="255">
<red>255</red>
<green>0</green>
<blue>0</blue>
</color>
</brush>
</colorrole>
</inactive>
<disabled>
<colorrole role="WindowText">
<brush brushstyle="SolidPattern">
<color alpha="255">
<red>118</red>
<green>118</green>
<blue>117</blue>
</color>
</brush>
</colorrole>
</disabled>
</palette>
</property>
<property name="numDigits">
<number>3</number>
</property>
<property name="segmentStyle">
<enum>QLCDNumber::Flat</enum>
</property>
</widget>
</item>
<item>
<widget class="QLabel" name="label_3">
<property name="text">
<string>SpO2</string>
</property>
</widget>
</item>
<item>
<widget class="QLCDNumber" name="spo2LCD">
<property name="sizePolicy">
<sizepolicy hsizetype="Fixed" vsizetype="Minimum">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<property name="palette">
<palette>
<active>
<colorrole role="WindowText">
<brush brushstyle="SolidPattern">
<color alpha="255">
<red>85</red>
<green>0</green>
<blue>255</blue>
</color>
</brush>
</colorrole>
</active>
<inactive>
<colorrole role="WindowText">
<brush brushstyle="SolidPattern">
<color alpha="255">
<red>85</red>
<green>0</green>
<blue>255</blue>
</color>
</brush>
</colorrole>
</inactive>
<disabled>
<colorrole role="WindowText">
<brush brushstyle="SolidPattern">
<color alpha="255">
<red>118</red>
<green>118</green>
<blue>117</blue>
</color>
</brush>
</colorrole>
</disabled>
</palette>
</property>
<property name="numDigits">
<number>3</number>
</property>
<property name="segmentStyle">
<enum>QLCDNumber::Flat</enum>
</property>
</widget>
</item>
<item>
<spacer name="horizontalSpacer">
<property name="orientation">