*CSR pinned on overview tab disappears on the first paint.
Cause
*Charts are automatically permanently disabled for charts that are initially empty
*This problem impacts any chart that is enabled in File => Preferences.
Details
*gGraph does not display layers (paint) when the layer is empty.
*gSummaryChart (layer) initializes empty to false and calculates empty during painting.
Causing empty charts to be permanently disabled.
*Once a chart has been displayed, it will always be displayed even if empty.
Solution
*Enabled painting charts when the DayRange is increased, triggering the empty flag to be recalculated
New non-empty charts will now be permanently displayed
*Overview maintains the largest DayRange used and list of empty charts.
*Overview receive these signals and calls recalculates when largest DayRange is increased.
*gSummaryChart implement recalculate method.
*gSummaryChart implement emit signal when empty flag changes.
*gGraphView implement emit signal when day range changes. (XBounds)
**** IMPORTANT MUST Rebuild makefiles.****
*Remove two Makefiles and execute qmake. (works for linux version).
This commit is contained in:
LoudSnorer 2022-02-22 07:48:30 -05:00
parent 3e069fa4de
commit 6b81db5bbb
6 changed files with 272 additions and 59 deletions

View File

@ -7,6 +7,20 @@
* License. See the file COPYING in the main directory of the source code * License. See the file COPYING in the main directory of the source code
* for more details. */ * 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
#include "Graphs/gGraphView.h" #include "Graphs/gGraphView.h"
#include <QDir> #include <QDir>
@ -470,7 +484,7 @@ void gGraphView::popoutGraph()
qDebug() << "widget geometry" << newDockWidget->frameGeometry() << "title bar height" << titleBarHeight; qDebug() << "widget geometry" << newDockWidget->frameGeometry() << "title bar height" << titleBarHeight;
if (newDockHeight > screenHeight) { if (newDockHeight > screenHeight) {
QMessageBox::warning(nullptr, STR_MessageBox_Warning, QMessageBox::warning(nullptr, STR_MessageBox_Warning,
QObject::tr("The popout window is full. You should capture the existing\npopout window, delete it, then pop out this graph again.")); QObject::tr("The popout window is full. You should capture the existing\npopout window, delete it, then pop out this graph again."));
return; return;
} }
qDebug() << "dock height" << dock->height() << "popout graph height" << popout_graph->height(); qDebug() << "dock height" << dock->height() << "popout graph height" << popout_graph->height();
@ -1151,6 +1165,8 @@ void gGraphView::GetXBounds(qint64 &st, qint64 &et)
// Supplies time range to all graph objects in linked group, refreshing if requested // Supplies time range to all graph objects in linked group, refreshing if requested
void gGraphView::SetXBounds(qint64 minx, qint64 maxx, short group, bool refresh) void gGraphView::SetXBounds(qint64 minx, qint64 maxx, short group, bool refresh)
{ {
bool changed= (minx!=m_minx)||(maxx!=m_maxx);
for (auto & graph : m_graphs) { for (auto & graph : m_graphs) {
if ((graph->group() == group)) { if ((graph->group() == group)) {
graph->SetXBounds(minx, maxx); graph->SetXBounds(minx, maxx);
@ -1161,6 +1177,7 @@ void gGraphView::SetXBounds(qint64 minx, qint64 maxx, short group, bool refresh)
m_maxx = maxx; m_maxx = maxx;
if (refresh) { timedRedraw(0); } if (refresh) { timedRedraw(0); }
if (changed) emit XBoundsChanged(minx ,maxx);
} }
void gGraphView::updateScrollBar() void gGraphView::updateScrollBar()

View File

@ -707,6 +707,7 @@ class gGraphView
void updateCurrentTime(double); void updateCurrentTime(double);
void updateRange(double,double); void updateRange(double,double);
void GraphsChanged(); void GraphsChanged();
void XBoundsChanged(qint64 ,qint64);
public slots: public slots:
//! \brief Callback from the ScrollBar, to change scroll position //! \brief Callback from the ScrollBar, to change scroll position

View File

@ -7,6 +7,20 @@
* License. See the file COPYING in the main directory of the source code * License. See the file COPYING in the main directory of the source code
* for more details. */ * 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
#include <math.h> #include <math.h>
#include <QLabel> #include <QLabel>
#include <QDateTime> #include <QDateTime>
@ -104,13 +118,11 @@ void gSummaryChart::SetDay(Day *unused_day)
m_maxy = 20; m_maxy = 20;
m_empty = false; m_empty = false;
m_emptyPrev = true;
} }
//QMap<QDate, int> gSummaryChart::dayindex;
//QList<Day *> gSummaryChart::daylist;
int gSummaryChart::addCalc(ChannelID code, SummaryType type, QColor color) int gSummaryChart::addCalc(ChannelID code, SummaryType type, QColor color)
{ {
calcitems.append(SummaryCalcItem(code, type, color)); calcitems.append(SummaryCalcItem(code, type, color));
@ -565,7 +577,8 @@ void gSummaryChart::paint(QPainter &painter, gGraph &graph, const QRegion &regio
totaldays++; totaldays++;
if (!day) {// || !day->hasMachine(m_machtype)) { if (!day)
{
// lasty1 = rect.bottom(); // lasty1 = rect.bottom();
lastx1 += barw; lastx1 += barw;
it++; it++;
@ -675,9 +688,13 @@ void gSummaryChart::paint(QPainter &painter, gGraph &graph, const QRegion &regio
// This could be turning off graphs prematurely.. // This could be turning off graphs prematurely..
if (cache.size() == 0) { if (cache.size() == 0) {
m_empty = true; m_empty = true;
m_emptyPrev = true;
graph.graphView()->updateScale(); graph.graphView()->updateScale();
emit summaryChartEmpty(this,m_minx,m_maxx,true);
} else if (m_emptyPrev) {
m_emptyPrev = false;
emit summaryChartEmpty(this,m_minx,m_maxx,false);
} }
} }
@ -1019,8 +1036,8 @@ void gSessionTimesChart::paint(QPainter &painter, gGraph &graph, const QRegion &
totaldays++; totaldays++;
if (!day)
if (!day) { // || !day->hasMachine(m_machtype)) { {
// lasty1 = rect.bottom(); // lasty1 = rect.bottom();
lastx1 += barw; lastx1 += barw;
nousedays++; nousedays++;
@ -1091,6 +1108,7 @@ void gSessionTimesChart::paint(QPainter &painter, gGraph &graph, const QRegion &
afterDraw(painter, graph, rect); afterDraw(painter, graph, rect);
} }
//////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////
/// Total Time in Apnea Chart Stuff /// Total Time in Apnea Chart Stuff
//////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////

View File

@ -155,8 +155,9 @@ struct SummaryChartSlice {
// QBrush brush; // QBrush brush;
}; };
class gSummaryChart : public Layer class gSummaryChart : public QObject , public Layer
{ {
Q_OBJECT;
public: public:
gSummaryChart(QString label, MachineType machtype); gSummaryChart(QString label, MachineType machtype);
gSummaryChart(ChannelID code, MachineType machtype); gSummaryChart(ChannelID code, MachineType machtype);
@ -171,6 +172,9 @@ public:
//! \brief Returns true if no data was found for this day during SetDay //! \brief Returns true if no data was found for this day during SetDay
virtual bool isEmpty() { return m_empty; } virtual bool isEmpty() { return m_empty; }
//! \brief Allows chart to recalculate empty flag.
void reCalculate() {m_empty=false;};
virtual void populate(Day *, int idx); virtual void populate(Day *, int idx);
//! \brief Override to setup custom stuff before main loop //! \brief Override to setup custom stuff before main loop
@ -217,6 +221,8 @@ public:
layer->dayindex = dayindex; layer->dayindex = dayindex;
layer->daylist = daylist; layer->daylist = daylist;
} }
signals:
void summaryChartEmpty(gSummaryChart*,qint64,qint64,bool);
protected: protected:
//! \brief Key was pressed that effects this layer //! \brief Key was pressed that effects this layer
@ -234,6 +240,7 @@ protected:
QString m_label; QString m_label;
MachineType m_machtype; MachineType m_machtype;
bool m_empty; bool m_empty;
bool m_emptyPrev;
int hl_day; int hl_day;
int tz_offset; int tz_offset;
float tz_hours; float tz_hours;
@ -261,6 +268,7 @@ protected:
}; };
/*! \class gSessionTimesChart /*! \class gSessionTimesChart
\brief Displays a summary of session times \brief Displays a summary of session times
*/ */

View File

@ -7,15 +7,30 @@
* License. See the file COPYING in the main directory of the source code * License. See the file COPYING in the main directory of the source code
* for more details. */ * for more details. */
#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 DEBUGT qDebug()<<QDateTime::currentDateTime().time().toString("hh:mm:ss.zzz")
#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 <QCalendarWidget> #include <QCalendarWidget>
#include <QTextCharFormat> #include <QTextCharFormat>
//#include <QSystemLocale>
#include <QDebug> #include <QDebug>
#include <QDateTimeEdit> #include <QDateTimeEdit>
#include <QCalendarWidget> #include <QCalendarWidget>
#include <QFileDialog> #include <QFileDialog>
#include <QMessageBox> #include <QMessageBox>
//#include <QProgressBar>
#include "SleepLib/profiles.h" #include "SleepLib/profiles.h"
#include "overview.h" #include "overview.h"
@ -30,11 +45,27 @@
#include "mainwindow.h" #include "mainwindow.h"
extern MainWindow *mainwin; extern MainWindow *mainwin;
qint64 convertDateToTimeRtn(const QDate &date,int hours,int min,int sec) {
return QDateTime(date).addSecs(((hours*60+min)*60)+sec).toMSecsSinceEpoch();
}
qint64 convertDateToStartTime(const QDate &date) {
return convertDateToTimeRtn(date,0,10,0);
}
qint64 convertDateToEndTime(const QDate &date) {
return convertDateToTimeRtn(date,23,0,0);
}
QDate convertTimeToDate(qint64 time) {
return QDateTime::fromMSecsSinceEpoch(time).date();
}
Overview::Overview(QWidget *parent, gGraphView *shared) : Overview::Overview(QWidget *parent, gGraphView *shared) :
QWidget(parent), QWidget(parent),
ui(new Ui::Overview), ui(new Ui::Overview),
m_shared(shared) m_shared(shared)
{ {
chartsToBeMonitored.clear();
chartsEmpty.clear();;
ui->setupUi(this); ui->setupUi(this);
// Set Date controls locale to 4 digit years // Set Date controls locale to 4 digit years
@ -67,7 +98,6 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
// Connect the signals to update which days have CPAP data when the month is changed // Connect the signals to update which days have CPAP data when the month is changed
connect(ui->dateStart->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateStart_currentPageChanged(int, int))); 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))); connect(ui->dateEnd->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateEnd_currentPageChanged(int, int)));
QVBoxLayout *framelayout = new QVBoxLayout; QVBoxLayout *framelayout = new QVBoxLayout;
ui->graphArea->setLayout(framelayout); ui->graphArea->setLayout(framelayout);
@ -143,15 +173,18 @@ Overview::Overview(QWidget *parent, gGraphView *shared) :
connect(GraphView, SIGNAL(updateCurrentTime(double)), this, SLOT(on_LineCursorUpdate(double))); connect(GraphView, SIGNAL(updateCurrentTime(double)), this, SLOT(on_LineCursorUpdate(double)));
connect(GraphView, SIGNAL(updateRange(double,double)), this, SLOT(on_RangeUpdate(double,double))); connect(GraphView, SIGNAL(updateRange(double,double)), this, SLOT(on_RangeUpdate(double,double)));
connect(GraphView, SIGNAL(GraphsChanged()), this, SLOT(updateGraphCombo())); connect(GraphView, SIGNAL(GraphsChanged()), this, SLOT(updateGraphCombo()));
connect(GraphView, SIGNAL(XBoundsChanged(qint64 ,qint64)), this, SLOT(on_XBoundsChanged(qint64 ,qint64)));
} }
Overview::~Overview() Overview::~Overview()
{ {
disconnect(GraphView, SIGNAL(XBoundsChanged(qint64 ,qint64)), this, SLOT(on_XBoundsChanged(qint64 ,qint64)));
disconnect(GraphView, SIGNAL(GraphsChanged()), this, SLOT(updateGraphCombo())); disconnect(GraphView, SIGNAL(GraphsChanged()), this, SLOT(updateGraphCombo()));
disconnect(GraphView, SIGNAL(updateRange(double,double)), this, SLOT(on_RangeUpdate(double,double))); disconnect(GraphView, SIGNAL(updateRange(double,double)), this, SLOT(on_RangeUpdate(double,double)));
disconnect(GraphView, SIGNAL(updateCurrentTime(double)), this, SLOT(on_LineCursorUpdate(double))); disconnect(GraphView, SIGNAL(updateCurrentTime(double)), this, SLOT(on_LineCursorUpdate(double)));
disconnect(ui->dateEnd->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateEnd_currentPageChanged(int, int))); disconnect(ui->dateEnd->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateEnd_currentPageChanged(int, int)));
disconnect(ui->dateStart->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateStart_currentPageChanged(int, int))); disconnect(ui->dateStart->calendarWidget(), SIGNAL(currentPageChanged(int, int)), this, SLOT(dateStart_currentPageChanged(int, int)));
disconnectgSummaryCharts() ;
// Save graph orders and pin status, etc... // Save graph orders and pin status, etc...
GraphView->SaveSettings("Overview");//no trans GraphView->SaveSettings("Overview");//no trans
@ -166,18 +199,63 @@ void Overview::ResetFont()
dateLabel->setFont(font); dateLabel->setFont(font);
} }
void Overview::connectgSummaryCharts()
{
for (auto it = chartsToBeMonitored.begin(); it != chartsToBeMonitored.end(); ++it) {
gSummaryChart* sc =it.key();
connect(sc, SIGNAL(summaryChartEmpty(gSummaryChart*,qint64,qint64,bool)), this, SLOT(on_summaryChartEmpty(gSummaryChart*,qint64,qint64,bool)));
}
}
void Overview::disconnectgSummaryCharts()
{
for (auto it = chartsToBeMonitored.begin(); it != chartsToBeMonitored.end(); ++it) {
gSummaryChart* sc =it.key();
disconnect(sc, SIGNAL(summaryChartEmpty(gSummaryChart*,qint64,qint64,bool)), this, SLOT(on_summaryChartEmpty(gSummaryChart*,qint64,qint64,bool)));
}
chartsToBeMonitored.clear();
chartsEmpty.clear();;
}
void Overview::on_summaryChartEmpty(gSummaryChart*sc,qint64 firstI,qint64 lastI,bool empty)
{
auto it = chartsToBeMonitored.find(sc);
if (it==chartsToBeMonitored.end()) {
return;
}
gGraph* graph=it.value();
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);
};
// Create all the graphs for the Overview page // Create all the graphs for the Overview page
void Overview::CreateAllGraphs() { void Overview::CreateAllGraphs() {
/////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////
// Add all the graphs // Add all the graphs
// Process is to createGraph() to make the graph object, then add a layer // Process is to createGraph() to make the graph object, then add a layer
// that provides the contents for that graph. // that provides the contents for that graph.
/////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////
if (chartsToBeMonitored.size()>0) {
disconnectgSummaryCharts();
}
chartsEmpty.clear();;
chartsToBeMonitored.clear();;
// Add graphs that are always included // Add graphs that are always included
ChannelID ahicode = p_profile->general->calculateRDI() ? CPAP_RDI : CPAP_AHI; ChannelID ahicode = p_profile->general->calculateRDI() ? CPAP_RDI : CPAP_AHI;
if (ahicode == CPAP_RDI) { if (ahicode == CPAP_RDI) {
AHI = createGraph("AHIBreakdown", STR_TR_RDI, tr("Respiratory\nDisturbance\nIndex")); AHI = createGraph("AHIBreakdown", STR_TR_RDI, tr("Respiratory\nDisturbance\nIndex"));
} else { } else {
@ -186,9 +264,11 @@ void Overview::CreateAllGraphs() {
ahi = new gAHIChart(); ahi = new gAHIChart();
AHI->AddLayer(ahi); AHI->AddLayer(ahi);
//chartsToBeMonitored.insert(ahi,AHI);
UC = createGraph(STR_GRAPH_Usage, tr("Usage"), tr("Usage\n(hours)")); UC = createGraph(STR_GRAPH_Usage, tr("Usage"), tr("Usage\n(hours)"));
UC->AddLayer(uc = new gUsageChart()); UC->AddLayer(uc = new gUsageChart());
//chartsToBeMonitored.insert(uc,UC);
STG = createGraph("New Session", tr("Session Times"), tr("Session Times"), YT_Time); STG = createGraph("New Session", tr("Session Times"), tr("Session Times"), YT_Time);
stg = new gSessionTimesChart(); stg = new gSessionTimesChart();
@ -197,10 +277,12 @@ void Overview::CreateAllGraphs() {
PR = createGraph("Pressure Settings", STR_TR_Pressure, STR_TR_Pressure + "\n(" + STR_UNIT_CMH2O + ")"); PR = createGraph("Pressure Settings", STR_TR_Pressure, STR_TR_Pressure + "\n(" + STR_UNIT_CMH2O + ")");
pres = new gPressureChart(); pres = new gPressureChart();
PR->AddLayer(pres); PR->AddLayer(pres);
//chartsToBeMonitored.insert(pres,PR);
TTIA = createGraph("TTIA", tr("Total Time in Apnea"), tr("Total Time in Apnea\n(Minutes)")); TTIA = createGraph("TTIA", tr("Total Time in Apnea"), tr("Total Time in Apnea\n(Minutes)"));
ttia = new gTTIAChart(); ttia = new gTTIAChart();
TTIA->AddLayer(ttia); TTIA->AddLayer(ttia);
//chartsToBeMonitored.insert(ttia,TTIA);
// Add graphs for all channels that have been marked in Preferences Dialog as wanting a graph // Add graphs for all channels that have been marked in Preferences Dialog as wanting a graph
QHash<ChannelID, schema::Channel *>::iterator chit; QHash<ChannelID, schema::Channel *>::iterator chit;
@ -212,43 +294,59 @@ void Overview::CreateAllGraphs() {
ChannelID code = chan->id(); ChannelID code = chan->id();
QString name = chan->fullname(); QString name = chan->fullname();
if (name.length() > 16) name = chan->label(); if (name.length() > 16) name = chan->label();
// qDebug() << "Channel" << name << "type" << chan->type() << "machine type" << chan->machtype();
gGraph *G = createGraph(chan->code(), name, chan->description()); gGraph *G = createGraph(chan->code(), name, chan->description());
gSummaryChart * sc = nullptr;
if ((chan->type() == schema::FLAG) || (chan->type() == schema::MINOR_FLAG)) { if ((chan->type() == schema::FLAG) || (chan->type() == schema::MINOR_FLAG)) {
gSummaryChart * sc = new gSummaryChart(chan->code(), chan->machtype()); // gts was MT_CPAP sc = new gSummaryChart(chan->code(), chan->machtype()); // gts was MT_CPAP
sc->addCalc(code, ST_CPH, schema::channel[code].defaultColor()); sc->addCalc(code, ST_CPH, schema::channel[code].defaultColor());
G->AddLayer(sc); G->AddLayer(sc);
chartsToBeMonitored.insert(sc,G);
} else if (chan->type() == schema::SPAN) { } else if (chan->type() == schema::SPAN) {
gSummaryChart * sc = new gSummaryChart(chan->code(), MT_CPAP); sc = new gSummaryChart(chan->code(), MT_CPAP);
sc->addCalc(code, ST_SPH, schema::channel[code].defaultColor()); sc->addCalc(code, ST_SPH, schema::channel[code].defaultColor());
G->AddLayer(sc); G->AddLayer(sc);
chartsToBeMonitored.insert(sc,G);
} else if (chan->type() == schema::WAVEFORM) { } else if (chan->type() == schema::WAVEFORM) {
G->AddLayer(new gSummaryChart(code, chan->machtype())); sc= new gSummaryChart(code, chan->machtype());
G->AddLayer(sc);
chartsToBeMonitored.insert(sc,G);
} else if (chan->type() == schema::UNKNOWN) { } else if (chan->type() == schema::UNKNOWN) {
gSummaryChart * sc = new gSummaryChart(chan->code(), MT_CPAP); sc = new gSummaryChart(chan->code(), MT_CPAP);
sc->addCalc(code, ST_CPH, schema::channel[code].defaultColor()); sc->addCalc(code, ST_CPH, schema::channel[code].defaultColor());
G->AddLayer(sc); G->AddLayer(sc);
chartsToBeMonitored.insert(sc,G);
}
if (sc== nullptr) {
//DEBUGF << "Channel" << name << "type" << chan->type() << "machine type" << chan->machtype() << "IGNORED";
} else {
sc ->reCalculate();
//DEBUGF << "Channel" << name << "type" << chan->type() << "machine type" << chan->machtype() << OO(Empty,sc->isEmpty());
} }
} // if showInOverview() } // if showInOverview()
} // for chit } // for chit
// Note The following don not use gSummaryChart. They use SummaryChart instead. and can not be monitored.
WEIGHT = createGraph(STR_GRAPH_Weight, STR_TR_Weight, STR_TR_Weight, YT_Weight); WEIGHT = createGraph(STR_GRAPH_Weight, STR_TR_Weight, STR_TR_Weight, YT_Weight);
weight = new SummaryChart("Weight", GT_LINE); weight = new SummaryChart("Weight", GT_LINE);
weight->setMachineType(MT_JOURNAL); weight->setMachineType(MT_JOURNAL);
weight->addSlice(Journal_Weight, QColor("black"), ST_SETAVG); weight->addSlice(Journal_Weight, QColor("black"), ST_SETAVG);
WEIGHT->AddLayer(weight); WEIGHT->AddLayer(weight);
BMI = createGraph(STR_GRAPH_BMI, STR_TR_BMI, tr("Body\nMass\nIndex")); BMI = createGraph(STR_GRAPH_BMI, STR_TR_BMI, tr("Body\nMass\nIndex"));
bmi = new SummaryChart("BMI", GT_LINE); bmi = new SummaryChart("BMI", GT_LINE);
bmi->setMachineType(MT_JOURNAL); bmi->setMachineType(MT_JOURNAL);
bmi->addSlice(Journal_BMI, QColor("black"), ST_SETAVG); bmi->addSlice(Journal_BMI, QColor("black"), ST_SETAVG);
BMI->AddLayer(bmi); BMI->AddLayer(bmi);
ZOMBIE = createGraph(STR_GRAPH_Zombie, STR_TR_Zombie, tr("How you felt\n(0-10)")); ZOMBIE = createGraph(STR_GRAPH_Zombie, STR_TR_Zombie, tr("How you felt\n(0-10)"));
zombie = new SummaryChart("Zombie", GT_LINE); zombie = new SummaryChart("Zombie", GT_LINE);
zombie->setMachineType(MT_JOURNAL); zombie->setMachineType(MT_JOURNAL);
zombie->addSlice(Journal_ZombieMeter, QColor("black"), ST_SETAVG); zombie->addSlice(Journal_ZombieMeter, QColor("black"), ST_SETAVG);
ZOMBIE->AddLayer(zombie); ZOMBIE->AddLayer(zombie);
connectgSummaryCharts();
} }
// Recalculates Overview chart info // Recalculates Overview chart info
void Overview::RebuildGraphs(bool reset) void Overview::RebuildGraphs(bool reset)
{ {
@ -257,14 +355,16 @@ void Overview::RebuildGraphs(bool reset)
GraphView->GetXBounds(minx, maxx); GraphView->GetXBounds(minx, maxx);
} }
minRangeStartDate=p_profile->LastDay(MT_CPAP);
maxRangeEndDate=minRangeStartDate.addDays(-1); // force a range change;
disconnectgSummaryCharts() ;
GraphView->trashGraphs(true); // Remove all existing graphs GraphView->trashGraphs(true); // Remove all existing graphs
CreateAllGraphs(); CreateAllGraphs();
if (reset) { if (reset) {
GraphView->resetLayout(); GraphView->resetLayout();
GraphView->setDay(nullptr); GraphView->setDay(nullptr);
GraphView->SetXBounds(minx, maxx, 0, false); SetXBounds(minx, maxx, 0, false);
GraphView->resetLayout(); GraphView->resetLayout();
updateGraphCombo(); updateGraphCombo();
} }
@ -374,35 +474,12 @@ void Overview::updateGraphCombo()
updateCube(); updateCube();
} }
#if 0
void Overview::ResetGraphs()
{
QDate start = ui->dateStart->date();
QDate end = ui->dateEnd->date();
GraphView->setDay(nullptr);
updateCube();
if (start.isValid() && end.isValid()) {
setRange(start, end);
}
}
void Overview::ResetGraph(QString name)
{
gGraph *g = GraphView->findGraph(name);
if (!g) { return; }
g->setDay(nullptr);
GraphView->redraw();
}
#endif
void Overview::RedrawGraphs() void Overview::RedrawGraphs()
{ {
GraphView->redraw(); GraphView->redraw();
} }
// Updates calendar format and profile data.
void Overview::UpdateCalendarDay(QDateEdit *dateedit, QDate date) void Overview::UpdateCalendarDay(QDateEdit *dateedit, QDate date)
{ {
QCalendarWidget *calendar = dateedit->calendarWidget(); QCalendarWidget *calendar = dateedit->calendarWidget();
@ -433,6 +510,71 @@ void Overview::UpdateCalendarDay(QDateEdit *dateedit, QDate date)
calendar->setHorizontalHeaderFormat(QCalendarWidget::ShortDayNames); calendar->setHorizontalHeaderFormat(QCalendarWidget::ShortDayNames);
} }
void Overview::SetXBounds(qint64 start, qint64 end, short group , bool refresh )
{
GraphView->SetXBounds( start , end ,group,refresh);
}
void Overview::on_XBoundsChanged(qint64 start,qint64 end)
{
displayStartDate = convertTimeToDate(start);
displayEndDate = convertTimeToDate(end);
bool largerRange=false;
if (displayStartDate>maxRangeEndDate || minRangeStartDate>displayEndDate) {
// have non-overlaping ranges
// 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;
maxRangeEndDate = displayEndDate;
} 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++) {
gSummaryChart* sc = it.key();
bool empty=sc->isEmpty();
if (empty) {
sc ->reCalculate();
GraphView->updateScale();
GraphView->redraw();
GraphView->timedRedraw(150);
}
}
chartsEmpty.clear();
updateGraphCombo();
}
}
void Overview::dateStart_currentPageChanged(int year, int month) void Overview::dateStart_currentPageChanged(int year, int month)
{ {
QDate d(year, month, 1); QDate d(year, month, 1);
@ -443,6 +585,7 @@ void Overview::dateStart_currentPageChanged(int year, int month)
UpdateCalendarDay(ui->dateStart, d); UpdateCalendarDay(ui->dateStart, d);
} }
} }
void Overview::dateEnd_currentPageChanged(int year, int month) void Overview::dateEnd_currentPageChanged(int year, int month)
{ {
QDate d(year, month, 1); QDate d(year, month, 1);
@ -459,7 +602,7 @@ void Overview::on_dateEnd_dateChanged(const QDate &date)
{ {
qint64 d1 = qint64(QDateTime(ui->dateStart->date(), QTime(0, 10, 0)/*, Qt::UTC*/).toTime_t()) * 1000L; qint64 d1 = qint64(QDateTime(ui->dateStart->date(), QTime(0, 10, 0)/*, Qt::UTC*/).toTime_t()) * 1000L;
qint64 d2 = qint64(QDateTime(date, QTime(23, 0, 0)/*, Qt::UTC*/).toTime_t()) * 1000L; qint64 d2 = qint64(QDateTime(date, QTime(23, 0, 0)/*, Qt::UTC*/).toTime_t()) * 1000L;
GraphView->SetXBounds(d1, d2); SetXBounds(d1, d2);
ui->dateStart->setMaximumDate(date); ui->dateStart->setMaximumDate(date);
if (customMode) { if (customMode) {
p_profile->general->setCustomOverviewRangeEnd(date); p_profile->general->setCustomOverviewRangeEnd(date);
@ -470,12 +613,11 @@ void Overview::on_dateStart_dateChanged(const QDate &date)
{ {
qint64 d1 = qint64(QDateTime(date, QTime(0, 10, 0)/*, Qt::UTC*/).toTime_t()) * 1000L; qint64 d1 = qint64(QDateTime(date, QTime(0, 10, 0)/*, Qt::UTC*/).toTime_t()) * 1000L;
qint64 d2 = qint64(QDateTime(ui->dateEnd->date(), QTime(23, 0, 0)/*, Qt::UTC*/).toTime_t()) * 1000L; qint64 d2 = qint64(QDateTime(ui->dateEnd->date(), QTime(23, 0, 0)/*, Qt::UTC*/).toTime_t()) * 1000L;
GraphView->SetXBounds(d1, d2); SetXBounds(d1, d2);
ui->dateEnd->setMinimumDate(date); ui->dateEnd->setMinimumDate(date);
if (customMode) { if (customMode) {
p_profile->general->setCustomOverviewRangeStart(date); p_profile->general->setCustomOverviewRangeStart(date);
} }
} }
// Zoom to 100% button clicked or called back from 100% zoom in popup menu // Zoom to 100% button clicked or called back from 100% zoom in popup menu
@ -483,7 +625,7 @@ void Overview::on_zoomButton_clicked()
{ {
qint64 d1 = qint64(QDateTime(ui->dateStart->date(), QTime(0, 10, 0)/*, Qt::UTC*/).toTime_t()) * 1000L; // GTS why UTC? qint64 d1 = qint64(QDateTime(ui->dateStart->date(), QTime(0, 10, 0)/*, Qt::UTC*/).toTime_t()) * 1000L; // GTS why UTC?
qint64 d2 = qint64(QDateTime(ui->dateEnd->date(), QTime(23, 0, 0)/*, Qt::UTC*/).toTime_t()) * 1000L; // Interesting: start date set to 10 min after midnight, ending at 11 pm qint64 d2 = qint64(QDateTime(ui->dateEnd->date(), QTime(23, 0, 0)/*, Qt::UTC*/).toTime_t()) * 1000L; // Interesting: start date set to 10 min after midnight, ending at 11 pm
GraphView->SetXBounds(d1, d2); SetXBounds(d1, d2);
} }
void Overview::ResetGraphLayout() void Overview::ResetGraphLayout()
@ -501,8 +643,14 @@ void Overview::ResetGraphOrder(int type)
// Process new range selection from combo button // Process new range selection from combo button
void Overview::on_rangeCombo_activated(int index) void Overview::on_rangeCombo_activated(int index)
{ {
// Block signal so that graphs are not updated for these.
ui->dateEnd->blockSignals(true);
ui->dateStart->blockSignals(true);
ui->dateStart->setMinimumDate(p_profile->FirstDay()); // first and last dates for ANY machine type ui->dateStart->setMinimumDate(p_profile->FirstDay()); // first and last dates for ANY machine type
ui->dateEnd->setMaximumDate(p_profile->LastDay()); ui->dateEnd->setMaximumDate(p_profile->LastDay());
// these signals will be reenabled in setRange.
//ui->dateEnd->blockSignals(false);
//ui->dateStart->blockSignals(false);
// Exclude Journal in calculating the last day // Exclude Journal in calculating the last day
QDate end = p_profile->LastDay(MT_CPAP); QDate end = p_profile->LastDay(MT_CPAP);
@ -592,21 +740,27 @@ void Overview::on_rangeCombo_activated(int index)
delete progress; delete progress;
// first and last dates for ANY machine type // first and last dates for ANY machine type
//uiStartDate=start;
//uiEndDate=end;
setRange(start, end); setRange(start, end);
} }
// Saves dates in UI, clicks zoom button, and updates combo box // Saves dates in UI, clicks zoom button, and updates combo box
void Overview::setRange(QDate start, QDate end) // 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*/)
{ {
ui->dateEnd->blockSignals(true); ui->dateEnd->blockSignals(true);
ui->dateStart->blockSignals(true); ui->dateStart->blockSignals(true);
ui->dateStart->setMaximumDate(end); ui->dateStart->setMaximumDate(end);
ui->dateEnd->setMinimumDate(start); ui->dateEnd->setMinimumDate(start);
ui->dateStart->setDate(start); ui->dateStart->setDate(start);
ui->dateEnd->setDate(end); ui->dateEnd->setDate(end);
ui->dateEnd->blockSignals(false); ui->dateEnd->blockSignals(false);
ui->dateStart->blockSignals(false); ui->dateStart->blockSignals(false);
this->on_zoomButton_clicked(); // Click on zoom-out to 100% button if (updateGraphs) this->on_zoomButton_clicked(); // Click on zoom-out to 100% button
updateGraphCombo(); updateGraphCombo();
} }
@ -631,7 +785,7 @@ void Overview::on_graphCombo_activated(int index)
g = GraphView->findGraphTitle(s); g = GraphView->findGraphTitle(s);
g->setVisible(b); g->setVisible(b);
} }
ui->graphCombo->setCurrentIndex(0); ui->graphCombo->setCurrentIndex(0);
updateCube(); updateCube();
setGraphText(); setGraphText();

View File

@ -65,7 +65,7 @@ class Overview : public QWidget
void RedrawGraphs(); void RedrawGraphs();
//! \brief Sets the currently selected date range of the overview display //! \brief Sets the currently selected date range of the overview display
void setRange(QDate start, QDate end); void setRange(QDate& start, QDate& end,bool updateGraphs=true);
/*! \brief Create an overview graph, adding it to the overview gGraphView object /*! \brief Create an overview graph, adding it to the overview gGraphView object
\param QString name The title of the graph \param QString name The title of the graph
@ -78,11 +78,6 @@ class Overview : public QWidget
gSummaryChart * stg, *uc, *ahi, * pres, *lk, *npb, *rr, *mv, *tv, *nll, *sn, *ttia; gSummaryChart * stg, *uc, *ahi, * pres, *lk, *npb, *rr, *mv, *tv, *nll, *sn, *ttia;
//! \brief List of SummaryCharts shown on the overview page
QVector<SummaryChart *> OverviewCharts;
//void ResetGraph(QString name);
void RebuildGraphs(bool reset = true); void RebuildGraphs(bool reset = true);
public slots: public slots:
@ -93,6 +88,8 @@ class Overview : public QWidget
private slots: private slots:
void updateGraphCombo(); void updateGraphCombo();
void on_XBoundsChanged(qint64 ,qint64);
void on_summaryChartEmpty(gSummaryChart*,qint64,qint64,bool);
//! \brief Resets the graph view because the Start date has been changed //! \brief Resets the graph view because the Start date has been changed
void on_dateStart_dateChanged(const QDate &date); void on_dateStart_dateChanged(const QDate &date);
@ -120,6 +117,7 @@ class Overview : public QWidget
private: private:
void CreateAllGraphs(); void CreateAllGraphs();
void timedUpdateOverview(int ms=0);
Ui::Overview *ui; Ui::Overview *ui;
gGraphView *GraphView; gGraphView *GraphView;
@ -139,6 +137,23 @@ class Overview : public QWidget
Day *day; // dummy in this case Day *day; // dummy in this case
bool checkRangeChanged(QDate& first, QDate& last);
void connectgSummaryCharts() ;
void disconnectgSummaryCharts() ;
void SetXBounds(qint64 minx, qint64 maxx, short group = 0, bool refresh = true);
// Start and of dates of the current graph display
QDate displayStartDate;
QDate displayEndDate;
// min / max dates of the graph Range
QDate minRangeStartDate;
QDate maxRangeEndDate;
QHash<gSummaryChart*,gGraph*> chartsToBeMonitored;
QHash<gSummaryChart*,gGraph* > chartsEmpty;
}; };
#endif // OVERVIEW_H #endif // OVERVIEW_H