Temporary debug tab + Oximetry Live view improvements

This commit is contained in:
Mark Watkins 2011-07-18 01:57:45 +10:00
parent f94ae05ca8
commit 766f09e0e8
6 changed files with 171 additions and 67 deletions

View File

@ -17,12 +17,35 @@
#include "SleepLib/loader_plugins/zeo_loader.h" #include "SleepLib/loader_plugins/zeo_loader.h"
#include "SleepLib/loader_plugins/resmed_loader.h" #include "SleepLib/loader_plugins/resmed_loader.h"
MainWindow *mainwin;
void MyOutputHandler(QtMsgType type, const char *msg) {
if (!mainwin) return;
switch (type) {
case QtDebugMsg:
mainwin->Log(msg);
break;
case QtWarningMsg:
mainwin->Log(QString("Warning: ")+msg);
break;
case QtFatalMsg:
mainwin->Log(QString("Fatal: ")+msg);
break;
case QtCriticalMsg:
mainwin->Log(QString("Critical: ")+msg);
break;
// Popup a messagebox
//abort();
}
}
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
QApplication a(argc, argv); QApplication a(argc, argv);
qInstallMsgHandler(MyOutputHandler);
a.setApplicationName("SleepyHead"); a.setApplicationName("SleepyHead");
int id=QFontDatabase::addApplicationFont(":/fonts/FreeSans.ttf"); int id=QFontDatabase::addApplicationFont(":/fonts/FreeSans.ttf");
QStringList ffam=QFontDatabase::applicationFontFamilies(id); QStringList ffam=QFontDatabase::applicationFontFamilies(id);
for (QStringList::iterator i=ffam.begin();i!=ffam.end();i++) { for (QStringList::iterator i=ffam.begin();i!=ffam.end();i++) {
@ -30,13 +53,14 @@ int main(int argc, char *argv[])
} }
a.setFont(QFont("FreeSans",10)); a.setFont(QFont("FreeSans",10));
MainWindow w;
mainwin=&w;
PRS1Loader::Register(); PRS1Loader::Register();
CMS50Loader::Register(); CMS50Loader::Register();
ZEOLoader::Register(); ZEOLoader::Register();
ResmedLoader::Register(); ResmedLoader::Register();
MainWindow w;
w.show(); w.show();
return a.exec(); return a.exec();

View File

