mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
Merge branch 'master' into vAuto-Settings
This commit is contained in:
commit
a5f3caa125
@ -17,9 +17,12 @@
|
||||
<ul>
|
||||
<li>[new] Additional Philips Respironics devices tested and fully supported:
|
||||
<ul>
|
||||
<li>DreamStation 2 CPAP (410X150C)</li>
|
||||
<li>BiPAP Auto (System One 60 Series) (761P)</li>
|
||||
<li>BiPAP autoSV Advanced 30 (System One 60 Series) (961TCA)</li>
|
||||
</ul>
|
||||
</li>
|
||||
<li>[fix] Added support for pressure pulse, CA, and VS on BiPAP autoSV Advanced 30 (System One 60 Series) (960T).</li>
|
||||
</ul>
|
||||
<p>
|
||||
<b>Changes and fixes in OSCAR v1.3.5-alpha.2</b>
|
||||
|
@ -143,34 +143,10 @@ some messages from Apnea Board.
|
||||
//#define MAP_LOG_EVENTS
|
||||
//#define ENABLE_UNEVEN_MACHINE_MIN_MAX_TEST
|
||||
|
||||
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
|
||||
/ <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
|
||||
// Define Display macros to enhance displays
|
||||
|
||||
|
||||
#define DEBUGQ qDebug()
|
||||
#define DEBUGT qDebug()<<QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz") << QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__
|
||||
#define DEBUGF qDebug()<<QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
#define DEBUGTF qDebug()<<QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz") << QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
|
||||
#include <QRegularExpression>
|
||||
#define DEBUG qDebug()<<QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__
|
||||
#define TIMEO( DD , XX ) " " #DD ":" << QDateTime::fromMSecsSinceEpoch( XX ).time().toString("hh:mm:ss.zzz")
|
||||
#define TIME( XX ) QDateTime::fromMSecsSinceEpoch( XX ).time().toString("hh:mm:ss.zzz")
|
||||
#define O( XX ) " " #XX ":" << XX
|
||||
#define OO( XX , YY ) " " #XX ":" << YY
|
||||
#define FULLNAME( id) schema::channel[ id ].fullname()
|
||||
#define PLACE (QString(basename( __FILE__)).remove(QRegularExpression("\\..*$"))) << __LINE__
|
||||
|
||||
#define DATETIME( XX ) QDateTime::fromMSecsSinceEpoch( XX ).toString("dd MMM yyyy hh:mm:ss.zzz")
|
||||
#define DATE( XX ) QDateTime::fromMSecsSinceEpoch( XX ).toString("dd MMM yyyy")
|
||||
#define DATEO( XX ) " " #XX ":" << QDateTime::fromMSecsSinceEpoch( XX ).toString("dd MMM yyyy")
|
||||
#define TIME2( XX ) " " #XX ":" << XX << QDateTime::fromMSecsSinceEpoch( XX ).time().toString("hh:mm:ss.zzz")
|
||||
#define NAME( id) schema::channel[ id ].label()
|
||||
|
||||
//#define ENABLE_TEST_NODATA // no valid samples.
|
||||
//#define ENABLE_TEST_SINGLE // a single value.
|
||||
//#define ENABLE_TEST_CPAP // change all values to a single value.
|
||||
//#define ENABLE_TEST_SAWTOOTH // change all sample time and value to a sawtooth
|
||||
#endif
|
||||
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
|
||||
|
||||
@ -180,6 +156,9 @@ some messages from Apnea Board.
|
||||
#define test_data(A,B,C,D,E,F,G,H)
|
||||
#endif
|
||||
|
||||
#define TEST_MACROS_ENABLEDoff
|
||||
#include "test_macros.h"
|
||||
|
||||
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
|
||||
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
|
||||
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
|
||||
@ -553,7 +532,6 @@ void RecalcMAP::updateTimesValues(qint64 d1,qint64 d2, int key,PressureInfo & in
|
||||
|
||||
//! \brief Updates Time At Pressure from session *sess
|
||||
void RecalcMAP::updateTimes(PressureInfo & info) {
|
||||
//DEBUGF <<DATETIME(sessStartTime) << "CALLED" <<TIME(sessStartTime) <<TIME(sessEndTime);
|
||||
qint64 d1,d2;
|
||||
qint64 minx=0,maxx=0;
|
||||
//qint64 prevSessDuration=info.totalDuration;
|
||||
@ -1080,7 +1058,6 @@ void MinutesAtPressure::setEnabled(gGraph &graph) {
|
||||
} else {
|
||||
value &= pressureGraphLC->m_flags_enabled[ch];
|
||||
}
|
||||
//DEBUGF << FULLNAME(ch) << O(value);
|
||||
m_enabled[ch]=value;
|
||||
}
|
||||
};
|
||||
|
@ -7,6 +7,9 @@
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
#define TEST_MACROS_ENABLEDoff
|
||||
#include "test_macros.h"
|
||||
|
||||
#include "Graphs/gGraph.h"
|
||||
|
||||
#include <QLabel>
|
||||
@ -131,6 +134,7 @@ gGraph::gGraph(QString name, gGraphView *graphview, QString title, QString units
|
||||
m_units(units),
|
||||
m_visible(true)
|
||||
{
|
||||
// DEBUGF Q(name) Q(title) QQ("UNITS",units) Q(height) Q(group);
|
||||
if (height == 0) {
|
||||
height = AppSetting->graphHeight();
|
||||
Q_UNUSED(height)
|
||||
@ -153,7 +157,7 @@ gGraph::gGraph(QString name, gGraphView *graphview, QString title, QString units
|
||||
min_x = min_y = 0;
|
||||
rec_miny = rec_maxy = 0;
|
||||
rphysmax_y = rphysmin_y = 0;
|
||||
m_zoomY = 0;
|
||||
m_zoomY = ZS_AUTO_FIT;
|
||||
m_selectedDuration = 0;
|
||||
|
||||
if (graphview) {
|
||||
@ -281,9 +285,23 @@ void gGraph::setDay(Day *day)
|
||||
// ResetBounds();
|
||||
}
|
||||
|
||||
void gGraph::setZoomY(short zoom)
|
||||
{
|
||||
m_zoomY = zoom;
|
||||
void gGraph::setZoomY(ZoomyScaling zoomy) {
|
||||
m_zoomY = zoomy;
|
||||
dynamicScalingOn =false;
|
||||
timedRedraw(0);
|
||||
}
|
||||
|
||||
void gGraph::mouseDoubleClickYAxis(QMouseEvent * ) {
|
||||
if ( isDynamicScalingEnabled() ) {
|
||||
dynamicScalingOn = !dynamicScalingOn;
|
||||
} else {
|
||||
dynamicScalingOn =false;
|
||||
if (m_zoomY == ZS_AUTO_FIT ) {
|
||||
m_zoomY = ZS_DEFAULT;
|
||||
} else if (m_zoomY == ZS_DEFAULT) {
|
||||
m_zoomY = ZS_AUTO_FIT ;
|
||||
}
|
||||
}
|
||||
timedRedraw(0);
|
||||
}
|
||||
|
||||
@ -564,22 +582,67 @@ void gGraph::ToolTip(QString text, int x, int y, ToolTipAlignment align, int tim
|
||||
m_graphview->m_tooltip->display(text, x, y, align, timeout);
|
||||
}
|
||||
|
||||
bool gGraph::isDynamicScalingEnabled() {
|
||||
return ((m_lineChart_layer!=nullptr) && AppSetting->allowYAxisScaling() );
|
||||
}
|
||||
|
||||
QString gGraph::unitsTooltip() {
|
||||
if (isDynamicScalingEnabled()) {
|
||||
if(dynamicScalingOn) {
|
||||
if (zoomY() == ZS_AUTO_FIT ) {
|
||||
return QString("%1%2%3").arg(m_units).arg("\n").arg(tr("Double click Y-axis: Return to AUTO-FIT Scaling"));
|
||||
} else if (zoomY() == ZS_DEFAULT ) {
|
||||
return QString("%1%2%3").arg(m_units).arg("\n").arg(tr("Double click Y-axis: Return to DEFAULT Scaling"));
|
||||
} else {
|
||||
return QString("%1%2%3").arg(m_units).arg("\n").arg(tr("Double click Y-axis: Return to OVERRIDE Scaling"));
|
||||
}
|
||||
} else {
|
||||
return QString("%1%2%3").arg(m_units).arg("\n").arg(tr("Double click Y-axis: For Dynamic Scaling"));
|
||||
}
|
||||
} else {
|
||||
if (zoomY() == ZS_AUTO_FIT ) {
|
||||
return QString("%1%2%3").arg(m_units).arg("\n").arg(tr("Double click Y-axis: Select DEFAULT Scaling"));
|
||||
} else if (zoomY() == ZS_DEFAULT ) {
|
||||
return QString("%1%2%3").arg(m_units).arg("\n").arg(tr("Double click Y-axis: Select AUTO-FIT Scaling"));
|
||||
}
|
||||
}
|
||||
return m_units;
|
||||
}
|
||||
|
||||
void gGraph::dynamicScaling(EventDataType &miny, EventDataType &maxy) {
|
||||
// Have new Dynamic mode;
|
||||
miny = m_lineChart_layer->actualMinY();
|
||||
maxy = m_lineChart_layer->actualMaxY();
|
||||
EventDataType diff= (maxy-miny);
|
||||
maxy += diff*0.08; // more space at top for event ticks.
|
||||
miny -= diff*0.04;
|
||||
if (m_saved_minY!=m_lineChart_layer->actualMinY() || m_saved_maxY!=m_lineChart_layer->actualMaxY() ) {
|
||||
// DEBUGF O(m_name) Q(m_saved_minY) QQ("==>",m_lineChart_layer->actualMinY() ) QQ("==>",miny) Q(m_saved_maxY) QQ("==>",m_lineChart_layer->actualMaxY() ) QQ("==>",maxy);
|
||||
m_saved_minY=m_lineChart_layer->actualMinY();
|
||||
m_saved_maxY=m_lineChart_layer->actualMaxY();
|
||||
}
|
||||
}
|
||||
|
||||
// YAxis Autoscaling code
|
||||
void gGraph::roundY(EventDataType &miny, EventDataType &maxy)
|
||||
{
|
||||
|
||||
if (zoomY() == 2) {
|
||||
if (dynamicScalingOn) {
|
||||
dynamicScaling(miny, maxy) ;
|
||||
if (maxy > miny) return;
|
||||
};
|
||||
if (zoomY() == ZS_OVERRIDE) { // Have override mode
|
||||
// set min and max to override values.
|
||||
miny = rec_miny;
|
||||
maxy = rec_maxy;
|
||||
if (maxy > miny) return;
|
||||
} else if (zoomY() ==1) {
|
||||
if (maxy > miny) return; // Not Autoscaling
|
||||
} else if (zoomY() ==ZS_DEFAULT) { // Have Default mode
|
||||
// set min and max to physical Min / max values.
|
||||
miny = physMinY();
|
||||
maxy = physMaxY();
|
||||
if (maxy > miny) return;
|
||||
if (maxy > miny) return; // Not Autoscaling
|
||||
}
|
||||
miny = MinY();
|
||||
maxy = MaxY();
|
||||
|
||||
int m, t;
|
||||
bool ymin_good = false, ymax_good = false;
|
||||
|
||||
@ -685,6 +748,11 @@ void gGraph::AddLayer(Layer *l, LayerPosition position, short width, short heigh
|
||||
l->setPos(x, y);
|
||||
l->addref();
|
||||
m_layers.push_back(l);
|
||||
|
||||
if (l->layerType()==LT_LineChart) {
|
||||
m_lineChart_layer = dynamic_cast<gLineChart *>(l);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void gGraph::dataChanged()
|
||||
@ -1111,24 +1179,6 @@ void gGraph::mouseDoubleClickEvent(QMouseEvent *event)
|
||||
layer->mouseDoubleClickEvent(event, this);
|
||||
}
|
||||
}
|
||||
|
||||
//int w=m_lastbounds.width()-(m_marginleft+left+right+m_marginright);
|
||||
//int h=m_lastbounds.height()-(bottom+m_marginbottom);
|
||||
//int x2=m_graphview->pointClicked().x(),y2=m_graphview->pointClicked().y();
|
||||
// if ((m_graphview->horizTravel()<mouse_movement_threshold) && (x>left+m_marginleft && x<w+m_marginleft+left && y>top+m_margintop && y<h)) { // normal click in main area
|
||||
// if (event->button() & Qt::RightButton) {
|
||||
// ZoomX(1.66,x); // Zoon out
|
||||
// return;
|
||||
// } else if (event->button() & Qt::LeftButton) {
|
||||
// ZoomX(0.75/2.0,x); // zoom in.
|
||||
// return;
|
||||
// }
|
||||
// } else {
|
||||
// Propagate the events to graph Layers
|
||||
// }
|
||||
//mousePressEvent(event);
|
||||
//mouseReleaseEvent(event);
|
||||
//qDebug() << m_title << "Double Clicked" << event->x() << event->y();
|
||||
}
|
||||
void gGraph::keyPressEvent(QKeyEvent *event)
|
||||
{
|
||||
@ -1416,16 +1466,14 @@ void gGraph::SetMaxY(EventDataType v)
|
||||
|
||||
Layer *gGraph::getLineChart()
|
||||
{
|
||||
gLineChart *lc;
|
||||
|
||||
if (m_lineChart_layer) return m_lineChart_layer;
|
||||
for (auto & layer : m_layers) {
|
||||
lc = dynamic_cast<gLineChart *>(layer);
|
||||
|
||||
if (lc) { return lc; }
|
||||
Layer *tmp = dynamic_cast<gLineChart *>(layer);
|
||||
if (tmp) m_lineChart_layer = tmp;
|
||||
}
|
||||
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
int gGraph::minHeight()
|
||||
{
|
||||
int minheight = m_min_height;
|
||||
|
@ -35,6 +35,8 @@ void DestroyGraphGlobals();
|
||||
|
||||
const int mouse_movement_threshold = 6;
|
||||
|
||||
enum ZoomyScaling {ZS_AUTO_FIT =0 , ZS_DEFAULT =1 , ZS_OVERRIDE =2};
|
||||
|
||||
float CatmullRomSpline(float p0, float p1, float p2, float p3, float t = 0.5);
|
||||
|
||||
inline void GetTextExtent(QString text, int &width, int &height, QFont *font)
|
||||
@ -164,6 +166,9 @@ class gGraph : public QObject
|
||||
//! \brief Returns the measurement Units the Y scale is referring to
|
||||
QString units() { return m_units; }
|
||||
|
||||
//! \brief Returns the measurement Units the Y scale is referring to plus tooltip.
|
||||
QString unitsTooltip() ;
|
||||
|
||||
//! \brief Sets the measurement Units the Y scale is referring to
|
||||
void setUnits(const QString units) { m_units = units; }
|
||||
|
||||
@ -303,9 +308,14 @@ class gGraph : public QObject
|
||||
bool isSnapshot() { return m_snapshot; }
|
||||
void setSnapshot(bool b) { m_snapshot = b; }
|
||||
|
||||
// returns if graph is a daily line chart with Dynamic Scaling mode enabled.
|
||||
bool isDynamicScalingEnabled();
|
||||
|
||||
short left, right, top, bottom; // dirty magin hacks..
|
||||
|
||||
Layer *getLineChart();
|
||||
Layer *m_lineChart_layer =nullptr ;
|
||||
bool dynamicScalingOn =false;
|
||||
QTimer *timer;
|
||||
|
||||
// This gets set to true to force a redraw of the yAxis tickers when graphs are resized.
|
||||
@ -318,10 +328,8 @@ class gGraph : public QObject
|
||||
gGraphView *graphView() { return m_graphview; }
|
||||
short m_marginleft, m_marginright, m_margintop, m_marginbottom;
|
||||
|
||||
inline short zoomY() { return m_zoomY; }
|
||||
void setZoomY(short zoom);
|
||||
|
||||
static const short maxZoomY = 2;
|
||||
inline ZoomyScaling zoomY() { return m_zoomY; }
|
||||
void setZoomY(ZoomyScaling zoomy);
|
||||
|
||||
inline qint64 selectedDuration() const { return m_selectedDuration; }
|
||||
inline QString selDurString() const { return m_selDurString; }
|
||||
@ -331,6 +339,8 @@ class gGraph : public QObject
|
||||
|
||||
inline bool printing() const { return m_printing; }
|
||||
|
||||
void mouseDoubleClickYAxis(QMouseEvent *event);
|
||||
|
||||
protected:
|
||||
//! \brief Mouse Wheel events
|
||||
virtual void wheelEvent(QWheelEvent *event);
|
||||
@ -353,6 +363,8 @@ class gGraph : public QObject
|
||||
//! \brief Key Pressed event
|
||||
virtual void keyReleaseEvent(QKeyEvent *event);
|
||||
|
||||
void dynamicScaling(EventDataType &miny, EventDataType &maxy);
|
||||
|
||||
void cancelSelection() {
|
||||
m_selecting_area = false;
|
||||
m_selection = QRect(0,0,0,0);
|
||||
@ -389,7 +401,7 @@ class gGraph : public QObject
|
||||
bool m_showTitle;
|
||||
bool m_printing;
|
||||
bool m_pinned;
|
||||
short m_zoomY;
|
||||
ZoomyScaling m_zoomY;
|
||||
bool m_block_select;
|
||||
QRect m_rect;
|
||||
|
||||
@ -401,6 +413,8 @@ class gGraph : public QObject
|
||||
QString m_selDurString;
|
||||
|
||||
bool m_snapshot;
|
||||
EventDataType m_saved_minY=0;
|
||||
EventDataType m_saved_maxY=0;
|
||||
|
||||
protected slots:
|
||||
//! \brief Deselects any highlights, and schedules a main gGraphView redraw
|
||||
|
@ -7,19 +7,9 @@
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
#define xDEBUG_FUNCTIONS
|
||||
#ifdef DEBUG_FUNCTIONS
|
||||
#include <QRegularExpression>
|
||||
#define DEBUG qDebug()<<QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__
|
||||
#define DEBUGF qDebug()<<QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
#define DEBUGTF qDebug()<<QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz") << QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
#define TEST_MACROS_ENABLEDoff
|
||||
#include "test_macros.h"
|
||||
|
||||
#define O( XX ) " " #XX ":" << XX
|
||||
#define OO( XX , YY ) " " #XX ":" << YY
|
||||
#define NAME( id) schema::channel[ id ].label()
|
||||
#define DATE( XX ) QDateTime::fromMSecsSinceEpoch(XX).toString("dd MMM yyyy")
|
||||
#define DATETIME( XX ) QDateTime::fromMSecsSinceEpoch(XX).toString("dd MMM yyyy hh:mm:ss.zzz")
|
||||
#endif
|
||||
|
||||
#include "Graphs/gGraphView.h"
|
||||
|
||||
@ -139,10 +129,8 @@ void gToolTip::display(QString text, int x, int y, ToolTipAlignment align, int t
|
||||
if (timer->isActive()) {
|
||||
timer->stop();
|
||||
}
|
||||
|
||||
timer->setSingleShot(true);
|
||||
timer->start(timeout);
|
||||
m_invalidate = true;
|
||||
}
|
||||
|
||||
void gToolTip::cancel()
|
||||
@ -151,17 +139,14 @@ void gToolTip::cancel()
|
||||
timer->stop();
|
||||
}
|
||||
|
||||
void gToolTip::paint(QPainter &painter) //actually paints it.
|
||||
QRect gToolTip::calculateRect(QPainter &painter)
|
||||
{
|
||||
if (!m_visible) { return; }
|
||||
|
||||
int x = m_pos.x();
|
||||
int y = m_pos.y();
|
||||
|
||||
QRect rect(x, y, 0, 0);
|
||||
|
||||
painter.setFont(*defaultfont);
|
||||
|
||||
painter.setFont(*m_font);
|
||||
rect = painter.boundingRect(rect, Qt::AlignCenter, m_text);
|
||||
|
||||
int w = rect.width() + m_spacer * 2;
|
||||
@ -201,19 +186,25 @@ void gToolTip::paint(QPainter &painter) //actually paints it.
|
||||
rect.setTop(rect.top()-bot);
|
||||
rect.setBottom(m_graphview->height());
|
||||
}
|
||||
return rect;
|
||||
}
|
||||
|
||||
void gToolTip::paint(QPainter &painter) //actually paints it.
|
||||
{
|
||||
if (!m_visible) { return; }
|
||||
QRect a_rect=calculateRect(painter);
|
||||
|
||||
QBrush brush(QColor(255, 255, 128, 230));
|
||||
brush.setStyle(Qt::SolidPattern);
|
||||
painter.setBrush(brush);
|
||||
painter.setPen(QColor(0, 0, 0, 255));
|
||||
|
||||
painter.drawRoundedRect(rect, 5, 5);
|
||||
painter.drawRoundedRect(a_rect, 5, 5);
|
||||
painter.setBrush(Qt::black);
|
||||
|
||||
painter.setFont(*defaultfont);
|
||||
painter.setFont(*m_font);
|
||||
|
||||
painter.drawText(rect, Qt::AlignCenter, m_text);
|
||||
painter.drawText(a_rect, Qt::AlignCenter, m_text);
|
||||
}
|
||||
|
||||
void gToolTip::timerDone()
|
||||
@ -223,6 +214,88 @@ void gToolTip::timerDone()
|
||||
m_graphview->resetMouse();
|
||||
}
|
||||
|
||||
/* Parent tool tip
|
||||
Allow the parent (overview or daily) to add tooltip or short messages to the user.
|
||||
The basic problem is that the parent does not know the current dimensions of the graph view.
|
||||
the parent does have knowledge of the location of fixed widgets which makes it possible to
|
||||
locate tool tips in an appropiate location.
|
||||
*/
|
||||
gParentToolTip::gParentToolTip(gGraphView *graphview)
|
||||
: gToolTip(graphview) {
|
||||
m_parent_visible=false;
|
||||
}
|
||||
|
||||
gParentToolTip::~gParentToolTip() {
|
||||
}
|
||||
|
||||
void gParentToolTip::display(gGraphView* gv,QString text, int verticalOffset, int alignOffset, ToolTipAlignment align , int timeout ,QFont *font ) {
|
||||
m_text=text;
|
||||
m_verticalOffset=verticalOffset;
|
||||
m_alignOffset=alignOffset;
|
||||
m_alignment=align;
|
||||
m_timeout=timeout;
|
||||
m_font=font;
|
||||
m_parent_visible=true;
|
||||
gv->timedRedraw(0);
|
||||
};
|
||||
|
||||
|
||||
QRect gParentToolTip::calculateRect(QPainter& painter ) {
|
||||
QRect rect(0, 0, 0, 0);
|
||||
painter.setFont(*m_font);
|
||||
rect = painter.boundingRect(rect, m_alignment, m_text);
|
||||
|
||||
// update space arround text
|
||||
int space=2*m_spacer;
|
||||
rect.setHeight(space+rect.height());
|
||||
rect.setWidth(space+rect.width());
|
||||
|
||||
rect.moveTo(m_alignOffset,m_height-(m_verticalOffset+rect.height()));
|
||||
|
||||
// move rect accounding to alignment. default is left.
|
||||
|
||||
if (m_alignment == TT_AlignRight) {
|
||||
// move rect left by width of rect. if <0 use 0;
|
||||
rect.moveLeft(rect.left()-rect.width());
|
||||
} else if (m_alignment == TT_AlignCenter) {
|
||||
// left by 1/2 width of rect. if < 0 then use 0
|
||||
rect.moveLeft(rect.left()-rect.width()/2);
|
||||
}
|
||||
|
||||
if (rect.top()<0) {rect.setTop(0);};
|
||||
if (rect.left()<0) {rect.setLeft(0);};
|
||||
|
||||
return rect;
|
||||
}
|
||||
|
||||
void gParentToolTip::paint(QPainter &painter,int width,int height) {
|
||||
if (!m_parent_visible) {return ;};
|
||||
m_width=width;
|
||||
m_height=height;
|
||||
gToolTip::display(m_text, 0, 0,m_alignment, m_timeout);
|
||||
gToolTip::paint(painter);
|
||||
};
|
||||
|
||||
void gParentToolTip::timerDone() {
|
||||
gToolTip::timerDone();
|
||||
if (m_parent_visible) {
|
||||
m_graphview->timedRedraw(0);
|
||||
}
|
||||
m_parent_visible=false;
|
||||
};
|
||||
|
||||
void gParentToolTip::cancel() {
|
||||
gToolTip::cancel();
|
||||
m_parent_visible=false;
|
||||
};
|
||||
|
||||
bool gParentToolTip::visible() {
|
||||
return gToolTip::visible() && m_parent_visible;
|
||||
};
|
||||
|
||||
|
||||
|
||||
|
||||
#ifdef ENABLE_THREADED_DRAWING
|
||||
|
||||
gThread::gThread(gGraphView *g)
|
||||
@ -359,6 +432,7 @@ gGraphView::gGraphView(QWidget *parent, gGraphView *shared, QWidget *caller)
|
||||
masterlock = new QSemaphore(m_idealthreads);
|
||||
#endif
|
||||
m_tooltip = new gToolTip(this);
|
||||
m_parent_tooltip = new gParentToolTip(this);
|
||||
/*for (int i=0;i<m_idealthreads;i++) {
|
||||
gThread * gt=new gThread(this);
|
||||
m_threads.push_back(gt);
|
||||
@ -474,7 +548,6 @@ void gGraphView::popoutGraph()
|
||||
dock->resize(width(),0);
|
||||
// QScrollArea
|
||||
}
|
||||
|
||||
//////// Create dock widget and resize dock to hold new widget
|
||||
QDockWidget * newDockWidget = new QDockWidget(dock);
|
||||
newDockWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
|
||||
@ -634,6 +707,7 @@ gGraphView::~gGraphView()
|
||||
}
|
||||
|
||||
delete m_tooltip;
|
||||
delete m_parent_tooltip;
|
||||
m_graphs.clear();
|
||||
}
|
||||
|
||||
@ -1499,6 +1573,7 @@ void gGraphView::paintGL()
|
||||
AppSetting->usePixmapCaching() ? DrawTextQueCached(painter) :DrawTextQue(painter);
|
||||
|
||||
m_tooltip->paint(painter);
|
||||
m_parent_tooltip->paint(painter,width(), height() );
|
||||
|
||||
#ifdef DEBUG_EFFICIENCY
|
||||
const int rs = 20;
|
||||
@ -1973,8 +2048,8 @@ void MinMaxWidget::onMaxChanged(double d)
|
||||
}
|
||||
void MinMaxWidget::onResetClicked()
|
||||
{
|
||||
int tmp = graph->zoomY();
|
||||
graph->setZoomY(0);
|
||||
ZoomyScaling tmp = graph->zoomY();
|
||||
graph->setZoomY(ZS_AUTO_FIT);
|
||||
EventDataType miny = graph->MinY(),
|
||||
maxy = graph->MaxY();
|
||||
|
||||
@ -1996,15 +2071,16 @@ void MinMaxWidget::onResetClicked()
|
||||
graph->setZoomY(tmp);
|
||||
}
|
||||
|
||||
void MinMaxWidget::onComboChanged(int idx)
|
||||
void MinMaxWidget::onComboChanged(int _idx)
|
||||
{
|
||||
minbox->setEnabled(idx == 2);
|
||||
maxbox->setEnabled(idx == 2);
|
||||
reset->setEnabled(idx == 2);
|
||||
ZoomyScaling idx = static_cast<ZoomyScaling>(_idx) ;
|
||||
minbox->setEnabled(idx == ZS_OVERRIDE);
|
||||
maxbox->setEnabled(idx == ZS_OVERRIDE);
|
||||
reset->setEnabled(idx == ZS_OVERRIDE);
|
||||
|
||||
graph->setZoomY(idx);
|
||||
|
||||
if (idx == 2) {
|
||||
if (idx == ZS_OVERRIDE) {
|
||||
if (qAbs(graph->rec_maxy - graph->rec_miny) < 0.0001) {
|
||||
onResetClicked();
|
||||
}
|
||||
@ -2018,9 +2094,9 @@ void MinMaxWidget::createLayout()
|
||||
layout->setSpacing(4);
|
||||
|
||||
combobox = new QComboBox(this);
|
||||
combobox->addItem(tr("Auto-Fit"), 0);
|
||||
combobox->addItem(tr("Defaults"), 1);
|
||||
combobox->addItem(tr("Override"), 2);
|
||||
combobox->addItem(tr("Auto-Fit"), ZS_AUTO_FIT);
|
||||
combobox->addItem(tr("Defaults"), ZS_DEFAULT);
|
||||
combobox->addItem(tr("Override"), ZS_OVERRIDE);
|
||||
combobox->setToolTip(tr("The Y-Axis scaling mode, 'Auto-Fit' for automatic scaling, 'Defaults' for settings according to manufacturer, and 'Override' to choose your own."));
|
||||
connect(combobox, SIGNAL(activated(int)), this, SLOT(onComboChanged(int)));
|
||||
|
||||
@ -3445,7 +3521,7 @@ void gGraphView::SaveSettings(QString title)
|
||||
out << graph->visible();
|
||||
out << graph->RecMinY();
|
||||
out << graph->RecMaxY();
|
||||
out << graph->zoomY();
|
||||
out << (short)graph->zoomY(); // the return type of zoomY was changed from a short to an enum (int) so much type cast it here
|
||||
out << (bool)graph->isPinned();
|
||||
|
||||
gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(graph, LT_LineChart));
|
||||
@ -3575,7 +3651,7 @@ bool gGraphView::LoadSettings(QString title)
|
||||
g->setVisible(vis);
|
||||
g->setRecMinY(recminy);
|
||||
g->setRecMaxY(recmaxy);
|
||||
g->setZoomY(zoomy);
|
||||
g->setZoomY(static_cast<ZoomyScaling>(zoomy));
|
||||
g->setPinned(pinned);
|
||||
|
||||
if (gvversion >= 4) {
|
||||
|
@ -240,10 +240,12 @@ class gToolTip : public QObject
|
||||
virtual void paint(QPainter &paint); //actually paints it.
|
||||
|
||||
//! \brief Close the tooltip early.
|
||||
void cancel();
|
||||
virtual void cancel();
|
||||
|
||||
//! \brief Returns true if the tooltip is currently visible
|
||||
bool visible() { return m_visible; }
|
||||
virtual bool visible() { return m_visible; }
|
||||
|
||||
virtual QRect calculateRect(QPainter &painter);
|
||||
|
||||
protected:
|
||||
gGraphView *m_graphview;
|
||||
@ -254,13 +256,37 @@ class gToolTip : public QObject
|
||||
bool m_visible;
|
||||
int m_spacer;
|
||||
QImage m_image;
|
||||
bool m_invalidate;
|
||||
ToolTipAlignment m_alignment;
|
||||
QFont* m_font=defaultfont;
|
||||
|
||||
protected slots:
|
||||
|
||||
//! \brief Timeout to hide tooltip, and redraw without it.
|
||||
void timerDone();
|
||||
virtual void timerDone();
|
||||
};
|
||||
|
||||
class gParentToolTip : public gToolTip
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
gParentToolTip(gGraphView *graphview);
|
||||
~gParentToolTip();
|
||||
using gToolTip::display;
|
||||
virtual void display(gGraphView* gv,QString text,int verticalOffset, int alignOffset , ToolTipAlignment align = TT_AlignCenter, int timeout = 0,QFont *font = defaultfont) ;
|
||||
virtual void cancel();
|
||||
virtual bool visible();
|
||||
virtual QRect calculateRect(QPainter &painter);
|
||||
using gToolTip::paint;
|
||||
virtual void paint(QPainter &paint,int width,int height) ;
|
||||
private:
|
||||
int m_verticalOffset;
|
||||
int m_alignOffset;
|
||||
int m_width;
|
||||
int m_height;
|
||||
bool m_parent_visible;
|
||||
int m_timeout;
|
||||
protected slots:
|
||||
virtual void timerDone();
|
||||
};
|
||||
|
||||
struct SelectionHistoryItem {
|
||||
@ -425,6 +451,7 @@ class gGraphView
|
||||
|
||||
gGraph *m_selected_graph;
|
||||
gToolTip *m_tooltip;
|
||||
gParentToolTip *m_parent_tooltip;
|
||||
QTimer *timer;
|
||||
|
||||
//! \brief Add the Text information to the Text Drawing Queue (called by gGraphs renderText method)
|
||||
|
@ -7,6 +7,10 @@
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
|
||||
#define TEST_MACROS_ENABLEDoff
|
||||
#include "test_macros.h"
|
||||
|
||||
#include "Graphs/gLineChart.h"
|
||||
|
||||
#include <QString>
|
||||
@ -275,6 +279,7 @@ skipcheck:
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
EventDataType gLineChart::Miny()
|
||||
{
|
||||
if (m_codes.size() == 0) return 0;
|
||||
@ -373,6 +378,7 @@ QString gLineChart::getMetaString(qint64 time)
|
||||
// Time Domain Line Chart
|
||||
void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
{
|
||||
EventDataType actual_min_y, actual_max_y;
|
||||
QRectF rect = region.boundingRect();
|
||||
rect.translate(0.0f, 0.001f);
|
||||
// TODO: Just use QRect directly.
|
||||
@ -395,6 +401,8 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
return;
|
||||
}
|
||||
|
||||
actual_min_y = INT_MAX;
|
||||
actual_max_y = -INT_MAX;
|
||||
|
||||
top++;
|
||||
|
||||
@ -551,6 +559,7 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
painter.setPen(QPen(QBrush(color), lineThickness, Qt::DotLine));
|
||||
EventDataType y=top + height + 1 - ((dot.value - miny) * ymult);
|
||||
painter.drawLine(left + 1, y, left + 1 + width, y);
|
||||
DEBUGF NAME(dot.code) Q(dot.type) QQ(y,(int)y) Q(ratioX) O(QLine(left + 1, y, left + 1 + width, y)) Q(legendx) O(dot.value) ;
|
||||
|
||||
}
|
||||
}
|
||||
@ -742,6 +751,8 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
time += rate;
|
||||
// This is much faster than QVector access.
|
||||
data = *ptr * gain;
|
||||
if (actual_min_y>data) { actual_min_y=data; }
|
||||
if (actual_max_y<data) { actual_max_y=data; }
|
||||
|
||||
// Scale the time scale X to pixel scale X
|
||||
px = ((time - minx) * xmult);
|
||||
@ -822,6 +833,8 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
time += rate;
|
||||
|
||||
data = (*ptr + el.offset()) * gain;
|
||||
if (actual_min_y>data) { actual_min_y=data; }
|
||||
if (actual_max_y<data) { actual_max_y=data; }
|
||||
|
||||
px = xst + ((time - minx) * xmult); // Scale the time scale X to pixel scale X
|
||||
py = yst - ((data - miny) * ymult); // Same for Y scale, with precomputed gain
|
||||
@ -880,6 +893,8 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
|
||||
time = start + *tptr++;
|
||||
data = (*dptr++ + el.offset()) * gain;
|
||||
if (actual_min_y>data) { actual_min_y=data; }
|
||||
if (actual_max_y<data) { actual_max_y=data; }
|
||||
|
||||
idx++;
|
||||
|
||||
@ -897,10 +912,11 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
// Unrolling square plot outside of loop to gain a minor speed improvement.
|
||||
EventStoreType *eptr = dptr + siz;
|
||||
|
||||
if (square_plot) {
|
||||
for (; dptr < eptr; dptr++) {
|
||||
time = start + *tptr++;
|
||||
data = gain * (*dptr + el.offset());
|
||||
if (actual_min_y>data) { actual_min_y=data; }
|
||||
if (actual_max_y<data) { actual_max_y=data; }
|
||||
|
||||
px = xst + ((time - minx) * xmult); // Scale the time scale X to pixel scale X
|
||||
py = yst - ((data - miny) * ymult); // Same for Y scale without precomputed gain
|
||||
@ -912,50 +928,18 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
|
||||
// Cap px to right margin
|
||||
if (px > xst + width) { px = xst + width; }
|
||||
|
||||
// lines.append(QLine(lastpx, lastpy, px, lastpy));
|
||||
// lines.append(QLine(px, lastpy, px, py));
|
||||
} // else {
|
||||
}
|
||||
//else {
|
||||
// Letting the scissor do the dirty work for non horizontal lines
|
||||
// This really should be changed, as it might be cause that weird
|
||||
// display glitch on Linux..
|
||||
|
||||
//}
|
||||
if (square_plot) {
|
||||
lines.append(QLine(lastpx, lastpy, px, lastpy));
|
||||
lines.append(QLine(px, lastpy, px, py));
|
||||
// }
|
||||
|
||||
lastpx = px;
|
||||
lastpy = py;
|
||||
|
||||
if (time > maxx) {
|
||||
done = true; // Let this iteration finish.. (This point will be in far clipping)
|
||||
break;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
for (; dptr < eptr; dptr++) {
|
||||
//for (int i=0;i<siz;i++) {
|
||||
time = start + *tptr++;
|
||||
data = gain * (*dptr + el.offset());
|
||||
|
||||
px = xst + ((time - minx) * xmult); // Scale the time scale X to pixel scale X
|
||||
py = yst - ((data - miny) * ymult); // Same for Y scale without precomputed gain
|
||||
|
||||
// Horizontal lines are easy to cap
|
||||
if (py == lastpy) {
|
||||
// Cap px to left margin
|
||||
if (lastpx < xst) { lastpx = xst; }
|
||||
|
||||
// Cap px to right margin
|
||||
if (px > xst + width) { px = xst + width; }
|
||||
|
||||
// lines.append(QLine(lastpx, lastpy, px, py));
|
||||
} //else {
|
||||
// Letting the scissor do the dirty work for non horizontal lines
|
||||
// This really should be changed, as it might be cause that weird
|
||||
// display glitch on Linux..
|
||||
lines.append(QLine(lastpx, lastpy, px, py));
|
||||
//}
|
||||
}
|
||||
|
||||
lastpx = px;
|
||||
lastpy = py;
|
||||
@ -965,7 +949,7 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (w.printing() && AppSetting->monochromePrinting()) {
|
||||
painter.setPen(QPen(Qt::black, lineThickness + 0.5));
|
||||
} else {
|
||||
@ -1127,9 +1111,7 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
|
||||
double f = double(cnt) / hours; // / (sum / 3600.0);
|
||||
QString txt = QObject::tr("Duration %1:%2:%3").arg(h,2,10,QChar('0')).arg(m,2,10,QChar('0')).arg(s,2,10,QChar('0')) + " "+
|
||||
QObject::tr("AHI %1").arg(f,0,'f',2);// +" " +
|
||||
// QObject::tr("Events %1").arg(cnt) + " " +
|
||||
// QObject::tr("Hours %1").arg(hours,0,'f',2);
|
||||
QObject::tr("AHI %1").arg(f,0,'f',2);// +" "
|
||||
|
||||
if (linecursormode) txt+=lasttext;
|
||||
|
||||
@ -1138,4 +1120,9 @@ void gLineChart::paint(QPainter &painter, gGraph &w, const QRegion ®ion)
|
||||
|
||||
|
||||
// painter.setRenderHint(QPainter::Antialiasing, false);
|
||||
|
||||
if (actual_max_y>0) {
|
||||
m_actual_min_y=actual_min_y;
|
||||
m_actual_max_y=actual_max_y;
|
||||
}
|
||||
}
|
||||
|
@ -112,6 +112,12 @@ class gLineChart: public Layer
|
||||
//! \brief Returns Maximum Y-axis value for this layer
|
||||
virtual EventDataType Maxy();
|
||||
|
||||
//! \brief Returns Minimum Y-axis value for this layer
|
||||
virtual EventDataType actualMinY() {return m_actual_min_y;};
|
||||
|
||||
//! \brief Returns Maximum Y-axis value for this layer
|
||||
virtual EventDataType actualMaxY() {return m_actual_max_y;};
|
||||
|
||||
//! \brief Returns true if all subplots contain no data
|
||||
virtual bool isEmpty();
|
||||
|
||||
@ -164,6 +170,7 @@ class gLineChart: public Layer
|
||||
bool m_report_empty;
|
||||
bool m_square_plot;
|
||||
bool m_disable_accel;
|
||||
EventDataType m_actual_min_y=0,m_actual_max_y=0;
|
||||
|
||||
//! \brief Used by accelerated waveform plots. Must be >= Screen Resolution (or at least graph width)
|
||||
static const int max_drawlist_size = 10000;
|
||||
|
@ -7,19 +7,8 @@
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
#define xDEBUG_FUNCTIONS
|
||||
#ifdef DEBUG_FUNCTIONS
|
||||
#include <QRegularExpression>
|
||||
#define DEBUG qDebug()<<QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__
|
||||
#define DEBUGF qDebug()<<QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
#define DEBUGTF qDebug()<<QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz") << QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
|
||||
#define O( XX ) " " #XX ":" << XX
|
||||
#define OO( XX , YY ) " " #XX ":" << YY
|
||||
#define NAME( id) schema::channel[ id ].label()
|
||||
#define DATE( XX ) QDateTime::fromMSecsSinceEpoch(XX).toString("dd MMM yyyy")
|
||||
#define DATETIME( XX ) QDateTime::fromMSecsSinceEpoch(XX).toString("dd MMM yyyy hh:mm:ss.zzz")
|
||||
#endif
|
||||
#define TEST_MACROS_ENABLEDoff
|
||||
#include "test_macros.h"
|
||||
|
||||
#include <math.h>
|
||||
#include <QLabel>
|
||||
|
@ -7,6 +7,9 @@
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
#define TEST_MACROS_ENABLEDoff
|
||||
#include <test_macros.h>
|
||||
|
||||
#include "Graphs/gYAxis.h"
|
||||
|
||||
#include <QDebug>
|
||||
@ -20,6 +23,7 @@
|
||||
|
||||
#include <QFontMetrics>
|
||||
|
||||
|
||||
gXGrid::gXGrid(QColor col)
|
||||
: Layer(NoChannel)
|
||||
{
|
||||
@ -306,7 +310,7 @@ bool gYAxis::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
|
||||
int y = event->y();
|
||||
|
||||
if (!graph->units().isEmpty()) {
|
||||
graph->ToolTip(graph->units(), x+10, y+10, TT_AlignLeft);
|
||||
graph->ToolTip(graph->unitsTooltip(), x+10, y+10, TT_AlignLeft);
|
||||
// graph->redraw();
|
||||
}
|
||||
|
||||
@ -316,14 +320,8 @@ bool gYAxis::mouseMoveEvent(QMouseEvent *event, gGraph *graph)
|
||||
bool gYAxis::mouseDoubleClickEvent(QMouseEvent *event, gGraph *graph)
|
||||
{
|
||||
if (graph) {
|
||||
// int x=event->x();
|
||||
// int y=event->y();
|
||||
short z = (graph->zoomY() + 1) % gGraph::maxZoomY;
|
||||
graph->setZoomY(z);
|
||||
qDebug() << "Mouse double clicked for" << graph->name() << z;
|
||||
graph->mouseDoubleClickYAxis(event);
|
||||
}
|
||||
|
||||
Q_UNUSED(event);
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -7,6 +7,9 @@
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
#define TEST_MACROS_ENABLEDoff
|
||||
#include "test_macros.h"
|
||||
|
||||
#include "Graphs/layer.h"
|
||||
|
||||
Layer::~Layer()
|
||||
|
@ -230,6 +230,10 @@ public:
|
||||
Q_UNUSED(graph);
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual EventDataType actualMinY() {return 0;};
|
||||
virtual EventDataType actualMaxY() {return 0;};
|
||||
|
||||
};
|
||||
|
||||
/*! \class LayerGroup
|
||||
|
@ -132,6 +132,7 @@ static const PRS1TestedModel s_PRS1TestedModels[] = {
|
||||
{ "700X130", 0, 6, "DreamStation Auto BiPAP" },
|
||||
{ "700X150", 0, 6, "DreamStation Auto BiPAP" },
|
||||
|
||||
{ "410X150C", 0, 6, "DreamStation 2 CPAP" },
|
||||
{ "520X110C", 0, 6, "DreamStation 2 Auto CPAP Advanced" }, // based on bottom label, boot screen says "Advanced Auto CPAP"
|
||||
{ "520X150C", 0, 6, "DreamStation 2 Auto CPAP Advanced" }, // from user report
|
||||
{ "521X120C", 0, 6, "DreamStation 2 Auto CPAP Advanced with P-Flex" }, // inferred from 501X120 and presence of "P-Flex" on bottom label
|
||||
@ -141,6 +142,7 @@ static const PRS1TestedModel s_PRS1TestedModels[] = {
|
||||
{ "960P", 5, 1, "BiPAP autoSV Advanced (System One 60 Series)" },
|
||||
{ "961P", 5, 1, "BiPAP autoSV Advanced (System One 60 Series)" },
|
||||
{ "960T", 5, 2, "BiPAP autoSV Advanced 30 (System One 60 Series)" }, // omits "(System One 60 Series)" on official reports
|
||||
{ "961TCA", 5, 2, "BiPAP autoSV Advanced 30 (System One 60 Series)" },
|
||||
{ "900X110", 5, 3, "DreamStation BiPAP autoSV" },
|
||||
{ "900X120", 5, 3, "DreamStation BiPAP autoSV" },
|
||||
{ "900X150", 5, 3, "DreamStation BiPAP autoSV" },
|
||||
|
@ -713,12 +713,11 @@ void PRS1DataChunk::ParseHumidifierSetting60Series(unsigned char humid1, unsigne
|
||||
} else if (this->familyVersion == 2) {
|
||||
// F5V2
|
||||
if (tubepresent) {
|
||||
CHECK_VALUES(tubetemp, 0, 3);
|
||||
// all tube temperatures seen
|
||||
if (tubetemp) {
|
||||
CHECK_VALUE(tubehumidlevel, 1);
|
||||
CHECK_VALUES(tubehumidlevel, 1, 3);
|
||||
}
|
||||
}
|
||||
CHECK_VALUE(humidsystemone, false);
|
||||
CHECK_VALUE(humidclassic, false);
|
||||
}
|
||||
}
|
||||
|
@ -10,11 +10,6 @@
|
||||
#include "prs1_parser.h"
|
||||
#include "prs1_loader.h"
|
||||
|
||||
static QString hex(int i)
|
||||
{
|
||||
return QString("0x") + QString::number(i, 16).toUpper();
|
||||
}
|
||||
|
||||
//********************************************************************************************
|
||||
// MARK: -
|
||||
// MARK: 50 and 60 Series
|
||||
@ -664,7 +659,7 @@ bool PRS1DataChunk::ParseEventsF5V1(void)
|
||||
elapsed = data[pos];
|
||||
this->AddEvent(new PRS1FlowLimitationEvent(t - elapsed, 0));
|
||||
break;
|
||||
case 0x0a: // Vibratory Snore, note this is 0x9 in F5V3
|
||||
case 0x0a: // Vibratory Snore, note this is 0xb in F5V2 and 0x9 in F5V3
|
||||
// VS events are instantaneous flags with no duration, drawn on the official waveform.
|
||||
// The current thinking is that these are the snores that cause a change in auto-titrating
|
||||
// pressure. The snoring statistic above seems to be a total count. It's unclear whether
|
||||
@ -672,7 +667,7 @@ bool PRS1DataChunk::ParseEventsF5V1(void)
|
||||
// no data bytes
|
||||
this->AddEvent(new PRS1VibratorySnoreEvent(t, 0));
|
||||
break;
|
||||
case 0x0b: // Periodic Breathing, note this is 0xa in F5V3
|
||||
case 0x0b: // Periodic Breathing, note this is 0xc in F5V2 and 0xa in F5V3
|
||||
// PB events are reported some time after they conclude, and they do have a reported duration.
|
||||
duration = 2 * (data[pos] | (data[pos+1] << 8)); // confirmed to double in F5V0
|
||||
elapsed = data[pos+2];
|
||||
@ -721,12 +716,13 @@ bool PRS1DataChunk::ParseEventsF5V1(void)
|
||||
|
||||
const QVector<PRS1ParsedEventType> ParsedEventsF5V2 = {
|
||||
PRS1EPAPSetEvent::TYPE,
|
||||
PRS1PressurePulseEvent::TYPE,
|
||||
PRS1TimedBreathEvent::TYPE,
|
||||
PRS1ObstructiveApneaEvent::TYPE,
|
||||
//PRS1ClearAirwayEvent::TYPE, // not yet seen
|
||||
PRS1ClearAirwayEvent::TYPE,
|
||||
PRS1HypopneaEvent::TYPE,
|
||||
PRS1FlowLimitationEvent::TYPE,
|
||||
//PRS1VibratorySnoreEvent::TYPE, // not yet seen
|
||||
PRS1VibratorySnoreEvent::TYPE,
|
||||
PRS1PeriodicBreathingEvent::TYPE,
|
||||
//PRS1LargeLeakEvent::TYPE, // not yet seen
|
||||
PRS1IPAPAverageEvent::TYPE,
|
||||
@ -751,7 +747,7 @@ bool PRS1DataChunk::ParseEventsF5V2(void)
|
||||
}
|
||||
const unsigned char * data = (unsigned char *)this->m_data.constData();
|
||||
int chunk_size = this->m_data.size();
|
||||
static const QMap<int,int> event_sizes = { {0,4}, {1,2}, {3,4}, {8,3}, {9,4}, {0xa,3}, {0xb,5}, {0xc,5}, {0xd,5}, {0xe,0xd}, {0xf,5}, {0x10,5}, {0x11,2}, {0x12,6} };
|
||||
static const QMap<int,int> event_sizes = { {0,4}, {1,2}, {8,3}, {9,4}, {0xa,3}, {0xb,2}, {0xc,5}, {0xd,5}, {0xe,0xd}, {0xf,5}, {0x10,5}, {0x11,2}, {0x12,6} };
|
||||
|
||||
if (chunk_size < 1) {
|
||||
// This does occasionally happen in F0V6.
|
||||
@ -780,51 +776,20 @@ bool PRS1DataChunk::ParseEventsF5V2(void)
|
||||
}
|
||||
startpos = pos;
|
||||
if (code != 0 && code != 0x12) { // These two codes have no timestamp TODO: verify this applies to F5V012
|
||||
t += data[pos] /*| (data[pos+1] << 8)*/; // TODO: Is this really only 1 byte?
|
||||
if (data[pos+1] != 0) qWarning() << this->sessionid << "nonzero time? byte" << hex(startpos);
|
||||
CHECK_VALUE(data[pos+1], 0);
|
||||
t += data[pos] | (data[pos+1] << 8);
|
||||
pos += 2;
|
||||
}
|
||||
|
||||
switch (code) {
|
||||
/*
|
||||
case 0x00: // Unknown (ASV Pressure value)
|
||||
DUMP_EVENT();
|
||||
// offset?
|
||||
data0 = data[pos++];
|
||||
|
||||
if (!data[pos - 1]) { // WTH???
|
||||
data1 = data[pos++];
|
||||
}
|
||||
|
||||
if (!data[pos - 1]) {
|
||||
//data2 = data[pos++];
|
||||
pos++;
|
||||
}
|
||||
|
||||
break;
|
||||
|
||||
case 0x01: // Unknown
|
||||
DUMP_EVENT();
|
||||
this->AddEvent(new PRS1UnknownValueEvent(code, t, 0, 0.1F));
|
||||
break;
|
||||
*/
|
||||
//case 0x00: // never seen on F5V2
|
||||
//case 0x01: // never seen on F5V2
|
||||
case 0x02: // Pressure adjustment
|
||||
this->AddEvent(new PRS1EPAPSetEvent(t, data[pos++], GAIN));
|
||||
break;
|
||||
/*
|
||||
case 0x03: // BIPAP Pressure
|
||||
DUMP_EVENT();
|
||||
qDebug() << "0x03 Observed in ASV data!!????";
|
||||
|
||||
data0 = data[pos++];
|
||||
data1 = data[pos++];
|
||||
// data0/=10.0;
|
||||
// data1/=10.0;
|
||||
// session->AddEvent(new Event(t,CPAP_EAP, 0, data, 1));
|
||||
// session->AddEvent(new Event(t,CPAP_IAP, 0, &data1, 1));
|
||||
case 0x03: // Pressure Pulse
|
||||
duration = data[pos]; // TODO: is this a duration?
|
||||
this->AddEvent(new PRS1PressurePulseEvent(t, duration));
|
||||
break;
|
||||
*/
|
||||
case 0x04: // Timed Breath
|
||||
// TB events have a duration in 0.1s, based on the review of pressure waveforms.
|
||||
// TODO: Ideally the starting time here would be adjusted here, but PRS1ParsedEvents
|
||||
@ -839,15 +804,13 @@ bool PRS1DataChunk::ParseEventsF5V2(void)
|
||||
elapsed = data[pos];
|
||||
this->AddEvent(new PRS1ObstructiveApneaEvent(t - elapsed, 0));
|
||||
break;
|
||||
/*
|
||||
case 0x06:
|
||||
DUMP_EVENT();
|
||||
//code=CPAP_ClearAirway;
|
||||
data0 = data[pos++];
|
||||
this->AddEvent(new PRS1ClearAirwayEvent(t - data0, data0));
|
||||
case 0x06: // Clear Airway Apnea
|
||||
// CA events are instantaneous flags with no duration: reviewing waveforms
|
||||
// shows that the time elapsed between the flag and reporting often includes
|
||||
// non-apnea breathing.
|
||||
elapsed = data[pos];
|
||||
this->AddEvent(new PRS1ClearAirwayEvent(t - elapsed, 0));
|
||||
break;
|
||||
*/
|
||||
|
||||
case 0x07: // Hypopnea
|
||||
// NOTE: No additional (unknown) first byte as in F5V3 0x07, but see below.
|
||||
// This seems closer to F5V3 0x0d or 0x0e.
|
||||
@ -862,21 +825,7 @@ bool PRS1DataChunk::ParseEventsF5V2(void)
|
||||
elapsed = data[pos]; // based on sample waveform, the hypopnea is over after this
|
||||
this->AddEvent(new PRS1HypopneaEvent(t - elapsed, 0));
|
||||
break;
|
||||
/*
|
||||
case 0x09: // ASV Codes
|
||||
DUMP_EVENT();
|
||||
/ *
|
||||
if (this->familyVersion<2) {
|
||||
//code=CPAP_FlowLimit;
|
||||
data0 = data[pos++];
|
||||
|
||||
this->AddEvent(new PRS1FlowLimitationEvent(t - data0, data0));
|
||||
} else {
|
||||
* /
|
||||
data0 = data[pos++];
|
||||
data1 = data[pos++];
|
||||
break;
|
||||
*/
|
||||
//case 0x09: // never seen on F5V2
|
||||
case 0x0a: // Flow Limitation, note this is 0x9 in F5V1 and 0x8 in F5V3
|
||||
// TODO: We should revisit whether this is elapsed or duration once (if)
|
||||
// we start calculating flow limitations ourselves. Flow limitations aren't
|
||||
@ -884,35 +833,21 @@ bool PRS1DataChunk::ParseEventsF5V2(void)
|
||||
elapsed = data[pos];
|
||||
this->AddEvent(new PRS1FlowLimitationEvent(t - elapsed, 0));
|
||||
break;
|
||||
/*
|
||||
case 0x0b: // Cheyne Stokes
|
||||
DUMP_EVENT();
|
||||
data0 = ((unsigned char *)data)[pos + 1] << 8 | ((unsigned char *)data)[pos];
|
||||
//data0*=2;
|
||||
pos += 2;
|
||||
data1 = ((unsigned char *)data)[pos]; //|data[pos+1] << 8
|
||||
pos += 1;
|
||||
//tt-=delta;
|
||||
this->AddEvent(new PRS1PeriodicBreathingEvent(t - data1, data0));
|
||||
case 0x0b: // Vibratory Snore, note this is 0xa in F5V1 and 0x9 in F5V3
|
||||
// VS events are instantaneous flags with no duration, drawn on the official waveform.
|
||||
// The current thinking is that these are the snores that cause a change in auto-titrating
|
||||
// pressure. The snoring statistic above seems to be a total count. It's unclear whether
|
||||
// the trigger for pressure change is severity or count or something else.
|
||||
// no data bytes
|
||||
this->AddEvent(new PRS1VibratorySnoreEvent(t, 0));
|
||||
break;
|
||||
*/
|
||||
case 0x0c: // Periodic Breathing, note this is 0xb in F5V1 and 0xa in F5V3
|
||||
// PB events are reported some time after they conclude, and they do have a reported duration.
|
||||
duration = 2 * (data[pos] | (data[pos+1] << 8)); // confirmed to double in F5V0
|
||||
elapsed = data[pos+2];
|
||||
this->AddEvent(new PRS1PeriodicBreathingEvent(t - elapsed - duration, duration));
|
||||
break;
|
||||
/*
|
||||
case 0x0d:
|
||||
DUMP_EVENT();
|
||||
|
||||
data0 = (data[pos + 1] << 8 | data[pos]);
|
||||
data0 *= 2;
|
||||
pos += 2;
|
||||
data1 = data[pos++];
|
||||
//tt = t - qint64(data1) * 1000L;
|
||||
break;
|
||||
*/
|
||||
//case 0x0d: // never seen on F5V2
|
||||
case 0x0e: // Statistics, note this was 0x0d in F5V0 and F5V1
|
||||
// These appear every 2 minutes, so presumably summarize the preceding period.
|
||||
this->AddEvent(new PRS1IPAPAverageEvent(t, data[pos+0], GAIN)); // 00=IPAP
|
||||
@ -928,53 +863,6 @@ bool PRS1DataChunk::ParseEventsF5V2(void)
|
||||
this->AddEvent(new PRS1LeakEvent(t, data[pos+0xa])); // 0A=Leak (average?) new to F5V1 (originally found in F5V3)
|
||||
this->AddEvent(new PRS1IntervalBoundaryEvent(t));
|
||||
break;
|
||||
/*
|
||||
case 0x0f:
|
||||
DUMP_EVENT();
|
||||
qDebug() << "0x0f Observed in ASV data!!????";
|
||||
|
||||
data0 = data[pos + 1] << 8 | data[pos];
|
||||
pos += 2;
|
||||
data1 = data[pos]; //|data[pos+1] << 8
|
||||
pos += 1;
|
||||
//tt -= qint64(data1) * 1000L;
|
||||
//session->AddEvent(new Event(tt,cpapcode, 0, data, 2));
|
||||
break;
|
||||
|
||||
case 0x10: // Unknown
|
||||
DUMP_EVENT();
|
||||
data0 = data[pos + 1] << 8 | data[pos];
|
||||
pos += 2;
|
||||
data1 = data[pos++];
|
||||
|
||||
this->AddEvent(new PRS1LargeLeakEvent(t - data1, data0));
|
||||
|
||||
// qDebug() << "0x10 Observed in ASV data!!????";
|
||||
// data0 = data[pos++]; // << 8) | data[pos];
|
||||
// data1 = data[pos++];
|
||||
// data2 = data[pos++];
|
||||
//session->AddEvent(new Event(t,cpapcode, 0, data, 3));
|
||||
break;
|
||||
case 0x11: // Not Leak Rate
|
||||
DUMP_EVENT();
|
||||
qDebug() << "0x11 Observed in ASV data!!????";
|
||||
//if (!Code[24]) {
|
||||
// Code[24]=new EventList(cpapcode,EVL_Event);
|
||||
//}
|
||||
//Code[24]->AddEvent(t,data[pos++]);
|
||||
break;
|
||||
|
||||
|
||||
case 0x12: // Summary
|
||||
DUMP_EVENT();
|
||||
qDebug() << "0x12 Observed in ASV data!!????";
|
||||
data0 = data[pos++];
|
||||
data1 = data[pos++];
|
||||
//data2 = data[pos + 1] << 8 | data[pos];
|
||||
pos += 2;
|
||||
//session->AddEvent(new Event(t,cpapcode, 0, data,3));
|
||||
break;
|
||||
*/
|
||||
default:
|
||||
DUMP_EVENT();
|
||||
UNEXPECTED_VALUE(code, "known event code");
|
||||
|
@ -2078,6 +2078,14 @@ bool PRS1DataChunk::ParseSettingsF0V6(const unsigned char* data, int size)
|
||||
// TODO: Confirm that 4 is 12HT and update ParseTubingTypeV3.
|
||||
this->ParseTubingTypeV3(data[pos]);
|
||||
break;
|
||||
case 0x48: // ??? Seen on DreamStation 2 non-Advanced (410)
|
||||
// Appears between 0x2C and 0x2E, but the only values we've seen other than 0 is
|
||||
// 1, which doesn't seem to be like a reasonable pressure (for ramp pressure) nor
|
||||
// a boolean setting, which tends to be 0x80.
|
||||
CHECK_VALUE(len, 1);
|
||||
CHECK_VALUES(data[pos], 0, 1);
|
||||
//this->AddEvent(new PRS1UnknownDataEvent(QByteArray((const char*) data, size), pos, len));
|
||||
break;
|
||||
case 0x4a: // Patient controls access, specific to DreamStation 2.
|
||||
CHECK_VALUE(len, 1);
|
||||
CHECK_VALUES(data[pos], 0, 0x80);
|
||||
|
@ -527,7 +527,6 @@ message("CXXFLAGS pre-mods $$QMAKE_CXXFLAGS ")
|
||||
QMAKE_CFLAGS += -Werror
|
||||
QMAKE_CXXFLAGS += -Werror
|
||||
|
||||
|
||||
gcc | clang {
|
||||
COMPILER_VERSION = $$system($$QMAKE_CXX " -dumpversion")
|
||||
COMPILER_MAJOR = $$split(COMPILER_VERSION, ".")
|
||||
@ -536,25 +535,28 @@ gcc | clang {
|
||||
message("$$QMAKE_CXX major version $$COMPILER_MAJOR")
|
||||
}
|
||||
|
||||
## equals($$QMAKE_CXX, "gcc") : { // guess what! the name is really "g++"
|
||||
equals(QMAKE_CXX, g++) {
|
||||
message("Detected compiler g++")
|
||||
gcc:!clang {
|
||||
message("Building for $$QMAKE_HOST.os")
|
||||
greaterThan(COMPILER_MAJOR, 10) : {
|
||||
QMAKE_CFLAGS += -Wno-error=stringop-overread
|
||||
QMAKE_CXXFLAGS += -Wno-error=stringop-overread
|
||||
message("Making stringop-overread a non-error")
|
||||
}
|
||||
}
|
||||
equals(QMAKE_CXX, clang++) {
|
||||
message("Detected compiler clang++")
|
||||
}
|
||||
|
||||
clang {
|
||||
message("Building for $$QMAKE_HOST.os")
|
||||
QMAKE_CFLAGS_WARN_ON += -Wno-error=deprecated-copy
|
||||
QMAKE_CXXFLAGS_WARN_ON += -Wno-error=deprecated-copy
|
||||
message("Making deprecated-copy a non-error")
|
||||
}
|
||||
|
||||
# Make deprecation warnings just warnings
|
||||
QMAKE_CFLAGS += -Wno-error=deprecated-declarations
|
||||
QMAKE_CXXFLAGS += -Wno-error=deprecated-declarations
|
||||
|
||||
message("CXXFLAGS post-mods $$QMAKE_CXXFLAGS ")
|
||||
message("CXXFLAGS_WARN_ON $$QMAKE_CXXFLAGS_WARN_ON")
|
||||
|
||||
lessThan(QT_MAJOR_VERSION,5)|lessThan(QT_MINOR_VERSION,9) {
|
||||
QMAKE_CFLAGS += -Wno-error=strict-aliasing
|
||||
|
@ -7,16 +7,19 @@
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
#define NEWSTUFF
|
||||
|
||||
#define xDEBUG_FUNCTIONS
|
||||
#ifdef DEBUG_FUNCTIONS
|
||||
#include <QRegularExpression>
|
||||
#define DEBUGQ qDebug()
|
||||
#define DEBUGL qDebug()<<QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__
|
||||
#define DEBUGF qDebug()<<QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
#define DEBUGTF qDebug()<<QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz") << QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
#define DEBUGF qDebug()<< QString("%1[%2]%3").arg( QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) ).arg(__LINE__).arg(__func__)
|
||||
#define DEBUGT qDebug()<<QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz")
|
||||
|
||||
#define DEBUGTF qDebug()<<QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz") << QString(basename( __FILE__)).remove(QRegularExpression("\\..*$")) << __LINE__ << __func__
|
||||
#define O( XX ) " " #XX ":" << XX
|
||||
#define Q( XX ) << #XX ":" << XX
|
||||
#define R( XX )
|
||||
#define OO( XX , YY ) " " #XX ":" << YY
|
||||
#define NAME( id) schema::channel[ id ].label()
|
||||
#define DATE( XX ) QDateTime::fromMSecsSinceEpoch(XX).toString("dd MMM yyyy")
|
||||
@ -74,7 +77,7 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
|
||||
|
||||
// Set Date controls locale to 4 digit years
|
||||
QLocale locale = QLocale::system();
|
||||
QString shortformat = locale.dateFormat(QLocale::ShortFormat);
|
||||
shortformat = locale.dateFormat(QLocale::ShortFormat);
|
||||
|
||||
if (!shortformat.toLower().contains("yyyy")) {
|
||||
shortformat.replace("yy", "yyyy");
|
||||
@ -173,6 +176,7 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
|
||||
GraphView->LoadSettings("Overview"); //no trans
|
||||
|
||||
GraphView->setEmptyImage(QPixmap(":/icons/logo-md.png"));
|
||||
dateErrorDisplay = new DateErrorDisplay(this);
|
||||
|
||||
connect(ui->dateStart->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateStart_currentPageChanged(int, int)));
|
||||
connect(ui->dateEnd->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateEnd_currentPageChanged(int, int)));
|
||||
@ -196,6 +200,11 @@ Overview::~Overview()
|
||||
GraphView->SaveSettings("Overview");//no trans
|
||||
|
||||
delete ui;
|
||||
delete dateErrorDisplay;
|
||||
delete icon_on ;
|
||||
delete icon_off ;
|
||||
delete icon_up_down ;
|
||||
delete icon_warning ;
|
||||
}
|
||||
|
||||
void Overview::ResetFont()
|
||||
@ -235,12 +244,10 @@ void Overview::on_summaryChartEmpty(gSummaryChart*sc,qint64 firstI,qint64 lastI,
|
||||
if (empty) {
|
||||
// on next range change allow empty flag to be recalculated
|
||||
chartsEmpty.insert(sc,graph);
|
||||
//DEBUGF << graph->name() << "Chart is Empty" << "Range" << convertTimeToDate(firstI) << convertTimeToDate(lastI);
|
||||
} else {
|
||||
// The chart has some entry with data.
|
||||
chartsEmpty.remove(sc);
|
||||
updateGraphCombo();
|
||||
//DEBUGF << graph->name() << "Chart is enabled with range:" << convertTimeToDate(firstI) << "==>" << convertTimeToDate(lastI) ;
|
||||
}
|
||||
Q_UNUSED(firstI);
|
||||
Q_UNUSED(lastI);
|
||||
@ -322,11 +329,8 @@ void Overview::CreateAllGraphs() {
|
||||
G->AddLayer(sc);
|
||||
chartsToBeMonitored.insert(sc,G);
|
||||
}
|
||||
if (sc== nullptr) {
|
||||
//DEBUGF << "Channel" << name << "type" << chan->type() << "machine type" << chan->machtype() << "IGNORED";
|
||||
} else {
|
||||
if (sc!= nullptr) {
|
||||
sc ->reCalculate();
|
||||
//DEBUGF << "Channel" << name << "type" << chan->type() << "machine type" << chan->machtype() << OO(Empty,sc->isEmpty());
|
||||
}
|
||||
} // if showInOverview()
|
||||
} // for chit
|
||||
@ -549,12 +553,6 @@ void Overview::on_XBoundsChanged(qint64 start,qint64 end)
|
||||
// Only occurs when custom mode is switched to/from a latest mode. custom mode to/from last week.
|
||||
// All other displays expand the existing range.
|
||||
// reset all empty flags to not empty
|
||||
if (displayStartDate>maxRangeEndDate) {
|
||||
//DEBUGF << "Two ranges" O(displayStartDate) <<">" << O(maxRangeEndDate);
|
||||
}
|
||||
if (minRangeStartDate>displayEndDate) {
|
||||
//DEBUGF << "Two ranges" O(minRangeStartDate) <<">" << O(displayEndDate);
|
||||
}
|
||||
largerRange=true;
|
||||
chartsEmpty = QHash<gSummaryChart*, gGraph*>( chartsToBeMonitored );
|
||||
minRangeStartDate = displayStartDate;
|
||||
@ -562,24 +560,14 @@ void Overview::on_XBoundsChanged(qint64 start,qint64 end)
|
||||
} else {
|
||||
// new range overlaps with old range
|
||||
if (displayStartDate<minRangeStartDate) {
|
||||
//DEBUGF << "Start lower" <<O(minRangeStartDate)<< ">" <<O(displayStartDate);
|
||||
largerRange=true;
|
||||
minRangeStartDate = displayStartDate;
|
||||
}
|
||||
if (displayEndDate>maxRangeEndDate) {
|
||||
//DEBUGF << "End Higher" <<O(maxRangeEndDate)<< "<" <<O(displayEndDate);
|
||||
largerRange=true;
|
||||
maxRangeEndDate = displayEndDate;
|
||||
}
|
||||
}
|
||||
if (!largerRange) {
|
||||
if (displayStartDate<minRangeStartDate ) {
|
||||
//DEBUGF << "ERROR" <<O(minRangeStartDate)<< "==" <<O(displayStartDate);
|
||||
}
|
||||
if (displayEndDate>maxRangeEndDate) {
|
||||
//DEBUGF << "ERROR" <<O(maxRangeEndDate)<< "==" <<O(displayEndDate);
|
||||
}
|
||||
}
|
||||
|
||||
if (largerRange) {
|
||||
for (auto it= chartsEmpty.begin();it!=chartsEmpty.end();it++) {
|
||||
@ -624,6 +612,10 @@ void Overview::dateEnd_currentPageChanged(int year, int month)
|
||||
|
||||
void Overview::on_dateEnd_dateChanged(const QDate &date)
|
||||
{
|
||||
if (date<uiStartDate) {
|
||||
dateErrorDisplay->error(false,date);
|
||||
return;
|
||||
}
|
||||
QDate d2(date);
|
||||
if (customMode) {
|
||||
p_profile->general->setCustomOverviewRangeEnd(d2);
|
||||
@ -636,6 +628,11 @@ void Overview::on_dateEnd_dateChanged(const QDate &date)
|
||||
|
||||
void Overview::on_dateStart_dateChanged(const QDate &date)
|
||||
{
|
||||
if (date>uiEndDate) {
|
||||
// change date back to last date.
|
||||
dateErrorDisplay->error(true,date);
|
||||
return;
|
||||
}
|
||||
QDate d1(date);
|
||||
if (customMode) {
|
||||
p_profile->general->setCustomOverviewRangeStart(d1);
|
||||
@ -654,6 +651,7 @@ void Overview::on_zoomButton_clicked()
|
||||
on_rangeCombo_activated(p_profile->general->lastOverviewRange()); // type of range in last use
|
||||
}
|
||||
|
||||
|
||||
void Overview::ResetGraphLayout()
|
||||
{
|
||||
GraphView->resetLayout();
|
||||
@ -732,7 +730,7 @@ void Overview::on_rangeCombo_activated(int index)
|
||||
|
||||
// Ensure that all summary files are available and update version numbers if required
|
||||
int size = start.daysTo(end);
|
||||
qDebug() << "Overview range combo from" << start << "to" << end << "with" << size << "days";
|
||||
// qDebug() << "Overview range combo from" << start << "to" << end << "with" << size << "days";
|
||||
QDate dateback = end;
|
||||
CProgressBar * progress = new CProgressBar (QObject::tr("Loading summaries"), mainwin, size);
|
||||
for (int i=1; i < size; ++i) {
|
||||
@ -754,12 +752,101 @@ void Overview::on_rangeCombo_activated(int index)
|
||||
setRange(start, end);
|
||||
}
|
||||
|
||||
|
||||
DateErrorDisplay::DateErrorDisplay (Overview* overview)
|
||||
: m_overview(overview)
|
||||
{
|
||||
m_visible=false;
|
||||
m_timer = new QTimer();
|
||||
connect(m_timer, SIGNAL(timeout()),this, SLOT(timerDone()));
|
||||
};
|
||||
|
||||
DateErrorDisplay::~DateErrorDisplay() {
|
||||
disconnect(m_timer, SIGNAL(timeout()),this, SLOT(timerDone()));
|
||||
delete m_timer;
|
||||
};
|
||||
|
||||
void DateErrorDisplay::cancel() {
|
||||
m_visible=false;
|
||||
m_timer->stop();
|
||||
};
|
||||
|
||||
void DateErrorDisplay::timerDone() {
|
||||
m_visible=false;
|
||||
m_overview->resetUiDates();
|
||||
m_overview->graphView()->m_parent_tooltip->cancel();
|
||||
};
|
||||
|
||||
int Overview::calculatePixels(bool startDate,ToolTipAlignment& align) {
|
||||
// Center error message over start and end dates combined.
|
||||
// Other allignement were tested but this is the best for this problem.
|
||||
Q_UNUSED(startDate);
|
||||
int space=4;
|
||||
align=TT_AlignCenter;
|
||||
return ((4*space) + ui->label_3->width() + ui->rangeCombo->width() + ui->dateStartLabel->width() +ui->dateStart->width() );
|
||||
}
|
||||
|
||||
void DateErrorDisplay::error(bool startDate,const QDate& dateEntered) {
|
||||
m_startDate =m_overview->uiStartDate;
|
||||
m_endDate =m_overview->uiEndDate;
|
||||
ToolTipAlignment align=TT_AlignCenter;
|
||||
|
||||
|
||||
QString dateFormatted =dateEntered.toString(m_overview->shortformat);
|
||||
|
||||
QString txt = QString(tr("ERROR\nThe start date MUST be before the end date"));
|
||||
txt.append("\n");
|
||||
if (startDate) {
|
||||
txt.append(tr("The entered start date %1 is after the end date %2").arg(dateFormatted).arg(m_endDate.toString(m_overview->shortformat)));
|
||||
txt.append(tr("\nHint: Change the end date first"));
|
||||
} else {
|
||||
txt.append(tr("The entered end date %1 ").arg(dateFormatted));
|
||||
txt.append(tr("is before the start date %1").arg(m_startDate.toString(m_overview->shortformat)));
|
||||
txt.append(tr("\nHint: Change the start date first"));
|
||||
}
|
||||
|
||||
gGraphView* gv=m_overview->graphView();
|
||||
|
||||
int pixelsFromLeft = m_overview->calculatePixels(startDate,align);
|
||||
int pixelsAboveBottom = 0;
|
||||
int warningDurationMS =4000;
|
||||
int resetUiDatesTimerDelayMS =1000;
|
||||
//QFont* font=mediumfont;
|
||||
QFont* font=defaultfont;
|
||||
|
||||
gv->m_parent_tooltip->display(gv,txt,pixelsAboveBottom,pixelsFromLeft, align, warningDurationMS , font);
|
||||
|
||||
m_timer->setInterval( resetUiDatesTimerDelayMS );
|
||||
m_timer->setSingleShot(true);
|
||||
m_timer->start();
|
||||
|
||||
};
|
||||
|
||||
void Overview::resetUiDates() {
|
||||
ui->dateStart->blockSignals(true);
|
||||
ui->dateStart->setMinimumDate(p_profile->FirstDay()); // first and last dates for ANY machine type
|
||||
ui->dateStart->setMaximumDate(p_profile->LastDay());
|
||||
ui->dateStart->setDate(uiStartDate);
|
||||
ui->dateStart->blockSignals(false);
|
||||
|
||||
ui->dateEnd->blockSignals(true);
|
||||
ui->dateEnd->setMinimumDate(p_profile->FirstDay()); // first and last dates for ANY machine type
|
||||
ui->dateEnd->setMaximumDate(p_profile->LastDay());
|
||||
ui->dateEnd->setDate(uiEndDate);
|
||||
ui->dateEnd->blockSignals(false);
|
||||
}
|
||||
|
||||
// Saves dates in UI, clicks zoom button, and updates combo box
|
||||
// 1. Updates the dates in the start / end date boxs
|
||||
// 2. optionally also changes display range for graphs.
|
||||
void Overview::setRange(QDate& start, QDate& end, bool updateGraphs/*zoom*/)
|
||||
{
|
||||
|
||||
if (start>end) {
|
||||
// this is an ERROR and shold NEVER occur.
|
||||
return;
|
||||
}
|
||||
|
||||
// first setting of the date (setDate) will cause pageChanged to be executed.
|
||||
// The pageChanged processing requires access to the other ui's date.
|
||||
// so save them to memory before the first call.
|
||||
@ -777,21 +864,7 @@ void Overview::setRange(QDate& start, QDate& end, bool updateGraphs/*zoom*/)
|
||||
samePage=nextSamePage;
|
||||
}
|
||||
|
||||
ui->dateEnd->blockSignals(true);
|
||||
ui->dateStart->blockSignals(true);
|
||||
|
||||
// Calling these methods for the first time trigger pageChange actions.
|
||||
ui->dateEnd->setDate(end);
|
||||
ui->dateEnd->setMinimumDate(start);
|
||||
|
||||
ui->dateStart->setMinimumDate(p_profile->FirstDay()); // first and last dates for ANY machine type
|
||||
ui->dateEnd->setMaximumDate(p_profile->LastDay());
|
||||
|
||||
ui->dateStart->setDate(start);
|
||||
ui->dateStart->setMaximumDate(end);
|
||||
|
||||
ui->dateEnd->blockSignals(false);
|
||||
ui->dateStart->blockSignals(false);
|
||||
resetUiDates();
|
||||
if (updateGraphs) SetXBounds(uiStartDate,uiEndDate);
|
||||
updateGraphCombo();
|
||||
}
|
||||
|
@ -16,6 +16,7 @@
|
||||
#endif
|
||||
#include <QHBoxLayout>
|
||||
#include <QDateEdit>
|
||||
#include <QTimer>
|
||||
#include "SleepLib/profiles.h"
|
||||
#include "Graphs/gGraphView.h"
|
||||
#include "Graphs/gSummaryChart.h"
|
||||
@ -26,6 +27,27 @@ class Overview;
|
||||
}
|
||||
|
||||
class Report;
|
||||
class Overview;
|
||||
|
||||
class DateErrorDisplay:QObject {
|
||||
Q_OBJECT
|
||||
public:
|
||||
DateErrorDisplay (Overview* overview) ;
|
||||
~DateErrorDisplay() ;
|
||||
bool visible() {return m_visible;};
|
||||
void cancel();
|
||||
void error(bool startDate,const QDate& date);
|
||||
protected:
|
||||
|
||||
private:
|
||||
QTimer* m_timer;
|
||||
bool m_visible=false;
|
||||
Overview* m_overview;
|
||||
QDate m_startDate;
|
||||
QDate m_endDate;
|
||||
private slots:
|
||||
void timerDone();
|
||||
};
|
||||
|
||||
enum YTickerType { YT_Number, YT_Time, YT_Weight };
|
||||
|
||||
@ -35,6 +57,7 @@ enum YTickerType { YT_Number, YT_Time, YT_Weight };
|
||||
*/
|
||||
class Overview : public QWidget
|
||||
{
|
||||
friend class DateErrorDisplay;
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
@ -103,8 +126,6 @@ class Overview : public QWidget
|
||||
//! \brief Updates the calendar highlighting when changing to a new month
|
||||
void dateEnd_currentPageChanged(int year, int month);
|
||||
|
||||
//void on_printDailyButton_clicked();
|
||||
|
||||
void on_rangeCombo_activated(int index);
|
||||
|
||||
void on_graphCombo_activated(int index);
|
||||
@ -143,6 +164,10 @@ class Overview : public QWidget
|
||||
void disconnectgSummaryCharts() ;
|
||||
void SetXBounds(qint64 minx, qint64 maxx, short group = 0, bool refresh = true);
|
||||
void SetXBounds(QDate & start, QDate& end, short group =0 , bool refresh = true);
|
||||
void resetUiDates();
|
||||
DateErrorDisplay* dateErrorDisplay;
|
||||
int calculatePixels(bool startDate,ToolTipAlignment& align);
|
||||
QString shortformat;
|
||||
|
||||
// Start and of dates of the current graph display
|
||||
QDate displayStartDate;
|
||||
@ -166,4 +191,7 @@ class Overview : public QWidget
|
||||
|
||||
};
|
||||
|
||||
|
||||
|
||||
|
||||
#endif // OVERVIEW_H
|
||||
|
119
oscar/test_macros.h
Normal file
119
oscar/test_macros.h
Normal file
@ -0,0 +1,119 @@
|
||||
/* Test macros Implemntation
|
||||
*
|
||||
* Copyright (c) 2019-2022 The OSCAR Team
|
||||
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
||||
*
|
||||
* This file is subject to the terms and conditions of the GNU General Public
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
/*
|
||||
These functions will display formatted debug information.
|
||||
The macro TEST_MACROS_ENABLED will enable these macros to display information
|
||||
When The macro TEST_MACROS_ENABLED is undefined then these marcos will expand to white space.
|
||||
|
||||
When these macos are used then debugging is disabled
|
||||
When only these macos are used then debugging is disabled.
|
||||
SO for production code rename TEST_MACROS_ENABLED to TEST_MACROS_ENABLEDoff
|
||||
|
||||
###########################################
|
||||
The the following to source cpp files
|
||||
to turn on the debug macros for use.
|
||||
|
||||
#define TEST_MACROS_ENABLED
|
||||
#include <test_macros.h>
|
||||
|
||||
To turn off the the test macros.
|
||||
#define TEST_MACROS_ENABLEDoff
|
||||
#include <test_macros.h>
|
||||
###########################################
|
||||
|
||||
|
||||
*/
|
||||
|
||||
#ifndef TEST_MACROS_ENABLED_ONCE
|
||||
#define TEST_MACROS_ENABLED_ONCE
|
||||
|
||||
#ifdef TEST_MACROS_ENABLED
|
||||
#include <QRegularExpression>
|
||||
#include <QFileInfo>
|
||||
|
||||
#define DEBUGL qDebug() <<QString("%1[%2]").arg(QFileInfo( __FILE__).baseName()).arg(__LINE__)
|
||||
//example:
|
||||
//12361: Debug: "gGraphView[572]"
|
||||
|
||||
|
||||
#define DEBUGF qDebug() <<QString("%1[%2]%3").arg(QFileInfo( __FILE__).baseName()).arg(__LINE__).arg(__func__)
|
||||
//example:
|
||||
//12361: Debug: "gGraphView[572]popoutGraph"
|
||||
|
||||
|
||||
#define DEBUGT qDebug() <<QString("%1 %2[%3]%4").arg(QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz")).arg(QFileInfo( __FILE__).baseName()).arg(__LINE__)
|
||||
//example:
|
||||
//12645: Debug: "06:00:18.284 gGraphView[622]"
|
||||
|
||||
#define DEBUGTF qDebug() <<QString("%1 %2[%3]%4").arg(QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz")).arg(QFileInfo( __FILE__).baseName()).arg(__LINE__).arg(__func__)
|
||||
//example:
|
||||
//12645: Debug: "06:00:18.284 gGraphView[622]popoutGraph"
|
||||
|
||||
// Macros to display variables
|
||||
#define O( EXPRESSION ) << EXPRESSION
|
||||
|
||||
// return values from functions donr't work - QQ macro instead
|
||||
#define Q( VALUE ) << "" #VALUE ":" << VALUE
|
||||
//example:
|
||||
|
||||
//
|
||||
#define QQ( TEXT , EXPRESSION) << #TEXT ":" << EXPRESSION
|
||||
//example:
|
||||
|
||||
#define NAME( SCHEMECODE ) << schema::channel[ SCHEMECODE ].label()
|
||||
//example:
|
||||
|
||||
#define FULLNAME( SCHEMECODE ) << schema::channel[ SCHEMEcODE ].fullname()
|
||||
//example:
|
||||
|
||||
//display the date of an epoch time stamp "qint64"
|
||||
#define DATE( EPOCH ) << QDateTime::fromMSecsSinceEpoch( EPOCH ).toString("dd MMM yyyy")
|
||||
|
||||
//display the date and Time of an epoch time stamp "qint64"
|
||||
#define DATETIME( EPOCH ) << QDateTime::fromMSecsSinceEpoch( EPOCH ).toString("dd MMM yyyy hh:mm:ss.zzz")
|
||||
|
||||
/*
|
||||
sample Lines.
|
||||
|
||||
code
|
||||
DEBUGF Q(name) Q(title) QQ("UNITS",units) Q(height) Q(group);
|
||||
output
|
||||
00791: Debug: "gGraph[137]gGraph" name: "RespRate" title: "Respiratory Rate" "UNITS": "Rate of breaths per minute" height: 180 group: 0
|
||||
|
||||
|
||||
DEBUGTF Q(newname);
|
||||
12645: Debug: "06:00:18.284 gGraphView[622]popoutGraph" newname: "Pressure - Friday, April 15, 2022"
|
||||
|
||||
|
||||
DEBUGF NAME(dot.code) Q(dot.type) QQ(y,(int)y) Q(ratioX) O(QLine(left + 1, y, left + 1 + width, y)) Q(legendx) O(dot.value) ;
|
||||
92 00917: Debug: "gLineChart[568]paint" "Pressure" dot.type: 4 y: 341 ratioX: 1 QLine(QPoint(91,341),QPoint(464,341)) legendx: 463 12.04
|
||||
|
||||
|
||||
*/
|
||||
|
||||
#else
|
||||
// Turn debugging off. macros expands to white space
|
||||
|
||||
#define DEBUGL
|
||||
#define DEBUGF
|
||||
#define DEBUGT
|
||||
#define DEBUGTF
|
||||
|
||||
#define O( XX )
|
||||
#define Q( XX )
|
||||
#define QQ( XX , YY )
|
||||
#define NAME( id)
|
||||
#define FULLNAME( id)
|
||||
#define DATE( XX )
|
||||
#define DATETIME( XX )
|
||||
|
||||
#endif
|
||||
#endif
|
||||
|
Loading…
Reference in New Issue
Block a user