@ -20,6 +20,12 @@
QProgressBar *qprogress; QProgressBar *qprogress;
QLabel *qstatus; QLabel *qstatus;
void MainWindow::Log(QString s)
{
ui->logText->appendPlainText(s);
}
QString subversion="0"; QString subversion="0";
MainWindow::MainWindow(QWidget *parent) : MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent), QMainWindow(parent),
@ -31,6 +37,7 @@ MainWindow::MainWindow(QWidget *parent) :
#endif #endif
ui->setupUi(this); ui->setupUi(this);
this->setWindowTitle(tr("SleepyHead")+QString(" v0.8.")+subversion); this->setWindowTitle(tr("SleepyHead")+QString(" v0.8.")+subversion);
ui->tabWidget->setCurrentIndex(0);
QGLFormat fmt; QGLFormat fmt;
fmt.setDepth(false); fmt.setDepth(false);
@ -60,6 +67,9 @@ MainWindow::MainWindow(QWidget *parent) :
if (!pref.Exists("LinkGraphMovement")) pref["LinkGraphMovement"]=true; if (!pref.Exists("LinkGraphMovement")) pref["LinkGraphMovement"]=true;
else ui->action_Link_Graphs->setChecked(pref["LinkGraphMovement"].toBool()); else ui->action_Link_Graphs->setChecked(pref["LinkGraphMovement"].toBool());
if (!pref.Exists("ShowDebug")) pref["ShowDebug"]=false;
else ui->actionDebug->setChecked(pref["ShowDebug"].toBool());
if (!pref.Exists("NoonDateSplit")) pref["NoonDateSplit"]=false; if (!pref.Exists("NoonDateSplit")) pref["NoonDateSplit"]=false;
else ui->action_Noon_Date_Split->setChecked(pref["NoonDateSplit"].toBool()); else ui->action_Noon_Date_Split->setChecked(pref["NoonDateSplit"].toBool());
@ -99,13 +109,13 @@ void MainWindow::Startup()
profile->LoadMachineData(); profile->LoadMachineData();
daily=new Daily(ui->tabWidget,shared_context); daily=new Daily(ui->tabWidget,shared_context);
ui->tabWidget->addTab(daily,tr("Daily")); ui->tabWidget->insertTab(1,daily,tr("Daily"));
overview=new Overview(ui->tabWidget,shared_context); overview=new Overview(ui->tabWidget,shared_context);
ui->tabWidget->addTab(overview,tr("Overview")); ui->tabWidget->insertTab(2,overview,tr("Overview"));
oximetry=new Oximetry(ui->tabWidget); oximetry=new Oximetry(ui->tabWidget);
ui->tabWidget->addTab(oximetry,tr("Oximetry")); ui->tabWidget->insertTab(3,oximetry,tr("Oximetry"));
qprogress->hide(); qprogress->hide();
qstatus->setText(tr("Ready")); qstatus->setText(tr("Ready"));
@ -257,3 +267,16 @@ void MainWindow::on_action_Noon_Date_Split_toggled(bool checked)
{ {
pref["NoonDateSplit"]=checked; pref["NoonDateSplit"]=checked;
} }
void MainWindow::on_actionDebug_toggled(bool checked)
{
pref["ShowDebug"]=checked;
int idx=ui->tabWidget->indexOf(ui->debugTab);
if (checked) {
//ui->debugTab->show();
//ui->tabWidget->setTabEnabled(idx,true);
} else {
//ui->debugTab->hide();
// ui->tabWidget->setTabEnabled(idx,false);
}
}

View File

@ -25,6 +25,7 @@ class MainWindow : public QMainWindow
public: public:
explicit MainWindow(QWidget *parent = 0); explicit MainWindow(QWidget *parent = 0);
~MainWindow(); ~MainWindow();
void Log(QString s);
private slots: private slots:
void on_action_Import_Data_triggered(); void on_action_Import_Data_triggered();
@ -63,6 +64,8 @@ private slots:
void on_action_Noon_Date_Split_toggled(bool ); void on_action_Noon_Date_Split_toggled(bool );
void on_actionDebug_toggled(bool arg1);
private: private:
Ui::MainWindow *ui; Ui::MainWindow *ui;
Daily * daily; Daily * daily;

View File

@ -76,7 +76,7 @@
<enum>QTabWidget::North</enum> <enum>QTabWidget::North</enum>
</property> </property>
<property name="currentIndex"> <property name="currentIndex">
<number>0</number> <number>1</number>
</property> </property>
<property name="documentMode"> <property name="documentMode">
<bool>false</bool> <bool>false</bool>
@ -546,6 +546,20 @@
</item> </item>
</layout> </layout>
</widget> </widget>
<widget class="QWidget" name="debugTab">
<attribute name="title">
<string>Debug</string>
</attribute>
<layout class="QVBoxLayout" name="verticalLayout_3">
<item>
<widget class="QPlainTextEdit" name="logText">
<property name="readOnly">
<bool>true</bool>
</property>
</widget>
</item>
</layout>
</widget>
</widget> </widget>
</item> </item>
</layout> </layout>
@ -593,6 +607,8 @@
<property name="title"> <property name="title">
<string>&amp;Help</string> <string>&amp;Help</string>
</property> </property>
<addaction name="actionDebug"/>
<addaction name="separator"/>
<addaction name="action_About"/> <addaction name="action_About"/>
</widget> </widget>
<addaction name="menu_File"/> <addaction name="menu_File"/>
@ -692,6 +708,14 @@
<string>&amp;Noon Date Split</string> <string>&amp;Noon Date Split</string>
</property> </property>
</action> </action>
<action name="actionDebug">
<property name="checkable">
<bool>true</bool>
</property>
<property name="text">
<string>Debug</string>
</property>
</action>
</widget> </widget>
<customwidgets> <customwidgets>
<customwidget> <customwidget>

View File

@ -140,8 +140,8 @@ void Oximetry::on_RunButton_toggled(bool checked)
plethy->SetMinX(double(lasttime)/86400000.0); plethy->SetMinX(double(lasttime)/86400000.0);
plethy->SetMaxX(double(lasttime+30000)/86400000.0); plethy->SetMaxX(double(lasttime+30000)/86400000.0);
plethy->SetRealMinY(0); plethy->SetRealMinY(0);
plethy->SetRealMaxY(120); plethy->SetRealMaxY(130);
plethy->SetMaxY(120); plethy->SetMaxY(130);
plethy->SetMinY(0); plethy->SetMinY(0);
PLETHY->MinX(); PLETHY->MinX();
PLETHY->MaxX(); PLETHY->MaxX();
@ -250,6 +250,64 @@ void Oximetry::on_SerialPortsCombo_activated(const QString &arg1)
{ {
portname=arg1; portname=arg1;
} }
void Oximetry::UpdatePlethy(qint8 d)
{
plethy->point[0][plethy->np[0]].setX(double(lasttime)/86400000.0);
plethy->point[0][plethy->np[0]++].setY(d);
lasttime+=20; // 50 samples per second?
plethy->SetRealMaxX(lasttime/86400000.0);
if (plethy->RealMaxX()-plethy->RealMinX()>(1.0/(24.0*120.0))) {
plethy->SetMinX(lasttime/86400000.0-(1.0/(24.0*120.0)));
plethy->SetMaxX(lasttime/86400000.0);
}
PLETHY->MinX();
PLETHY->MaxX();
PLETHY->RealMaxX();
if (plethy->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
plethy->np[0]=0;
lasttime=0;
}
//PLETHY->updateGL(); // Move this to a timer.
}
bool Oximetry::UpdatePulseSPO2(qint8 pul,qint8 sp)
{
bool ret=false;
if (lastpulse!=pul) {
pulse->point[0][pulse->np[0]].setX(double(lasttime)/86400000.0);
pulse->point[0][pulse->np[0]++].setY(pul);
PULSE->updateGL();
if (pulse->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
pulse->np[0]=0;
lasttime=0;
}
ret=true;
//qDebug() << "Pulse=" << int(bytes[0]);
}
if (lastspo2!=sp) {
spo2->point[0][spo2->np[0]].setX(double(lasttime)/86400000.0);
spo2->point[0][spo2->np[0]++].setY(sp);
SPO2->updateGL();
if (spo2->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
spo2->np[0]=0;
lasttime=0;
}
ret=true;
//qDebug() << "SpO2=" << int(bytes[1]);
}
lastpulse=pul;
lastspo2=sp;
return ret;
}
void Oximetry::onReadyRead() void Oximetry::onReadyRead()
{ {
@ -259,68 +317,35 @@ void Oximetry::onReadyRead()
bytes.resize(a); bytes.resize(a);
port->read(bytes.data(), bytes.size()); port->read(bytes.data(), bytes.size());
const int max_data_points=1000000; int i=0;
if (bytes.size()==3) { bool redraw_ps=false;
EventDataType d=bytes[1] & 0x7f; while (i<bytes.size()) {
plethy->point[0][plethy->np[0]].setX(double(lasttime)/86400000.0); if (bytes[i]&0x80) {
plethy->point[0][plethy->np[0]++].setY(d); EventDataType d=bytes[i+1] & 0x7f;
lasttime+=20; // 50 samples per second? UpdatePlethy(d);
plethy->SetRealMaxX(lasttime/86400000.0); i+=3;
if (plethy->RealMaxX()-plethy->RealMinX()>(1.0/(24.0*120.0))) { } else {
plethy->SetMinX(lasttime/86400000.0-(1.0/(24.0*120.0))); if (UpdatePulseSPO2(bytes[i], bytes[i+1])) redraw_ps=true;
plethy->SetMaxX(lasttime/86400000.0); i+=2;
} }
PLETHY->MinX();
PLETHY->MaxX();
PLETHY->RealMaxX();
if (plethy->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
plethy->np[0]=0;
lasttime=0;
}
PLETHY->updateGL(); // Move this to a timer.
} }
if (bytes.size()==2) { // Data bytes in live mode PLETHY->updateGL();
if (redraw_ps) {
PULSE->updateGL();
SPO2->updateGL();
}
/*if (bytes.size()==3) {
} else if (bytes.size()==2) { // Data bytes in live mode
// Plethy data // Plethy data
if (lastpulse!=bytes[0]) { } else {
pulse->point[0][pulse->np[0]].setX(double(lasttime)/86400000.0); //qDebug() << "Got " << bytes.size() << " bytes";
pulse->point[0][pulse->np[0]++].setY(bytes[0]); }*/
//pulse->SetMinX(lasttime/86400000.0-(1.0/(24.0*15.0))); QString aa=QString::number(bytes.size(),16)+"bytes: ";
//pulse->SetMaxX(lasttime/86400000.0); for (int i=0;i<bytes.size();i++) {
//pulse->SetRealMaxX(lasttime/86400000.0); aa+=" "+QString::number((unsigned char)bytes[i],16);
//PULSE->MinX();
//PULSE->MaxX();
//PULSE->RealMaxX();
PULSE->updateGL();
if (pulse->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
pulse->np[0]=0;
lasttime=0;
}
//qDebug() << "Pulse=" << int(bytes[0]);
}
if (lastspo2!=bytes[1]) {
spo2->point[0][spo2->np[0]].setX(double(lasttime)/86400000.0);
spo2->point[0][spo2->np[0]++].setY(bytes[1]);
SPO2->updateGL();
if (spo2->np[0]>max_data_points) {
//TODO: Stop Serial recording..
// for now overwrite..
spo2->np[0]=0;
lasttime=0;
}
//qDebug() << "SpO2=" << int(bytes[1]);
}
lastpulse=bytes[0];
lastspo2=bytes[1];
} }
qDebug() << aa;
lastsize=bytes.size(); lastsize=bytes.size();
} }
void Oximetry::onDsrChanged(bool status) void Oximetry::onDsrChanged(bool status)

View File

@ -14,6 +14,7 @@ namespace Ui {
} }
enum PORTMODE { PM_LIVE, PM_RECORDING }; enum PORTMODE { PM_LIVE, PM_RECORDING };
const int max_data_points=1000000;
class Oximetry : public QWidget class Oximetry : public QWidget
{ {
@ -23,8 +24,8 @@ public:
explicit Oximetry(QWidget *parent = 0); explicit Oximetry(QWidget *parent = 0);
~Oximetry(); ~Oximetry();
void AddData(gPointData *d) { Data.push_back(d); }; void AddData(gPointData *d) { Data.push_back(d); }
void AddGraph(gGraphWindow *w) { Graphs.push_back(w); }; void AddGraph(gGraphWindow *w) { Graphs.push_back(w); }
void RedrawGraphs(); void RedrawGraphs();
private slots: private slots:
@ -39,6 +40,10 @@ private slots:
void on_ImportButton_clicked(); void on_ImportButton_clicked();
private: private:
bool UpdatePulseSPO2(qint8 pulse,qint8 spo2);
void UpdatePlethy(qint8 plethy);
Ui::Oximetry *ui; Ui::Oximetry *ui;
Profile *profile; Profile *profile;
QSplitter *gSplitter; QSplitter *gSplitter;