OSCAR-code/oscar/Graphs/gGraphView.cpp

3646 lines
104 KiB
C++
Raw Normal View History

2018-06-12 16:51:58 +00:00
/* gGraphView Implementation
*
* 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. */
#include "Graphs/gGraphView.h"
#include <QDir>
#include <QFontMetrics>
#include <QLabel>
#include <QPixmapCache>
#include <QTimer>
#include <QFontMetrics>
#include <QWidgetAction>
#include <QGridLayout>
#include <QVBoxLayout>
#include <QDockWidget>
#include <QMainWindow>
# include <QWindow>
#ifdef DEBUG_EFFICIENCY
# include <QElapsedTimer>
#endif
#include <cmath>
#include "mainwindow.h"
#include "Graphs/glcommon.h"
#include "Graphs/gLineChart.h"
#include "Graphs/gSummaryChart.h"
#include "Graphs/gSessionTimesChart.h"
#include "Graphs/gYAxis.h"
#include "Graphs/gFlagsLine.h"
#include "SleepLib/profiles.h"
#include "overview.h"
2015-08-27 09:46:05 +00:00
extern MainWindow *mainwin;
2014-08-17 12:56:05 +00:00
#include <QApplication>
MyLabel::MyLabel(QWidget * parent)
: QWidget(parent) {
2014-08-17 12:56:05 +00:00
m_font = QApplication::font();
time.start();
2014-08-17 12:56:05 +00:00
}
MyLabel::~MyLabel()
{
}
void MyLabel::setText(QString text) {
m_text = text;
update();
2014-08-17 12:56:05 +00:00
}
void MyLabel::setFont(QFont & font)
{
m_font=font;
}
void MyLabel::doRedraw()
{
update();
}
2014-08-17 12:56:05 +00:00
void MyLabel::setAlignment(Qt::Alignment alignment) {
m_alignment = alignment;
doRedraw();
2014-08-17 12:56:05 +00:00
}
void MyLabel::paintEvent(QPaintEvent * /*event*/)
2014-08-17 12:56:05 +00:00
{
QPainter painter(this);
painter.setFont(m_font);
painter.drawText(rect(), m_alignment, m_text);
2014-08-17 12:56:05 +00:00
}
gToolTip::gToolTip(gGraphView *graphview)
: m_graphview(graphview)
2011-09-05 10:28:41 +00:00
{
m_pos.setX(0);
m_pos.setY(0);
m_visible = false;
m_alignment = TT_AlignCenter;
m_spacer = 8; // pixels around text area
timer = new QTimer(graphview);
connect(timer, SIGNAL(timeout()), SLOT(timerDone()));
2011-09-05 10:28:41 +00:00
}
gToolTip::~gToolTip()
{
disconnect(timer, 0, 0, 0);
2011-09-05 10:28:41 +00:00
delete timer;
}
2011-12-05 15:03:16 +00:00
//void gToolTip::calcSize(QString text,int &w, int &h)
//{
/*GetTextExtent(text,w,h);
w+=m_spacer*2;
h+=m_spacer*2; */
2011-12-05 15:03:16 +00:00
//}
2011-09-05 10:28:41 +00:00
void gToolTip::display(QString text, int x, int y, ToolTipAlignment align, int timeout)
2011-09-05 10:28:41 +00:00
{
if (timeout <= 0) {
2018-04-22 12:06:48 +00:00
timeout = AppSetting->tooltipTimeout();
}
m_alignment = align;
m_text = text;
m_visible = true;
2011-09-05 10:28:41 +00:00
// TODO: split multiline here
//calcSize(m_text,tw,th);
m_pos.setX(x);
m_pos.setY(y);
//tw+=m_spacer*2;
//th+=m_spacer*2;
2011-09-18 14:43:15 +00:00
//th*=2;
2011-09-05 10:28:41 +00:00
if (timer->isActive()) {
timer->stop();
}
2011-09-05 10:28:41 +00:00
timer->setSingleShot(true);
timer->start(timeout);
m_invalidate = true;
2011-09-05 10:28:41 +00:00
}
void gToolTip::cancel()
{
m_visible = false;
2011-09-05 10:28:41 +00:00
timer->stop();
}
2014-05-07 19:52:59 +00:00
void gToolTip::paint(QPainter &painter) //actually paints it.
2011-09-05 10:28:41 +00:00
{
if (!m_visible) { return; }
int x = m_pos.x();
int y = m_pos.y();
2011-09-05 10:28:41 +00:00
QRect rect(x, y, 0, 0);
painter.setFont(*defaultfont);
rect = painter.boundingRect(rect, Qt::AlignCenter, m_text);
int w = rect.width() + m_spacer * 2;
int xx = rect.x() - m_spacer;
if (xx < 0) { xx = 0; }
rect.setLeft(xx);
rect.setTop(rect.y() - 15);
rect.setWidth(w);
int z = rect.x() + rect.width();
if (z > m_graphview->width() - 10) {
rect.setLeft(m_graphview->width() - 2 - rect.width());
rect.setRight(m_graphview->width() - 2);
}
int h = rect.height();
if (rect.y() < 0) {
rect.setY(0);
rect.setHeight(h);
}
if (m_alignment == TT_AlignRight) {
rect.moveTopRight(m_pos);
if ((x-w) < 0) {
rect.moveLeft(0);
}
} else if (m_alignment == TT_AlignLeft) {
rect.moveTopLeft(m_pos);
}
2018-05-05 09:53:52 +00:00
int bot = rect.bottom() - m_graphview->height();
if (bot > 0) {
rect.setTop(rect.top()-bot);
rect.setBottom(m_graphview->height());
}
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.setBrush(Qt::black);
painter.setFont(*defaultfont);
painter.drawText(rect, Qt::AlignCenter, m_text);
}
void gToolTip::timerDone()
{
m_visible = false;
m_graphview->redraw();
m_graphview->resetMouse();
}
#ifdef ENABLE_THREADED_DRAWING
gThread::gThread(gGraphView *g)
{
graphview = g;
mutex.lock();
}
gThread::~gThread()
{
if (isRunning()) {
m_running = false;
mutex.unlock();
wait();
terminate();
}
}
void gThread::run()
{
m_running = true;
gGraph *g;
while (m_running) {
mutex.lock();
//mutex.unlock();
if (!m_running) { break; }
do {
g = graphview->popGraph();
if (g) {
g->paint(QRegion(g->m_lastbounds));
//int i=0;
} else {
//mutex.lock();
graphview->masterlock->release(1); // This thread gives up for now..
}
} while (g);
}
2011-09-01 07:12:25 +00:00
}
#endif // ENABLE_THREADED_DRAWING
void gGraphView::queGraph(gGraph *g, int left, int top, int width, int height)
2011-09-01 07:12:25 +00:00
{
g->m_rect = QRect(left, top, width, height);
#ifdef ENABLED_THREADED_DRAWING
dl_mutex.lock();
#endif
m_drawlist.push_back(g);
#ifdef ENABLED_THREADED_DRAWING
dl_mutex.unlock();
#endif
2011-09-01 07:12:25 +00:00
}
2014-05-07 19:52:59 +00:00
void gGraphView::trashGraphs(bool destroy)
2011-09-03 12:59:08 +00:00
{
if (destroy) {
for (auto & graph : m_graphs) {
delete graph;
}
}
2014-05-07 19:52:59 +00:00
// Don't actually want to delete them here.. we are just borrowing the graphs
m_graphs.clear();
m_graphsbyname.clear();
2011-09-03 12:59:08 +00:00
}
2014-05-07 19:52:59 +00:00
// Take the next graph to render from the drawing list
gGraph *gGraphView::popGraph()
{
gGraph *g;
#ifdef ENABLED_THREADED_DRAWING
dl_mutex.lock();
#endif
if (!m_drawlist.isEmpty()) {
g = m_drawlist.at(0);
m_drawlist.pop_front();
} else { g = nullptr; }
#ifdef ENABLED_THREADED_DRAWING
dl_mutex.unlock();
#endif
return g;
}
gGraphView::gGraphView(QWidget *parent, gGraphView *shared, QWidget *caller)
#ifdef BROKEN_OPENGL_BUILD
: QWidget(parent),
#elif QT_VERSION < QT_VERSION_CHECK(5,4,0)
: QGLWidget(QGLFormat(QGL::DoubleBuffer | QGL::DirectRendering | QGL::HasOverlay | QGL::Rgba),parent,shared),
#else
:QOpenGLWidget(parent),
#endif
2014-08-11 04:52:14 +00:00
m_offsetY(0), m_offsetX(0), m_scaleY(0.0), m_scrollbar(nullptr)
{
this->caller = caller;
2015-08-27 09:46:05 +00:00
// this->grabGesture(Qt::SwipeGesture);
// this->grabGesture(Qt::PanGesture);
// this->grabGesture(Qt::TapGesture);
// this->grabGesture(Qt::TapAndHoldGesture);
// this->grabGesture(Qt::CustomGesture);
this->grabGesture(Qt::PinchGesture);
this->setAttribute(Qt::WA_AcceptTouchEvents);
// this->setAttribute(Qt::WA_TouchPadAcceptSingleTouchEvents);
m_shared = shared;
m_sizer_index = m_graph_index = 0;
m_metaselect = m_button_down = m_graph_dragging = m_sizer_dragging = false;
m_lastypos = m_lastxpos = 0;
m_horiz_travel = 0;
m_minx = m_maxx = 0;
m_day = nullptr;
m_selected_graph = nullptr;
2014-05-25 07:07:08 +00:00
m_scrollbar = nullptr;
m_point_released = m_point_clicked = QPoint(0,0);
m_showAuthorMessage = true;
horizScrollTime.start();
vertScrollTime.start();
this->setMouseTracking(true);
m_emptytext = STR_Empty_NoData;
m_emptyimage = QPixmap(":/icons/logo-md.png");
InitGraphGlobals(); // FIXME: sstangl: handle error return.
#ifdef ENABLE_THREADED_DRAWING
m_idealthreads = QThread::idealThreadCount();
if (m_idealthreads <= 0) { m_idealthreads = 1; }
masterlock = new QSemaphore(m_idealthreads);
#endif
m_tooltip = new gToolTip(this);
2011-09-07 14:26:00 +00:00
/*for (int i=0;i<m_idealthreads;i++) {
gThread * gt=new gThread(this);
m_threads.push_back(gt);
2011-09-07 14:26:00 +00:00
//gt->start();
}*/
2011-09-07 09:15:33 +00:00
setFocusPolicy(Qt::StrongFocus);
m_showsplitter = true;
timer = new QTimer(this);
connect(timer, SIGNAL(timeout()), SLOT(refreshTimeout()));
print_scaleY = print_scaleX = 1.0;
2011-12-19 18:49:35 +00:00
redrawtimer = new QTimer(this);
connect(redrawtimer, SIGNAL(timeout()), SLOT(redraw()));
2011-12-20 05:51:22 +00:00
m_fadingOut = false;
m_fadingIn = false;
m_inAnimation = false;
m_limbo = false;
m_fadedir = false;
m_blockUpdates = false;
2018-04-22 12:06:48 +00:00
use_pixmap_cache = AppSetting->usePixmapCaching();
2013-11-04 06:49:48 +00:00
2014-08-17 12:56:05 +00:00
pin_graph = nullptr;
popout_graph = nullptr;
// pixmapcache.setCacheLimit(10240*2);
m_dpr = devicePixelRatio();
m_dpr = 1; // meh???
#ifndef BROKEN_OPENGL_BUILD
setAutoFillBackground(false);
#if QT_VERSION < QT_VERSION_CHECK(5,4,0)
// happens no matter what in 5.4+
setAutoBufferSwap(false);
#endif
#endif
2014-08-17 12:56:05 +00:00
context_menu = new QMenu(this);
pin_action = context_menu->addAction(QString(), this, SLOT(togglePin()));
pin_icon = QPixmap(":/icons/pushpin.png");
2014-08-27 13:42:10 +00:00
popout_action = context_menu->addAction(QObject::tr("Pop out Graph"), this, SLOT(popoutGraph()));
2014-08-27 13:42:10 +00:00
snap_action = context_menu->addAction(QString(), this, SLOT(onSnapshotGraphToggle()));
2014-08-17 12:56:05 +00:00
context_menu->addSeparator();
2014-08-27 15:00:55 +00:00
zoom100_action = context_menu->addAction(tr("100% zoom level"), this, SLOT(resetZoom()));
if (caller)
zoom100_action->setToolTip(tr("Restore X-axis zoom to 100% to view entire selected period."));
else
zoom100_action->setToolTip(tr("Restore X-axis zoom to 100% to view entire day's data."));
2014-08-27 15:00:55 +00:00
QAction * action = context_menu->addAction(tr("Reset Graph Layout"), this, SLOT(resetLayout()));
action->setToolTip(tr("Resets all graphs to a uniform height and default order."));
2014-08-17 12:56:05 +00:00
context_menu->addSeparator();
limits_menu = context_menu->addMenu(tr("Y-Axis"));
2014-08-17 12:56:05 +00:00
plots_menu = context_menu->addMenu(tr("Plots"));
connect(plots_menu, SIGNAL(triggered(QAction*)), this, SLOT(onPlotsClicked(QAction*)));
2014-08-20 11:30:42 +00:00
// overlay_menu = context_menu->addMenu("Overlays");
cpap_menu = context_menu->addMenu(tr("CPAP Overlays"));
connect(cpap_menu, SIGNAL(triggered(QAction*)), this, SLOT(onOverlaysClicked(QAction*)));
oximeter_menu = context_menu->addMenu(tr("Oximeter Overlays"));
connect(oximeter_menu, SIGNAL(triggered(QAction*)), this, SLOT(onOverlaysClicked(QAction*)));
2014-08-20 11:30:42 +00:00
2014-08-17 12:56:05 +00:00
lines_menu = context_menu->addMenu(tr("Dotted Lines"));
connect(lines_menu, SIGNAL(triggered(QAction*)), this, SLOT(onLinesClicked(QAction*)));
#if !defined(Q_OS_MAC)
2014-08-26 18:23:27 +00:00
context_menu->setStyleSheet("QMenu {\
2014-08-26 18:43:01 +00:00
background-color: #f0f0f0; /* sets background of the menu */\
2014-08-26 18:23:27 +00:00
border: 1px solid black;\
}\
QMenu::item {\
/* sets background of menu item. set this to something non-transparent\
if you want menu color and menu item color to be different */\
2014-08-26 18:43:01 +00:00
background-color: #f0f0f0;\
2014-08-26 18:23:27 +00:00
}\
QMenu::item:selected { /* when user selects item using mouse or keyboard */\
background-color: #ABCDEF;\
}");
#else
context_menu->setStyleSheet("QMenu::item:selected { /* when user selects item using mouse or keyboard */\
background-color: #ABCDEF;\
}");
#endif
2014-08-17 12:56:05 +00:00
}
void MyDockWindow::closeEvent(QCloseEvent *event)
{
gGraphView::dock->deleteLater();
gGraphView::dock=nullptr;
QMainWindow::closeEvent(event);
}
MyDockWindow * gGraphView::dock = nullptr;
void gGraphView::popoutGraph()
{
if (popout_graph) {
if (dock == nullptr) {
dock = new MyDockWindow(mainwin->getDaily(), Qt::Window);
dock->resize(width(),0);
// QScrollArea
}
QDockWidget * widget = new QDockWidget(dock);
widget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
widget->setMouseTracking(true);
int h = dock->height()+popout_graph->height()+30;
if (h > height()) h = height();
dock->resize(dock->width(), h);
widget->resize(width(), popout_graph->height()+30);
gGraphView * gv = new gGraphView(widget, this);
widget->setWidget(gv);
gv->setMouseTracking(true);
gv->setDay(this->day());
dock->addDockWidget(Qt::BottomDockWidgetArea, widget,Qt::Vertical);
/////// Fix some resize glitches ///////
// https://stackoverflow.com/questions/26286646/create-a-qdockwidget-that-resizes-to-its-contents?rq=1
QDockWidget* dummy = new QDockWidget;
dock->addDockWidget(Qt::BottomDockWidgetArea, dummy);
dock->removeDockWidget(dummy);
QPoint mousePos = dock->mapFromGlobal(QCursor::pos());
mousePos.setY(dock->rect().bottom()+2);
QCursor::setPos(dock->mapToGlobal(mousePos));
QMouseEvent* grabSeparatorEvent =
new QMouseEvent(QMouseEvent::MouseButtonPress,mousePos,Qt::LeftButton,Qt::LeftButton,Qt::NoModifier);
qApp->postEvent(dock, grabSeparatorEvent);
/////////////////////////////////////////
// dock->updateGeometry();
if (!dock->isVisible()) dock->show();
gGraph * graph = popout_graph;
QString basename = graph->title()+" - ";
if (graph->m_day) {
// append the date of the graph's left edge to the snapshot name
// so the user knows what day the snapshot starts
// because the name is displayed to the user, use local time
QDateTime date = QDateTime::fromMSecsSinceEpoch(graph->min_x, Qt::LocalTime);
basename += date.date().toString(Qt::SystemLocaleLongDate);
}
QString newname = basename;
// Find a new name.. How many snapshots for each graph counts as stupid?
QString newtitle = graph->title();
widget->setWindowTitle(newname);
gGraph * newgraph = new gGraph(newname, nullptr, newtitle, graph->units(), graph->height(), graph->group());
newgraph->setHeight(graph->height());
short group = 0;
gv->m_graphs.insert(m_graphs.indexOf(graph)+1, newgraph);
gv->m_graphsbyname[newname] = newgraph;
newgraph->m_graphview = gv;
for (auto & l : graph->m_layers) {
Layer * layer = l->Clone();
if (layer) {
newgraph->m_layers.append(layer);
}
}
for (auto & g : m_graphs) {
group = qMax(g->group(), group);
}
newgraph->setGroup(group+1);
//newgraph->setMinHeight(pm.height());
newgraph->setDay(graph->m_day);
if (graph->m_day) {
graph->m_day->incUseCounter();
}
newgraph->min_x = graph->min_x;
newgraph->max_x = graph->max_x;
newgraph->setBlockSelect(false);
newgraph->setZoomY(graph->zoomY());
newgraph->setSnapshot(false);
newgraph->setShowTitle(true);
gv->resetLayout();
gv->timedRedraw(0);
//widget->setUpdatesEnabled(true);
}
}
2014-08-17 12:56:05 +00:00
void gGraphView::togglePin()
{
if (pin_graph) {
pin_graph->setPinned(!pin_graph->isPinned());
timedRedraw(0);
}
}
void gGraphView::closeEvent(QCloseEvent * event)
{
timer->stop();
redrawtimer->stop();
disconnect(redrawtimer, 0, 0, 0);
disconnect(timer, 0, 0, 0);
timer->deleteLater();
redrawtimer->deleteLater();
pixmapcache.clear();
if (m_scrollbar) {
this->disconnect(m_scrollbar, SIGNAL(sliderMoved(int)), 0, 0);
}
#ifdef BROKEN_OPENGL_BUILD
QWidget::closeEvent(event);
#elif QT_VERSION < QT_VERSION_CHECK(5,4,0)
QGLWidget::closeEvent(event);
#else
QOpenGLWidget::closeEvent(event);
#endif
}
gGraphView::~gGraphView()
{
#ifndef BROKEN_OPENGL_BUILD
doneCurrent();
#endif
#ifdef ENABLE_THREADED_DRAWING
for (int i=0; i < m_threads.size(); i++) {
delete m_threads[i];
}
delete masterlock;
#endif
// Note: This will cause a crash if two graphs accidentally have the same name
for (auto & graph : m_graphs) {
2014-08-27 13:42:10 +00:00
delete graph;
}
2011-09-05 10:28:41 +00:00
delete m_tooltip;
m_graphs.clear();
}
2015-08-27 09:46:05 +00:00
bool gGraphView::event(QEvent * event)
{
if (event->type() == QEvent::Gesture) {
return gestureEvent(static_cast<QGestureEvent *>(event));
}
return QWidget::event(event);
}
bool gGraphView::gestureEvent(QGestureEvent * event)
{
if (QGesture *pinch = event->gesture(Qt::PinchGesture))
pinchTriggered(static_cast<QPinchGesture *>(pinch));
return true;
}
bool gGraphView::pinchTriggered(QPinchGesture * gesture)
{
gGraph * graph = nullptr;
int group =0;
// if (!graph) {
2015-08-27 09:46:05 +00:00
// just pick any graph then
for (const auto & g : m_graphs) {
if (!g) continue;
if (!g->isEmpty()) {
graph = g;
2015-08-27 09:46:05 +00:00
group = graph->group();
break;
}
}
// } else group=graph->group();
2015-08-27 09:46:05 +00:00
if (!graph) { return true; }
Q_UNUSED(group)
// qDebug() << gesture << gesture->scaleFactor();
2015-08-27 09:46:05 +00:00
if (gesture->state() == Qt::GestureStarted) {
pinch_min = m_minx;
pinch_max = m_maxx;
}
int origin_px = gesture->centerPoint().x() - titleWidth;
// could use this instead, and have it more dynamic
// graph->ZoomX(gesture->scaleFactor(), x);
static const double zoom_hard_limit = 500.0;
qint64 min = pinch_min;
qint64 max = pinch_max;
int width = graph->m_rect.width() - graph->left - graph->right;
double hardspan = graph->rmax_x - graph->rmin_x;
double span = max - min;
double ww = double(origin_px) / double(width);
double origin = ww * span;
2018-06-10 19:11:35 +00:00
double q = span * gesture->totalScaleFactor();
2015-08-27 09:46:05 +00:00
if (q > hardspan) { q = hardspan; }
if (q < hardspan / zoom_hard_limit) { q = hardspan / zoom_hard_limit; }
min = min + origin - (q * ww);
max = min + q;
if (min < graph->rmin_x) {
min = graph->rmin_x;
max = min + q;
}
if (max > graph->rmax_x) {
max = graph->rmax_x;
min = max - q;
}
//extern const int max_history;
SetXBounds(min, max, graph->m_group);
return true;
}
void gGraphView::dumpInfo()
{
2014-08-05 11:17:03 +00:00
QDate date = mainwin->getDaily()->getDate();
QString text = "==================== CPAP Information Dump ====================";
mainwin->log(text);
2014-08-05 11:17:03 +00:00
Day * day = p_profile->GetGoodDay(date, MT_CPAP);
if (day) {
QDateTime dt=QDateTime::fromMSecsSinceEpoch(day->first(), Qt::UTC);
2014-08-05 11:17:03 +00:00
mainwin->log(QString("Available Channels for %1").arg(dt.toString("MMM dd yyyy")));
QHash<schema::ChanType, QList<schema::Channel *> > list;
for (const auto & sess : day->sessions) {
for (auto it=sess->eventlist.begin(), end=sess->eventlist.end(); it != end; ++it) {
2014-08-05 11:17:03 +00:00
ChannelID code = it.key();
schema::Channel * chan = &schema::channel[code];
list[chan->type()].append(chan);
}
}
QHash<schema::ChanType, QList<schema::Channel *> >::iterator lit;
for (auto lit = list.begin(), end=list.end(); lit != end; ++lit) {
2014-08-05 11:17:03 +00:00
switch (lit.key()) {
case schema::DATA:
text = "DATA: ";
break;
case schema::SETTING:
text = "SETTING: ";
break;
case schema::FLAG:
text = "FLAG: ";
break;
case schema::MINOR_FLAG:
text = "MINOR_FLAG: ";
break;
case schema::SPAN:
text = "SPAN: ";
break;
case schema::WAVEFORM:
text = "WAVEFORM: ";
break;
case schema::UNKNOWN:
text = "UNKNOWN: ";
break;
default:
break;
2014-08-05 11:17:03 +00:00
}
QStringList str;
for (const auto & chan : lit.value()) {
str.append(chan->code());
2014-08-05 11:17:03 +00:00
}
str.sort();
text.append(str.join(", "));
mainwin->log(text);
}
}
2014-08-05 11:17:03 +00:00
// for (int i=0;i<m_graphs.size();i++) {
// m_graphs[i]->dumpInfo();
// }
}
// Render graphs with QPainter or pixmap caching, depending on preferences
2014-05-07 19:52:59 +00:00
void gGraphView::DrawTextQue(QPainter &painter)
{
// process the text drawing queue
int h,w;
2014-05-07 19:52:59 +00:00
strings_drawn_this_frame += m_textque.size() + m_textqueRect.size();;
2014-05-07 19:52:59 +00:00
for (const TextQue & q : m_textque) {
// can do antialiased text via texture cache fine on mac
// Just draw the fonts..
painter.setPen(QColor(q.color));
painter.setFont(*q.font);
2014-05-07 19:52:59 +00:00
if (q.angle == 0) {
2014-05-07 19:52:59 +00:00
painter.drawText(q.x, q.y, q.text);
} else {
2014-05-07 19:52:59 +00:00
w = painter.fontMetrics().width(q.text);
h = painter.fontMetrics().xHeight() + 2;
2014-05-07 19:52:59 +00:00
painter.translate(q.x, q.y);
painter.rotate(-q.angle);
painter.drawText(floor(-w / 2.0)-6, floor(-h / 2.0), q.text);
2014-05-07 19:52:59 +00:00
painter.rotate(+q.angle);
painter.translate(-q.x, -q.y);
}
}
m_textque.clear();
////////////////////////////////////////////////////////////////////////
// Text Rectangle Queues..
////////////////////////////////////////////////////////////////////////
for (const TextQueRect & q : m_textqueRect) {
// Just draw the fonts..
painter.setPen(QColor(q.color));
painter.setFont(*q.font);
if (q.angle == 0) {
painter.drawText(q.rect, q.flags, q.text);
} else {
w = painter.fontMetrics().width(q.text);
h = painter.fontMetrics().xHeight() + 2;
painter.translate(q.rect.x(), q.rect.y());
painter.rotate(-q.angle);
painter.drawText(floor(-w / 2.0), floor(-h / 2.0), q.text);
painter.rotate(+q.angle);
painter.translate(-q.rect.x(), -q.rect.y());
}
}
m_textqueRect.clear();
2014-05-07 19:52:59 +00:00
}
const QString z__cacheStr = "%1:%2:%3";
void gGraphView::DrawTextQueCached(QPainter &painter)
{
// process the text drawing queue
int h,w;
QString hstr;
QPixmap pm;
2018-06-10 06:07:29 +00:00
float xxx, yyy;
const int buf = 8;
int fonta = defaultfont->pointSize();
int fontb = mediumfont->pointSize();
int fontc = bigfont->pointSize();
int size;
for (const TextQue & q : m_textque) {
// can do antialiased text via texture cache fine on mac
// Generate the pixmap cache "key"
size = (q.font == defaultfont) ? fonta : (q.font==mediumfont) ? fontb : (q.font == bigfont) ? fontc : q.font->pointSize();
hstr = z__cacheStr.arg(q.text).arg(q.color.name()).arg(size);
if (!QPixmapCache::find(hstr, &pm)) {
QFontMetrics fm(*q.font);
w = fm.width(q.text);
h = fm.height()+buf;
pm = QPixmap(w, h);
pm.fill(Qt::transparent);
QPainter imgpainter(&pm);
imgpainter.setPen(q.color);
imgpainter.setFont(*q.font);
imgpainter.setRenderHint(QPainter::TextAntialiasing, q.antialias);
imgpainter.drawText(0, h-buf, q.text);
imgpainter.end();
QPixmapCache::insert(hstr, pm);
}
h = pm.height();
w = pm.width();
if (q.angle != 0) {
xxx = q.x - h - (h / 2);
2018-06-10 06:07:29 +00:00
yyy = q.y + w / 2;
xxx += 4;
yyy += 4;
painter.translate(xxx, yyy);
painter.rotate(-q.angle);
painter.drawPixmap(QRect(0, h / 2, w, h), pm);
painter.rotate(+q.angle);
painter.translate(-xxx, -yyy);
} else {
QRect r1(q.x - buf / 2 + 4, q.y - h + buf, w, h);
painter.drawPixmap(r1, pm);
}
}
////////////////////////////////////////////////////////////////////////
// Text Rectangle Queues..
////////////////////////////////////////////////////////////////////////
for (const TextQueRect & q : m_textqueRect) {
// can do antialiased text via texture cache fine on mac
// Generate the pixmap cache "key"
size = (q.font == defaultfont) ? fonta : (q.font==mediumfont) ? fontb : (q.font == bigfont) ? fontc : q.font->pointSize();
hstr = z__cacheStr.arg(q.text).arg(q.color.name()).arg(size);
2014-08-17 12:56:05 +00:00
if (!QPixmapCache::find(hstr, &pm)) {
w = q.rect.width();
h = q.rect.height();
pm = QPixmap(w, h);
pm.fill(Qt::transparent);
QPainter imgpainter(&pm);
imgpainter.setPen(q.color);
imgpainter.setFont(*q.font);
imgpainter.setRenderHint(QPainter::TextAntialiasing, true);
imgpainter.drawText(QRect(0,0, w, h), q.flags, q.text);
imgpainter.end();
QPixmapCache::insert(hstr, pm);
} else {
2018-06-10 06:07:29 +00:00
h = pm.height();
w = pm.width();
}
if (q.angle != 0) {
xxx = q.rect.x() - h - (h / 2);
yyy = q.rect.y() + w / 2;
xxx += 4;
yyy += 4;
painter.translate(xxx, yyy);
painter.rotate(-q.angle);
2018-06-10 06:07:29 +00:00
painter.drawPixmap(QRect(0, h / 2, w, h), pm);
painter.rotate(+q.angle);
painter.translate(-xxx, -yyy);
} else {
painter.drawPixmap(q.rect,pm, QRect(0,0,w,h));
}
}
strings_drawn_this_frame += m_textque.size() + m_textqueRect.size();;
m_textque.clear();
m_textqueRect.clear();
}
2014-08-17 12:56:05 +00:00
void gGraphView::AddTextQue(const QString &text, QRectF rect, quint32 flags, float angle, QColor color, QFont *font, bool antialias)
{
#ifdef ENABLED_THREADED_DRAWING
text_mutex.lock();
#endif
m_textqueRect.append(TextQueRect(rect,flags,text,angle,color,font,antialias));
#ifdef ENABLED_THREADED_DRAWING
text_mutex.unlock();
#endif
}
void gGraphView::AddTextQue(const QString &text, short x, short y, float angle, QColor color, QFont *font, bool antialias)
{
#ifdef ENABLED_THREADED_DRAWING
2011-08-31 14:28:19 +00:00
text_mutex.lock();
#endif
m_textque.append(TextQue(x,y,angle,text,color,font,antialias));
#ifdef ENABLED_THREADED_DRAWING
2011-08-31 14:28:19 +00:00
text_mutex.unlock();
#endif
}
void gGraphView::addGraph(gGraph *g, short group)
{
2011-09-10 16:27:07 +00:00
if (!g) {
qDebug() << "Attempted to add an empty graph!";
return;
}
if (!m_graphs.contains(g)) {
g->setGroup(group);
m_graphs.push_back(g);
if (!m_graphsbyname.contains(g->name())) {
m_graphsbyname[g->name()] = g;
} else {
2014-08-11 04:52:14 +00:00
qDebug() << "Can't have two graphs with the same code string in the same GraphView!!";
}
// updateScrollBar();
}
}
2014-05-07 19:52:59 +00:00
// Calculate total height of all graphs including spacers
float gGraphView::totalHeight()
{
float th = 0;
for (const auto & g : m_graphs) {
2014-08-11 04:52:14 +00:00
if (g->isEmpty() || (!g->visible())) { continue; }
2014-08-11 04:52:14 +00:00
th += g->height() + graphSpacer;
}
return ceil(th);
}
2014-05-07 19:52:59 +00:00
float gGraphView::findTop(gGraph *graph)
{
float th = -m_offsetY;
for (const auto & g : m_graphs) {
if (g == graph) { break; }
if (g->isEmpty() || (!g->visible())) { continue; }
th += g->height() * m_scaleY + graphSpacer;
}
return ceil(th);
}
2014-05-07 19:52:59 +00:00
float gGraphView::scaleHeight()
{
float th = 0;
for (const auto & graph : m_graphs) {
if (graph->isEmpty() || (!graph->visible())) { continue; }
th += graph->height() * m_scaleY + graphSpacer;
}
return ceil(th);
}
2014-05-07 19:52:59 +00:00
2014-05-26 03:48:22 +00:00
void gGraphView::updateScale()
{
2014-08-11 04:52:14 +00:00
if (!isVisible()) {
m_scaleY = 0.0;
return;
}
2014-05-26 03:48:22 +00:00
float th = totalHeight(); // height of all graphs
float h = height(); // height of main widget
2014-08-11 04:52:14 +00:00
2014-05-26 03:48:22 +00:00
if (th < h) {
th -= graphSpacer;
// th -= visibleGraphs() * graphSpacer; // compensate for spacer height
2014-05-26 03:48:22 +00:00
m_scaleY = h / th; // less graphs than fits on screen, so scale to fit
} else {
m_scaleY = 1.0;
}
updateScrollBar();
}
void gGraphView::resizeEvent(QResizeEvent *e)
{
#if (QT_VERSION >= QT_VERSION_CHECK(5,4,0)) && !defined(BROKEN_OPENGL_BUILD)
// This ques a needed redraw event..
QOpenGLWidget::resizeEvent(e);
#endif
updateScale();
2014-08-11 04:52:14 +00:00
if (m_scaleY > 0.0001) {
for (auto & graph : m_graphs) {
graph->resize(e->size().width(), graph->height() * m_scaleY);
2014-08-11 04:52:14 +00:00
}
}
2014-08-11 04:52:14 +00:00
e->accept();
}
2014-05-07 19:52:59 +00:00
void gGraphView::scrollbarValueChanged(int val)
{
//qDebug() << "Scrollbar Changed" << val;
if (m_offsetY != val) {
m_offsetY = val;
#if QT_VERSION >= QT_VERSION_CHECK(5,4,0)
update();
#else
2014-08-29 04:02:16 +00:00
timedRedraw(); // do this on a timer?
#endif
}
}
void gGraphView::GetRXBounds(qint64 &st, qint64 &et)
{
for (const auto & graph : m_graphs) {
if (graph->group() == 0) {
st = graph->rmin_x;
et = graph->rmax_x;
break;
}
}
}
2014-05-07 19:52:59 +00:00
void gGraphView::resetZoom() {
Overview *overvw = qobject_cast<Overview*>(caller);
if (overvw) {
overvw->on_zoomButton_clicked();
return;
}
ResetBounds(true);
}
void gGraphView::ResetBounds(bool refresh) //short group)
{
if (m_graphs.size() == 0) return;
Q_UNUSED(refresh)
qint64 m1 = 0, m2 = 0;
gGraph *graph = nullptr;
for (auto & g : m_graphs) {
g->ResetBounds();
if (!g->min_x) { continue; }
graph = g;
if (!m1 || g->min_x < m1) { m1 = g->min_x; }
if (!m2 || g->max_x > m2) { m2 = g->max_x; }
}
// if (p_profile->general->linkGroups()) {
// for (int i = 0; i < m_graphs.size(); i++) {
// m_graphs[i]->SetMinX(m1);
// m_graphs[i]->SetMaxX(m2);
// }
// }
if (!graph) {
graph = m_graphs[0];
}
m_minx = graph->min_x;
m_maxx = graph->max_x;
updateScale();
}
2014-05-07 19:52:59 +00:00
void gGraphView::GetXBounds(qint64 &st, qint64 &et)
{
st = m_minx;
et = m_maxx;
}
// Supplies time range to all graph objects in linked group, refreshing if requested
void gGraphView::SetXBounds(qint64 minx, qint64 maxx, short group, bool refresh)
{
for (auto & graph : m_graphs) {
if ((graph->group() == group)) {
graph->SetXBounds(minx, maxx);
}
}
m_minx = minx; // left and right edges of graph, in msec in epoch
m_maxx = maxx;
if (refresh) { timedRedraw(0); }
}
2014-05-07 19:52:59 +00:00
void gGraphView::updateScrollBar()
{
if (!m_scrollbar || (m_graphs.size() == 0)) {
return;
}
float th = scaleHeight(); // height of all graphs
float h = height(); // height of main widget
float vis = 0;
for (const auto & graph : m_graphs) {
vis += (graph->isEmpty() || !graph->visible()) ? 0 : 1;
}
if (th < h) { // less graphs than fits on screen
m_scrollbar->setMaximum(0); // turn scrollbar off.
} else { // more graphs than fit on screen
//m_scaleY=1.0;
float avgheight = th / vis;
m_scrollbar->setPageStep(avgheight);
m_scrollbar->setSingleStep(avgheight / 8.0);
m_scrollbar->setMaximum(th - height());
if (m_offsetY > th - height()) {
m_offsetY = th - height();
}
}
}
void gGraphView::setScrollBar(MyScrollBar *sb)
{
m_scrollbar = sb;
m_scrollbar->setMinimum(0);
updateScrollBar();
this->connect(m_scrollbar, SIGNAL(valueChanged(int)), SLOT(scrollbarValueChanged(int)));
}
2014-05-07 19:52:59 +00:00
bool gGraphView::renderGraphs(QPainter &painter)
{
float px = m_offsetX;
float py = -m_offsetY;
int numgraphs = 0;
float h, w;
//ax=px;//-m_offsetX;
//bool threaded;
2011-09-01 09:03:23 +00:00
// Tempory hack using this pref..
//#ifdef ENABLED_THREADED_DRAWING
2011-12-21 12:47:47 +00:00
/*if (profile->session->multithreading()) { // && (m_idealthreads>1)) {
threaded=true;
2011-09-07 14:26:00 +00:00
for (int i=0;i<m_idealthreads;i++) {
if (!m_threads[i]->isRunning())
m_threads[i]->start();
2011-09-07 14:26:00 +00:00
}
} else threaded=false; */
//#endif
//threaded=false;
2014-08-11 04:52:14 +00:00
if (height() < 40) return false;
2014-04-05 01:28:07 +00:00
2014-08-11 04:52:14 +00:00
if (m_scaleY < 0.0000001) {
updateScale();
}
2014-04-05 01:28:07 +00:00
lines_drawn_this_frame = 0;
quads_drawn_this_frame = 0;
2014-04-05 01:28:07 +00:00
// Calculate the height of pinned graphs
float pinned_height = 0; // pixel height total
int pinned_graphs = 0; // count
for (auto & g : m_graphs) {
2014-08-11 04:52:14 +00:00
int minh = g->minHeight();
if (g->height() < minh) {
g->setHeight(minh);
}
2014-08-11 04:52:14 +00:00
if (g->isEmpty()) { continue; }
2014-08-11 04:52:14 +00:00
if (!g->visible()) { continue; }
2014-08-11 04:52:14 +00:00
if (!g->isPinned()) { continue; }
2014-04-05 01:28:07 +00:00
2014-08-11 04:52:14 +00:00
h = g->height() * m_scaleY;
pinned_height += h + graphSpacer;
2014-04-05 01:28:07 +00:00
pinned_graphs++;
}
py += pinned_height; // start drawing at the end of pinned space
2014-04-05 01:28:07 +00:00
// Draw non pinned graphs
for (auto & g : m_graphs) {
2014-08-11 04:52:14 +00:00
if (g->isEmpty()) { continue; }
2014-08-11 04:52:14 +00:00
if (!g->visible()) { continue; }
2014-08-11 04:52:14 +00:00
if (g->isPinned()) { continue; }
2011-08-29 07:13:58 +00:00
numgraphs++;
2014-08-11 04:52:14 +00:00
h = g->height() * m_scaleY;
// set clipping?
if (py > height()) {
break; // we are done.. can't draw anymore
}
if ((py + h + graphSpacer) >= 0) {
w = width();
2014-08-27 13:42:10 +00:00
int tw = 0; // (g->showTitle() ? titleWidth : 0);
2014-08-11 04:52:14 +00:00
queGraph(g, px + tw, py, width() - tw, h);
2014-05-25 16:20:33 +00:00
if ((m_graphs.size() > 1) && m_showsplitter) {
// draw the splitter handle
2014-05-17 12:20:06 +00:00
painter.setPen(QColor(220, 220, 220, 255));
painter.drawLine(0, py + h, w, py + h);
painter.setPen(QColor(158,158,158,255));
2014-05-07 19:52:59 +00:00
painter.drawLine(0, py + h + 1, w, py + h + 1);
painter.setPen(QColor(240, 240, 240, 255));
2014-05-07 19:52:59 +00:00
painter.drawLine(0, py + h + 2, w, py + h + 2);
}
2011-09-01 09:03:23 +00:00
}
py = ceil(py + h + graphSpacer);
}
2014-04-05 01:28:07 +00:00
// Physically draw the unpinned graphs
for (const auto & g : m_drawlist) {
g->paint(painter, QRegion(g->m_rect));
2014-04-05 01:28:07 +00:00
}
m_drawlist.clear();
if (m_graphs.size() > 1) {
AppSetting->usePixmapCaching() ? DrawTextQueCached(painter) :DrawTextQue(painter);
2014-04-05 01:28:07 +00:00
// Draw a gradient behind pinned graphs
2014-05-07 19:52:59 +00:00
QLinearGradient linearGrad(QPointF(100, 100), QPointF(width() / 2, 100));
linearGrad.setColorAt(0, QColor(216, 216, 255));
linearGrad.setColorAt(1, Qt::white);
painter.fillRect(0, 0, width(), pinned_height, QBrush(linearGrad));
}
2014-04-05 01:28:07 +00:00
py = 0; // start drawing at top...
2014-04-05 01:28:07 +00:00
// Draw Pinned graphs
for (const auto & g : m_graphs) {
2014-08-11 04:52:14 +00:00
if (g->isEmpty()) { continue; }
2014-08-11 04:52:14 +00:00
if (!g->visible()) { continue; }
2014-08-11 04:52:14 +00:00
if (!g->isPinned()) { continue; }
2014-08-11 04:52:14 +00:00
h = g->height() * m_scaleY;
2014-04-05 01:28:07 +00:00
numgraphs++;
if (py > height()) {
break; // we are done.. can't draw anymore
}
2014-04-05 01:28:07 +00:00
if ((py + h + graphSpacer) >= 0) {
w = width();
2014-08-27 13:42:10 +00:00
int tw = 0; //(g->showTitle() ? titleWidth : 0);
2014-04-05 01:28:07 +00:00
2014-08-11 04:52:14 +00:00
queGraph(g, px + tw, py, width() - tw, h);
2014-04-05 01:28:07 +00:00
2014-05-25 16:20:33 +00:00
if ((m_graphs.size() > 1) && m_showsplitter) {
2014-04-05 01:28:07 +00:00
// draw the splitter handle
2014-05-17 12:20:06 +00:00
painter.setPen(QColor(220, 220, 220, 255));
2014-05-07 19:52:59 +00:00
painter.drawLine(0, py + h, w, py + h);
2014-05-17 12:20:06 +00:00
painter.setPen(QColor(128, 128, 128, 255));
2014-05-07 19:52:59 +00:00
painter.drawLine(0, py + h + 1, w, py + h + 1);
2014-05-17 12:20:06 +00:00
painter.setPen(QColor(190, 190, 190, 255));
2014-05-07 19:52:59 +00:00
painter.drawLine(0, py + h + 2, w, py + h + 2);
2014-04-05 01:28:07 +00:00
}
}
py = ceil(py + h + graphSpacer);
2014-04-05 01:28:07 +00:00
}
//int thr=m_idealthreads;
#ifdef ENABLED_THREADED_DRAWING
if (threaded) {
for (int i = 0; i < m_idealthreads; i++) {
masterlock->acquire(1);
m_threads[i]->mutex.unlock();
}
// wait till all the threads are done
// ask for all the CPU's back..
masterlock->acquire(m_idealthreads);
masterlock->release(m_idealthreads);
} else {
#endif
for (const auto & g : m_drawlist) {
g->paint(painter, QRegion(g->m_rect));
}
m_drawlist.clear();
#ifdef ENABLED_THREADED_DRAWING
}
#endif
//int elapsed=time.elapsed();
//QColor col=Qt::black;
2011-09-01 09:03:23 +00:00
// lines->setSize(linesize);
AppSetting->usePixmapCaching() ? DrawTextQueCached(painter) :DrawTextQue(painter);
//glDisable(GL_TEXTURE_2D);
//glDisable(GL_DEPTH_TEST);
return numgraphs > 0;
}
2011-12-20 11:12:52 +00:00
#include "version.h"
#ifdef BROKEN_OPENGL_BUILD
void gGraphView::paintEvent(QPaintEvent *)
#else
void gGraphView::paintGL()
#endif
{
2014-08-11 04:52:14 +00:00
if (!isVisible()) {
// wtf is this even getting CALLED??
return;
}
#ifdef DEBUG_EFFICIENCY
QElapsedTimer time;
time.start();
#endif
if (redrawtimer->isActive()) {
redrawtimer->stop();
}
2011-12-20 11:12:52 +00:00
2014-07-11 12:09:38 +00:00
bool render_cube = false; //p_profile->appearance->animations(); // do something to
if (width() <= 0) { return; }
if (height() <= 0) { return; }
2014-08-19 14:04:29 +00:00
// Create QPainter object, note this is only valid from paintGL events!
2014-05-07 19:52:59 +00:00
QPainter painter(this);
2015-08-27 04:00:58 +00:00
painter.setRenderHint(QPainter::HighQualityAntialiasing, true);
2014-08-19 14:04:29 +00:00
painter.setRenderHint(QPainter::TextAntialiasing, true);
2014-05-07 19:52:59 +00:00
QRect bgrect(0, 0, width(), height());
painter.fillRect(bgrect,QBrush(QColor(255,255,255)));
2014-05-07 19:52:59 +00:00
bool graphs_drawn = true;
lines_drawn_this_frame = 0;
quads_drawn_this_frame = 0;
strings_drawn_this_frame = 0;
strings_cached_this_frame = 0;
// qDebug() << "About to call renderGraphs from paintGL()";
// sleep(3);
2014-05-07 19:52:59 +00:00
graphs_drawn = renderGraphs(painter);
// qDebug() << "Finished call renderGraphs from paintGL()";
// sleep(3);
if (!graphs_drawn) { // No graphs drawn? show something useful :)
qDebug() << "No graphs drawn";
// sleep(3);
QString txt;
if (m_showAuthorMessage) {
if (emptyText() == STR_Empty_Brick) {
txt = QObject::tr("I'm very sorry your machine doesn't record useful data to graph in Daily View :(");
} else {
// not proud of telling them their machine is a Brick.. ;)
txt = QObject::tr("There is no data to graph");
}
}
qDebug() << txt;
// sleep(3);
2014-08-21 14:16:23 +00:00
// int x2, y2;
// GetTextExtent(m_emptytext, x2, y2, bigfont);
// int tp2, tp1;
if (this->m_emptyimage.isNull() )
qDebug() << "m_emptyimage is NULL";
else
qDebug() << "m_emptyimage is not NULL";
// sleep(3);
2011-12-20 17:22:02 +00:00
if (! this->m_emptyimage.isNull()) {
int x = width()/2 - this->m_emptyimage.width()/2;
int y = height()/2 - this->m_emptyimage.height()/2;
// qDebug() << "size is " + QString::number(m_emptyimage.width()) + " by " + QString::number(m_emptyimage.height());
// qDebug() << "x-pos: " + QString::number(x) + " y-pos: " + QString::number(y);
// sleep(3);
QRectF target(QRect(x, y, this->m_emptyimage.width(), this->m_emptyimage.height()));
QRectF source(QRect(0, 0, this->m_emptyimage.width(), this->m_emptyimage.height()));
2019-03-12 16:32:18 +00:00
// painter.drawPixmap(target, this->m_emptyimage, source); // this is segfaulting! WHY ??
// tp2 = height() /2 + m_emptyimage.height()/2 + y2;
} /*else {
tp2 = height() / 2 + y2;
2014-08-21 14:16:23 +00:00
}*/
// qDebug() << "Break 9";
// sleep(3);
2014-05-07 19:52:59 +00:00
QColor col = Qt::black;
painter.setPen(col);
2014-08-21 14:16:23 +00:00
// painter.setFont(*bigfont);
// painter.drawText((width() / 2) - x2 / 2, tp2, m_emptytext);
QRectF rec(0,0,width(),0);
// if (defaultfont == nullptr)
// qDebug() << "Default font is NULL";
// else
// qDebug() << "Default font is not NULL";
// sleep(3);
painter.setFont(*defaultfont);
rec = painter.boundingRect(rec, Qt::AlignHCenter | Qt::AlignBottom, txt);
rec.moveBottom(height()-5);
// qDebug() << "Break 8";
// sleep(3);
painter.drawText(rec, Qt::AlignHCenter | Qt::AlignBottom, txt);
// qDebug() << "Break 7";
// sleep(3);
2014-08-21 14:16:23 +00:00
}
// qDebug() << "Break 1";
// sleep(3);
2018-04-22 12:06:48 +00:00
if (AppSetting->lineCursorMode()) {
emit updateCurrentTime(graphs_drawn ? m_currenttime : 0.0F);
2014-08-21 14:16:23 +00:00
} else {
emit updateRange(graphs_drawn ? m_minx : 0.0F, m_maxx);
}
AppSetting->usePixmapCaching() ? DrawTextQueCached(painter) :DrawTextQue(painter);
// qDebug() << "Break 2";
// sleep(3);
2014-05-07 19:52:59 +00:00
m_tooltip->paint(painter);
// qDebug() << "Break 3";
// sleep(3);
#ifdef DEBUG_EFFICIENCY
2018-06-12 16:51:58 +00:00
const int rs = 20;
static double ring[rs] = {0};
static int rp = 0;
// Show FPS and draw time
2018-04-22 12:06:48 +00:00
if (m_showsplitter && AppSetting->showPerformance()) {
QString ss;
qint64 ela = time.nsecsElapsed();
double ms = double(ela) / 1000000.0;
ring[rp++] = 1000.0 / ms;
rp %= rs;
double v = 0;
for (int i = 0; i < rs; i++) {
v += ring[i];
}
double fps = v / double(rs);
ss = "Debug Mode " + QString::number(fps, 'f', 1) + "fps "
+ QString::number(lines_drawn_this_frame, 'f', 0) + " lines "
// + QString::number(quads_drawn_this_frame, 'f', 0) + " quads "
+ QString::number(strings_drawn_this_frame, 'f', 0) + " strings "
+ QString::number(strings_cached_this_frame, 'f', 0) + " cached ";
int w, h;
2014-05-07 19:52:59 +00:00
// this uses tightBoundingRect, which is different on Mac than it is on Windows & Linux.
GetTextExtent(ss, w, h);
QColor col = Qt::white;
2014-05-25 07:07:08 +00:00
if (m_graphs.size() > 0) {
painter.fillRect(width() - m_graphs[0]->marginRight(), 0, m_graphs[0]->marginRight(), w, QBrush(col));
}
#ifndef Q_OS_MAC
// if (usePixmapCache()) xx+=4; else xx-=3;
#endif
2014-05-07 19:52:59 +00:00
AddTextQue(ss, width(), w / 2, 90, QColor(Qt::black), defaultfont);
AppSetting->usePixmapCaching() ? DrawTextQueCached(painter) :DrawTextQue(painter);
}
2014-05-17 11:33:33 +00:00
// painter.setPen(Qt::lightGray);
// painter.drawLine(0, 0, 0, height());
// painter.drawLine(0, 0, width(), 0);
// painter.setPen(Qt::darkGray);
//painter.drawLine(width(), 0, width(), height());
#endif
2014-05-17 11:33:33 +00:00
2014-05-16 09:30:11 +00:00
painter.end();
// qDebug() << "Break 4";
// sleep(3);
#ifndef BROKEN_OPENGL_BUILD
#if QT_VERSION < QT_VERSION_CHECK(5,4,0)
swapBuffers();
#endif
#endif
2014-05-07 19:52:59 +00:00
if (this->isVisible() && !graphs_drawn && render_cube) { // keep the cube spinning
redrawtimer->setInterval(1000.0 / 50); // 50 FPS
redrawtimer->setSingleShot(true);
redrawtimer->start();
}
2014-08-17 12:56:05 +00:00
// qDebug() << "Break 5";
// sleep(3);
2014-08-17 12:56:05 +00:00
}
QString gGraphView::getRangeInDaysString()
2014-08-17 12:56:05 +00:00
{
QDateTime st = QDateTime::fromMSecsSinceEpoch(m_minx);
QDateTime et = QDateTime::fromMSecsSinceEpoch(m_maxx);
QDate std = st.date();
QDate etd = et.date();
if (std.year() == etd.year())
return st.toString(" d MMM") + " - " + et.toString("d MMM yyyy");
else
return st.toString(" d MMM yyyy") + " - " + et.toString("d MMM yyyy");
}
QString gGraphView::getRangeString()
{
QDateTime st = QDateTime::fromMSecsSinceEpoch(m_minx);
QDateTime et = QDateTime::fromMSecsSinceEpoch(m_maxx);
QDate std = st.date();
QDate etd = et.date();
// Format if Begin and End are on different days
if (std != etd) { // further adjust formatting if on different years
if (std.year() == etd.year())
return st.toString(" d MMM [ HH:mm:ss") + " - " + et.toString("HH:mm:ss ] d MMM yyyy");
else
return st.toString(" d MMM yyyy [ HH:mm:ss") + " - " + et.toString("HH:mm:ss ] d MMM yyyy");
}
// Range is within one (local) day
qint64 diff = m_maxx - m_minx;
QString fmt;
if (diff > 60000) {
fmt = "HH:mm:ss";
} else {
fmt = "HH:mm:ss:zzz";
}
QString txt = st.toString(QObject::tr("d MMM yyyy [ %1 - %2 ]").arg(fmt).arg(et.toString(fmt))) ;
return txt;
/***** WTF is this code trying to do? Replaced by above 8/9/2019
// a note about time zone usage here
// even though this string will be displayed to the user
// the graph is drawn using UTC times, so no conversion
// is needed to format the date and time for the user
// i.e. if the graph says the cursor is at 5pm, then that
// is what we should display.
// passing in UTC below is necessary to prevent QT
// from automatically converting the time to local time
2014-08-17 12:56:05 +00:00
QString fmt;
qint64 diff = m_maxx - m_minx;
if (diff > 86400000) { // 86400000 is one day, in milliseconds
int days = ceil(double(m_maxx-m_minx) / 86400000.0);
qint64 minx = floor(double(m_minx)/86400000.0);
minx *= 86400000L;
qint64 maxx = minx + 86400000L * qint64(days)-1;
QDateTime st = QDateTime::fromMSecsSinceEpoch(minx, Qt::UTC);
QDateTime et = QDateTime::fromMSecsSinceEpoch(maxx, Qt::UTC);
QString txt = st.toString("d MMM") + " - " + et.addDays(-1).toString("d MMM yyyy");
return txt;
2014-08-17 12:56:05 +00:00
} else if (diff > 60000) {
fmt = "HH:mm:ss";
} else {
fmt = "HH:mm:ss:zzz";
}
QString txt = st.toString(QObject::tr("d MMM [ %1 - %2 ]").arg(fmt).arg(et.toString(fmt))) ;
return txt;
*/
}
void gGraphView::leaveEvent(QEvent * event)
{
Q_UNUSED(event);
if (m_metaselect) {
m_metaselect = false;
timedRedraw(0);
}
releaseKeyboard();
}
// For manual scrolling
void gGraphView::setOffsetY(int offsetY)
{
if (m_offsetY != offsetY) {
m_offsetY = offsetY;
2011-12-20 11:12:52 +00:00
redraw(); //issue full redraw..
}
}
// For manual X scrolling (not really needed)
void gGraphView::setOffsetX(int offsetX)
{
if (m_offsetX != offsetX) {
m_offsetX = offsetX;
2011-12-20 11:12:52 +00:00
redraw(); //issue redraw
}
}
void gGraphView::mouseMoveEvent(QMouseEvent *event)
{
grabKeyboard();
int x = event->x();
int y = event->y();
m_mouse = QPoint(x, y);
if (m_sizer_dragging) { // Resize handle being dragged
float my = y - m_sizer_point.y();
//qDebug() << "Sizer moved vertically" << m_sizer_index << my*m_scaleY;
float h = m_graphs[m_sizer_index]->height();
h += my / m_scaleY;
if (h > m_graphs[m_sizer_index]->minHeight()) {
m_graphs[m_sizer_index]->setHeight(h);
m_sizer_point.setX(x);
m_sizer_point.setY(y);
updateScrollBar();
2014-08-29 04:02:16 +00:00
timedRedraw();
}
return;
}
if (m_graph_dragging) { // Title bar being dragged to reorder
gGraph *p;
int yy = m_sizer_point.y();
bool empty;
if (y < yy) {
for (int i = m_graph_index - 1; i >= 0; i--) {
if (m_graphs[i]->isPinned() != m_graphs[m_graph_index]->isPinned()) {
2014-04-05 01:28:07 +00:00
// fix cursor
continue;
2014-04-05 01:28:07 +00:00
}
empty = m_graphs[i]->isEmpty() || (!m_graphs[i]->visible());
// swapping upwards.
int yy2 = yy - graphSpacer - m_graphs[i]->height() * m_scaleY;
yy2 += m_graphs[m_graph_index]->height() * m_scaleY;
if (y < yy2) {
//qDebug() << "Graph Reorder" << m_graph_index;
p = m_graphs[m_graph_index];
m_graphs[m_graph_index] = m_graphs[i];
m_graphs[i] = p;
if (!empty) {
m_sizer_point.setY(yy - graphSpacer - m_graphs[m_graph_index]->height()*m_scaleY);
2011-12-20 11:12:52 +00:00
redraw();
}
m_graph_index--;
}
if (!empty) { break; }
}
} else if (y > yy + graphSpacer + m_graphs[m_graph_index]->height()*m_scaleY) {
// swapping downwards
//qDebug() << "Graph Reorder" << m_graph_index;
for (int i = m_graph_index + 1; i < m_graphs.size(); i++) {
if (m_graphs[i]->isPinned() != m_graphs[m_graph_index]->isPinned()) {
//m_graph_dragging=false;
2014-04-05 01:28:07 +00:00
// fix cursor
continue;
2014-04-05 01:28:07 +00:00
}
empty = m_graphs[i]->isEmpty() || (!m_graphs[i]->visible());
p = m_graphs[m_graph_index];
m_graphs[m_graph_index] = m_graphs[i];
m_graphs[i] = p;
if (!empty) {
m_sizer_point.setY(yy + graphSpacer + m_graphs[m_graph_index]->height()*m_scaleY);
2014-08-29 04:02:16 +00:00
timedRedraw();
}
m_graph_index++;
if (!empty) { break; }
}
}
return;
}
float py = 0, pinned_height = 0, h;
bool done = false;
2014-04-05 01:28:07 +00:00
// Do pinned graphs first
for (const auto & graph : m_graphs) {
2014-04-05 01:28:07 +00:00
if (graph->isEmpty() || !graph->visible() || !graph->isPinned()) {
2014-04-05 01:28:07 +00:00
continue;
}
2014-04-05 01:28:07 +00:00
h = graph->height() * m_scaleY;
2014-04-05 01:28:07 +00:00
pinned_height += h + graphSpacer;
if (py > height()) {
break; // we are done.. can't draw anymore
}
2014-04-05 01:28:07 +00:00
if (!((y >= py + graph->top) && (y < py + h - graph->bottom))) {
if (graph->isSelected()) {
graph->deselect();
2014-04-05 01:28:07 +00:00
timedRedraw(150);
}
}
// Update Mouse Cursor shape
if ((y >= py + h - 1) && (y < (py + h + graphSpacer))) {
2014-04-05 01:28:07 +00:00
this->setCursor(Qt::SplitVCursor);
done = true;
} else if ((y >= py + 1) && (y < py + h)) {
if (x >= titleWidth + 10) {
this->setCursor(Qt::ArrowCursor);
} else {
2019-03-26 21:59:36 +00:00
m_tooltip->display(tr("Double click title to pin / unpin\nClick and drag to reorder graphs"), x + 10, y, TT_AlignLeft);
2014-08-17 12:56:05 +00:00
timedRedraw(0);
this->setCursor(Qt::OpenHandCursor);
}
m_horiz_travel += qAbs(x - m_lastxpos) + qAbs(y - m_lastypos);
m_lastxpos = x;
m_lastypos = y;
// QPoint p(x,y);
// QMouseEvent e(event->type(),p,event->button(),event->buttons(),event->modifiers());
graph->mouseMoveEvent(event);
done = true;
2014-04-05 01:28:07 +00:00
}
2014-04-05 01:28:07 +00:00
py += h + graphSpacer;
}
py = -m_offsetY;
py += pinned_height;
// Propagate mouseMove events to relevant graphs
2014-04-05 01:28:07 +00:00
if (!done)
for (const auto & graph : m_graphs) {
if (graph->isEmpty() || !graph->visible() || graph->isPinned()) {
continue;
}
h = graph->height() * m_scaleY;
if (py > height()) {
break; // we are done.. can't draw anymore
}
if (!((y >= py + graph->top) && (y < py + h - graph->bottom))) {
if (graph->isSelected()) {
graph->deselect();
timedRedraw(150);
}
}
// Update Mouse Cursor shape
if ((y >= py + h - 1) && (y < (py + h + graphSpacer))) {
this->setCursor(Qt::SplitVCursor);
} else if ((y >= py + 1) && (y < py + h)) {
if (x >= titleWidth + 10) {
this->setCursor(Qt::ArrowCursor);
} else {
m_tooltip->display(tr("Double click title to pin / unpin\nClick and drag to reorder graphs"), x + 10, y, TT_AlignLeft);
2014-08-17 12:56:05 +00:00
timedRedraw(0);
2014-08-10 00:28:02 +00:00
this->setCursor(Qt::OpenHandCursor);
}
m_horiz_travel += qAbs(x - m_lastxpos) + qAbs(y - m_lastypos);
m_lastxpos = x;
m_lastypos = y;
if (graph) {
graph->mouseMoveEvent(event);
}
}
/* else if (!m_button_down && (y >= py) && (y < py+m_graphs[i]->top)) {
// Mouse cursor is in top graph margin.
} else if (!m_button_down && (y >= py+h-m_graphs[i]->bottom) && (y <= py+h)) {
// Mouse cursor is in bottom grpah margin.
} else if (m_button_down || ((y >= py+m_graphs[i]->top) && (y < py + h-m_graphs[i]->bottom))) {
if (m_button_down || (x >= titleWidth+10)) { //(gYAxis::Margin-5)
this->setCursor(Qt::ArrowCursor);
m_horiz_travel+=qAbs(x-m_lastxpos)+qAbs(y-m_lastypos);
m_lastxpos=x;
m_lastypos=y;
QPoint p(x-titleWidth,y-py);
QMouseEvent e(event->type(),p,event->button(),event->buttons(),event->modifiers());
m_graphs[i]->mouseMoveEvent(&e);
if (!m_button_down && (x<=titleWidth+(gYAxis::Margin-5))) {
//qDebug() << "Hovering over" << m_graphs[i]->title();
if (m_graphsbytitle[STR_TR_EventFlags]==m_graphs[i]) {
QVector<Layer *> & layers=m_graphs[i]->layers();
gFlagsGroup *fg;
for (int i=0;i<layers.size();i++) {
if ((fg=dynamic_cast<gFlagsGroup *>(layers[i]))!=nullptr) {
float bh=fg->barHeight();
int count=fg->count();
float yp=py+m_graphs[i]->marginTop();
yp=y-yp;
float th=(float(count)*bh);
if (yp>=0 && yp<th) {
int i=yp/bh;
if (i<count) {
ChannelID code=fg->visibleLayers()[i]->code();
QString ttip=schema::channel[code].description();
2018-04-22 12:06:48 +00:00
m_tooltip->display(ttip,x,y-20,AppSetting->tooltipTimeout());
redraw();
//qDebug() << code << ttip;
}
}
break;
}
}
} else {
if (!m_graphs[i]->units().isEmpty()) {
2018-04-22 12:06:48 +00:00
m_tooltip->display(m_graphs[i]->units(),x,y-20,AppSetting->tooltipTimeout());
2011-12-20 11:12:52 +00:00
redraw();
2011-12-05 11:01:11 +00:00
}
2011-12-05 10:50:58 +00:00
}
2011-12-05 11:01:11 +00:00
}
} else {
2011-12-05 11:01:11 +00:00
this->setCursor(Qt::OpenHandCursor);
}
} */
// }
py += h + graphSpacer;
}
}
2014-08-17 12:56:05 +00:00
Layer * gGraphView::findLayer(gGraph * graph, LayerType type)
{
for (auto & layer : graph->m_layers) {
if (layer->layerType() == type) {
return layer;
2014-08-17 12:56:05 +00:00
}
}
return nullptr;
}
class MyWidgetAction : public QWidgetAction
{
public:
MyWidgetAction(ChannelID code, QObject * parent = nullptr) :QWidgetAction(parent), code(code) { chbox = nullptr; }
protected:
2014-10-02 07:56:57 +00:00
virtual QWidget * createWidget(QWidget * /*parent*/) {
connect(chbox, SIGNAL(toggled(bool)), this, SLOT(setChecked(bool)));
connect(chbox, SIGNAL(clicked()), this, SLOT(trigger()));
return chbox;
}
QCheckBox * chbox;
ChannelID code;
};
MinMaxWidget::MinMaxWidget(gGraph * graph, QWidget *parent)
:QWidget(parent), graph(graph)
{
step = 1;
createLayout();
}
void MinMaxWidget::onMinChanged(double d)
{
graph->rec_miny = d;
graph->timedRedraw(0);
}
void MinMaxWidget::onMaxChanged(double d)
{
graph->rec_maxy = d;
graph->timedRedraw(0);
}
void MinMaxWidget::onResetClicked()
{
int tmp = graph->zoomY();
graph->setZoomY(0);
EventDataType miny = graph->MinY(),
maxy = graph->MaxY();
graph->roundY(miny, maxy);
setMin(graph->rec_miny = miny);
setMax(graph->rec_maxy = maxy);
float r = maxy-miny;
if (r > 400) {
step = 50;
} else if (r > 100) {
step = 10;
} else if (r > 50) {
step = 5;
} else {
step = 1;
}
graph->setZoomY(tmp);
}
void MinMaxWidget::onComboChanged(int idx)
{
minbox->setEnabled(idx == 2);
maxbox->setEnabled(idx == 2);
reset->setEnabled(idx == 2);
graph->setZoomY(idx);
if (idx == 2) {
if (qAbs(graph->rec_maxy - graph->rec_miny) < 0.0001) {
onResetClicked();
}
}
}
void MinMaxWidget::createLayout()
{
QGridLayout * layout = new QGridLayout;
layout->setMargin(4);
layout->setSpacing(4);
combobox = new QComboBox(this);
combobox->addItem(tr("Auto-Fit"), 0);
combobox->addItem(tr("Defaults"), 1);
combobox->addItem(tr("Override"), 2);
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)));
minbox = new QDoubleSpinBox(this);
maxbox = new QDoubleSpinBox(this);
minbox->setToolTip(tr("The Minimum Y-Axis value.. Note this can be a negative number if you wish."));
maxbox->setToolTip(tr("The Maximum Y-Axis value.. Must be greater than Minimum to work."));
int idx = graph->zoomY();
combobox->setCurrentIndex(idx);
minbox->setEnabled(idx == 2);
maxbox->setEnabled(idx == 2);
minbox->setAlignment(Qt::AlignRight);
maxbox->setAlignment(Qt::AlignRight);
minbox->setMinimum(-9999.0);
maxbox->setMinimum(-9999.0);
minbox->setMaximum(9999.99);
maxbox->setMaximum(9999.99);
setMin(graph->rec_miny);
setMax(graph->rec_maxy);
float r = graph->rec_maxy - graph->rec_miny;
if (r > 400) {
step = 50;
} else if (r > 100) {
step = 10;
} else if (r > 50) {
step = 5;
} else {
step = 1;
}
minbox->setSingleStep(step);
maxbox->setSingleStep(step);
connect(minbox, SIGNAL(valueChanged(double)), this, SLOT(onMinChanged(double)));
connect(maxbox, SIGNAL(valueChanged(double)), this, SLOT(onMaxChanged(double)));
QLabel * label = new QLabel(tr("Scaling Mode"));
QFont font = label->font();
font.setBold(true);
label->setFont(font);
label->setAlignment(Qt::AlignCenter);
layout->addWidget(label,0,0);
layout->addWidget(combobox,1,0);
label = new QLabel(STR_TR_Min);
label->setFont(font);
label->setAlignment(Qt::AlignCenter);
layout->addWidget(label,0,1);
layout->addWidget(minbox,1,1);
label = new QLabel(STR_TR_Max);
label->setFont(font);
label->setAlignment(Qt::AlignCenter);
layout->addWidget(label,0,2);
layout->addWidget(maxbox,1,2);
reset = new QToolButton(this);
reset->setIcon(QIcon(":/icons/refresh.png"));
reset->setToolTip(tr("This button resets the Min and Max to match the Auto-Fit"));
reset->setEnabled(idx == 2);
layout->addWidget(reset,1,3);
connect(reset, SIGNAL(clicked()), this, SLOT(onResetClicked()));
this->setLayout(layout);
}
2014-08-17 12:56:05 +00:00
void gGraphView::populateMenu(gGraph * graph)
{
2014-08-23 07:11:50 +00:00
QAction * action;
2014-08-27 13:42:10 +00:00
if (graph->isSnapshot()) {
snap_action->setText(tr("Remove Clone"));
2014-08-27 13:42:10 +00:00
snap_action->setData(graph->name()+"|remove");
2014-08-27 15:00:55 +00:00
// zoom100_action->setVisible(false);
2014-08-27 13:42:10 +00:00
} else {
snap_action->setText(tr("Clone %1 Graph").arg(graph->title()));
2014-08-27 13:42:10 +00:00
snap_action->setData(graph->name()+"|snapshot");
2014-08-27 15:00:55 +00:00
// zoom100_action->setVisible(true);
2014-08-27 13:42:10 +00:00
}
2014-08-23 07:11:50 +00:00
// Menu title fonts
QFont font = QApplication::font();
font.setBold(true);
font.setPointSize(font.pointSize() + 3);
gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(graph,LT_LineChart));
SummaryChart * sc = dynamic_cast<SummaryChart *>(findLayer(graph,LT_SummaryChart));
gSummaryChart * stg = dynamic_cast<gSummaryChart *>(findLayer(graph,LT_Overview));
limits_menu->clear();
if (lc || sc || stg) {
QWidgetAction * widget = new QWidgetAction(this);
MinMaxWidget * minmax = new MinMaxWidget(graph, this);
widget->setDefaultWidget(minmax);
limits_menu->addAction(widget);
limits_menu->menuAction()->setVisible(true);
} else {
limits_menu->menuAction()->setVisible(false);
}
2014-08-17 12:56:05 +00:00
// First check for any linechart for this graph..
if (lc) {
lines_menu->clear();
for (int i=0, end=lc->m_dotlines.size(); i < end; i++) {
const DottedLine & dot = lc->m_dotlines[i];
if (!lc->m_enabled[dot.code]) continue;
2014-08-17 12:56:05 +00:00
schema::Channel &chan = schema::channel[dot.code];
if (dot.available) {
2014-08-23 07:11:50 +00:00
QWidgetAction * widget = new QWidgetAction(context_menu);
QCheckBox *chbox = new QCheckBox(chan.calc[dot.type].label(), context_menu);
chbox->setMouseTracking(true);
chbox->setStyleSheet(QString("QCheckBox:hover { background: %1; }").arg(QApplication::palette().highlight().color().name()));
2014-08-23 07:11:50 +00:00
widget->setDefaultWidget(chbox);
widget->setCheckable(true);
widget->setData(QString("%1|%2").arg(graph->name()).arg(i));
connect(chbox, SIGNAL(toggled(bool)), widget, SLOT(setChecked(bool)));
connect(chbox, SIGNAL(clicked()), widget, SLOT(trigger()));
bool b = lc->m_dot_enabled[dot.code][dot.type]; //chan.calc[dot.type].enabled;
2014-08-23 07:11:50 +00:00
chbox->setChecked(b);
lines_menu->addAction(widget);
2014-08-17 12:56:05 +00:00
}
}
2014-08-23 07:11:50 +00:00
lines_menu->menuAction()->setVisible(lines_menu->actions().size() > 0);
2014-08-23 07:11:50 +00:00
if (lines_menu->actions().size() > 0) {
lines_menu->insertSeparator(lines_menu->actions()[0]);
action = new QAction(QObject::tr("%1").arg(graph->title()), lines_menu);
lines_menu->insertAction(lines_menu->actions()[0], action);
action->setFont(font);
action->setData(QString(""));
action->setEnabled(false);
}
//////////////////////////////////////////////////////////////////////////////////////
// Populate Plots Menus
//////////////////////////////////////////////////////////////////////////////////////
2014-08-17 12:56:05 +00:00
plots_menu->clear();
for (const auto code : lc->m_codes) {
if (lc->m_day && !lc->m_day->channelHasData(code)) continue;
2014-08-23 07:11:50 +00:00
QWidgetAction * widget = new QWidgetAction(context_menu);
QCheckBox *chbox = new QCheckBox(schema::channel[code].label(), context_menu);
chbox->setMouseTracking(true);
chbox->setToolTip(schema::channel[code].description());
chbox->setStyleSheet(QString("QCheckBox:hover { background: %1; }").arg(QApplication::palette().highlight().color().name()));
2014-08-23 07:11:50 +00:00
widget->setDefaultWidget(chbox);
2014-08-23 07:11:50 +00:00
widget->setCheckable(true);
widget->setData(QString("%1|%2").arg(graph->name()).arg(code));
2014-08-23 07:11:50 +00:00
connect(chbox, SIGNAL(toggled(bool)), widget, SLOT(setChecked(bool)));
connect(chbox, SIGNAL(clicked()), widget, SLOT(trigger()));
2014-08-23 07:11:50 +00:00
bool b = lc->m_enabled[code];
chbox->setChecked(b);
2014-08-23 07:11:50 +00:00
plots_menu->addAction(widget);
2014-08-17 12:56:05 +00:00
}
2014-08-23 07:11:50 +00:00
plots_menu->menuAction()->setVisible((plots_menu->actions().size() > 1));
2014-08-23 07:11:50 +00:00
if (plots_menu->actions().size() > 0) {
plots_menu->insertSeparator(plots_menu->actions()[0]);
action = new QAction(QObject::tr("%1").arg(graph->title()), plots_menu);
plots_menu->insertAction(plots_menu->actions()[0], action);
action->setFont(font);
action->setData(QString(""));
action->setEnabled(false);
}
2014-08-20 11:30:42 +00:00
2014-08-23 07:11:50 +00:00
//////////////////////////////////////////////////////////////////////////////////////
// Populate Event Menus
//////////////////////////////////////////////////////////////////////////////////////
oximeter_menu->clear();
cpap_menu->clear();
2014-08-20 11:30:42 +00:00
using namespace schema;
quint32 showflags = schema::FLAG | schema::MINOR_FLAG | schema::SPAN;
if (p_profile->general->showUnknownFlags()) showflags |= schema::UNKNOWN;
QList<ChannelID> chans = lc->m_day->getSortedMachineChannels(showflags);
QHash<MachineType, int> Vis;
for (const auto code : chans) {
schema::Channel & chan = schema::channel[code];
2014-08-20 11:30:42 +00:00
QWidgetAction * widget = new QWidgetAction(context_menu);
2014-08-20 11:30:42 +00:00
QCheckBox *chbox = new QCheckBox(schema::channel[code].fullname(), context_menu);
chbox->setPalette(context_menu->palette());
chbox->setMouseTracking(true);
chbox->setToolTip(schema::channel[code].description());
chbox->setStyleSheet(QString("QCheckBox:hover { background: %1; }").arg(QApplication::palette().highlight().color().name()));
widget->setDefaultWidget(chbox);
widget->setCheckable(true);
widget->setData(QString("%1|%2").arg(graph->name()).arg(code));
connect(chbox, SIGNAL(toggled(bool)), widget, SLOT(setChecked(bool)));
connect(chbox, SIGNAL(clicked()), widget, SLOT(trigger()));
bool b = lc->m_flags_enabled[code];
chbox->setChecked(b);
Vis[chan.machtype()] += b ? 1 : 0;
action = nullptr;
if (chan.machtype() == MT_OXIMETER) {
oximeter_menu->insertAction(nullptr, widget);
} else if ( chan.machtype() == MT_CPAP) {
cpap_menu->insertAction(nullptr,widget);
}
}
QString HideAllEvents = QObject::tr("Hide All Events");
QString ShowAllEvents = QObject::tr("Show All Events");
2014-08-23 07:11:50 +00:00
oximeter_menu->menuAction()->setVisible(oximeter_menu->actions().size() > 0);
cpap_menu->menuAction()->setVisible(cpap_menu->actions().size() > 0);
2014-08-23 07:11:50 +00:00
if (cpap_menu->actions().size() > 0) {
cpap_menu->addSeparator();
if (Vis[MT_CPAP] > 0) {
action = cpap_menu->addAction(HideAllEvents);
action->setData(QString("%1|HideAll:CPAP").arg(graph->name()));
} else {
action = cpap_menu->addAction(ShowAllEvents);
action->setData(QString("%1|ShowAll:CPAP").arg(graph->name()));
}
2014-08-23 07:11:50 +00:00
// Show CPAP Events menu Header...
cpap_menu->insertSeparator(cpap_menu->actions()[0]);
2014-08-23 07:11:50 +00:00
action = new QAction(QObject::tr("%1").arg(graph->title()), cpap_menu);
cpap_menu->insertAction(cpap_menu->actions()[0], action);
action->setFont(font);
action->setData(QString(""));
action->setEnabled(false);
}
if (oximeter_menu->actions().size() > 0) {
oximeter_menu->addSeparator();
if (Vis[MT_OXIMETER] > 0) {
action = oximeter_menu->addAction(HideAllEvents);
action->setData(QString("%1|HideAll:OXI").arg(graph->name()));
} else {
action = oximeter_menu->addAction(ShowAllEvents);
action->setData(QString("%1|ShowAll:OXI").arg(graph->name()));
}
oximeter_menu->insertSeparator(oximeter_menu->actions()[0]);
2014-08-23 07:11:50 +00:00
action = new QAction(QObject::tr("%1").arg(graph->title()), oximeter_menu);
oximeter_menu->insertAction(oximeter_menu->actions()[0], action);
action->setFont(font);
action->setData(QString(""));
action->setEnabled(false);
}
2014-08-20 11:30:42 +00:00
2014-08-17 12:56:05 +00:00
} else {
lines_menu->clear();
lines_menu->menuAction()->setVisible(false);
plots_menu->clear();
plots_menu->menuAction()->setVisible(false);
oximeter_menu->clear();
oximeter_menu->menuAction()->setVisible(false);
cpap_menu->clear();
cpap_menu->menuAction()->setVisible(false);
2014-08-17 12:56:05 +00:00
}
}
2014-08-27 13:42:10 +00:00
void gGraphView::onSnapshotGraphToggle()
{
QString name = snap_action->data().toString().section("|",0,0);
QString cmd = snap_action->data().toString().section("|",-1).toLower();
auto it = m_graphsbyname.find(name);
2014-08-27 13:42:10 +00:00
if (it == m_graphsbyname.end()) return;
gGraph * graph = it.value();
if (cmd == "snapshot") {
QString basename = name+";";
if (graph->m_day) {
// append the date of the graph's left edge to the snapshot name
// so the user knows what day the snapshot starts
// because the name is displayed to the user, use local time
QDateTime date = QDateTime::fromMSecsSinceEpoch(graph->min_x, Qt::LocalTime);
basename += date.date().toString(Qt::SystemLocaleLongDate);
2014-08-27 13:42:10 +00:00
}
QString newname;
2014-08-27 13:42:10 +00:00
// Find a new name.. How many snapshots for each graph counts as stupid?
for (int i=1;i < 100;i++) {
newname = basename+" ("+QString::number(i)+")";
it = m_graphsbyname.find(newname);
if (it == m_graphsbyname.end()) {
break;
}
}
2014-08-29 10:38:46 +00:00
QString newtitle;
bool fnd = false;
// someday, some clown will keep adding new graphs to break this..
for (int i=1; i < 100; i++) {
newtitle = graph->title()+"-"+QString::number(i);
fnd = false;
for (const auto & graph : m_graphs) {
if (graph->title() == newtitle) {
2014-08-29 10:38:46 +00:00
fnd = true;
break;
}
}
if (!fnd) break;
}
if (fnd) {
// holy crap.. what patience. but not what I meant by as many as you like ;)
return;
}
gGraph * newgraph = new gGraph(newname, nullptr, newtitle, graph->units(), graph->height(), graph->group());
// newgraph->setBlockSelect(true);
newgraph->setHeight(graph->height());
short group = 0;
m_graphs.insert(m_graphs.indexOf(graph)+1, newgraph);
m_graphsbyname[newname] = newgraph;
newgraph->m_graphview = this;
for (const auto & l : graph->m_layers) {
Layer * layer = l->Clone();
if (layer) {
newgraph->m_layers.append(layer);
}
}
for (const auto & g : m_graphs) {
group = qMax(g->group(), group);
}
newgraph->setGroup(group+1);
//newgraph->setMinHeight(pm.height());
newgraph->setDay(graph->m_day);
if (graph->m_day) {
graph->m_day->incUseCounter();
}
newgraph->min_x = graph->min_x;
newgraph->max_x = graph->max_x;
if (graph->blockZoom()) {
newgraph->setBlockZoom(graph->blockZoom());
newgraph->setBlockSelect(true);
}
if (graph->blockSelect()) {
newgraph->setBlockSelect(true);
}
newgraph->setZoomY(graph->zoomY());
newgraph->setSnapshot(true);
2014-08-27 13:42:10 +00:00
emit GraphsChanged();
2014-08-27 13:42:10 +00:00
// addGraph(newgraph);
updateScale();
timedRedraw(0);
2014-08-27 13:42:10 +00:00
} else if (cmd == "remove") {
if (graph->m_day) {
graph->m_day->decUseCounter();
if (graph->m_day->useCounter() == 0) {
}
}
2014-08-27 13:42:10 +00:00
m_graphsbyname.remove(graph->name());
m_graphs.removeAll(it.value());
delete graph;
updateScale();
timedRedraw(0);
emit GraphsChanged();
2014-08-27 13:42:10 +00:00
}
qDebug() << cmd << name;
}
bool gGraphView::hasSnapshots()
{
bool snap = false;
for (const auto & graph : m_graphs) {
if (graph->isSnapshot()) {
snap = true;
break;
}
}
return snap;
}
2014-08-17 12:56:05 +00:00
void gGraphView::onPlotsClicked(QAction *action)
{
QString name = action->data().toString().section("|",0,0);
ChannelID code = action->data().toString().section("|",-1).toInt();
auto it = m_graphsbyname.find(name);
2014-08-17 12:56:05 +00:00
if (it == m_graphsbyname.end()) return;
gGraph * graph = it.value();
gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(graph, LT_LineChart));
if (!lc) return;
lc->m_enabled[code] = !lc->m_enabled[code];
graph->min_y = graph->MinY();
graph->max_y = graph->MaxY();
2014-08-23 07:11:50 +00:00
graph->timedRedraw(0);
2014-08-17 12:56:05 +00:00
// lc->Miny();
// lc->Maxy();
}
2014-08-20 11:30:42 +00:00
void gGraphView::onOverlaysClicked(QAction *action)
{
QString name = action->data().toString().section("|",0,0);
QString data = action->data().toString().section("|",-1);
auto it = m_graphsbyname.find(name);
if (it == m_graphsbyname.end()) return;
gGraph * graph = it.value();
gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(graph, LT_LineChart));
if (!lc) return;
bool ok;
ChannelID code = data.toInt(&ok);
if (ok) {
// Just toggling a flag on/off
bool b = ! lc->m_flags_enabled[code];
lc->m_flags_enabled[code] = b;
QWidgetAction * widget = qobject_cast<QWidgetAction *>(action);
if (widget) {
widget->setChecked(b);
}
timedRedraw(0);
return;
}
2014-08-20 11:30:42 +00:00
QString hideall = data.section(":",0,0);
2014-08-20 11:30:42 +00:00
if ((hideall == "HideAll") || (hideall == "ShowAll")) {
2014-08-20 11:30:42 +00:00
bool value = (hideall == "HideAll") ? false : true;
QString group = data.section(":",-1).toUpper();
MachineType mtype;
if (group == "CPAP") mtype = MT_CPAP;
else if (group == "OXI") mtype = MT_OXIMETER;
else mtype = MT_UNKNOWN;
2014-08-20 11:30:42 +00:00
// First toggle the actual flag bits
for (auto it=lc->m_flags_enabled.begin(), end=lc->m_flags_enabled.end(); it != end; ++it) {
if (schema::channel[it.key()].machtype() == mtype) {
lc->m_flags_enabled[it.key()] = value;
}
}
2014-08-20 11:30:42 +00:00
// Now toggle the menu actions.. bleh
if (mtype == MT_CPAP) {
for (auto & action : cpap_menu->actions()) {
if (action->isCheckable()) {
action->setChecked(value);
}
}
} else if (mtype == MT_OXIMETER) {
for (auto & action : oximeter_menu->actions()) {
if (action->isCheckable()) {
action->setChecked(value);
}
}
}
}
2014-08-20 11:30:42 +00:00
}
2014-08-17 12:56:05 +00:00
void gGraphView::onLinesClicked(QAction *action)
{
2014-08-23 07:11:50 +00:00
QString name = action->data().toString().section("|",0,0);
QString data = action->data().toString().section("|",-1);
auto it = m_graphsbyname.find(name);
2014-08-17 12:56:05 +00:00
if (it == m_graphsbyname.end()) return;
gGraph * graph = it.value();
gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(graph, LT_LineChart));
if (!lc) return;
2014-08-23 07:11:50 +00:00
bool ok;
int i = data.toInt(&ok);
if (ok) {
2014-08-17 12:56:05 +00:00
DottedLine & dot = lc->m_dotlines[i];
schema::Channel &chan = schema::channel[dot.code];
chan.calc[dot.type].enabled = !chan.calc[dot.type].enabled;
lc->m_dot_enabled[dot.code][dot.type] = !lc->m_dot_enabled[dot.code][dot.type];
2014-08-17 12:56:05 +00:00
}
2014-08-23 07:11:50 +00:00
timedRedraw(0);
2014-08-17 12:56:05 +00:00
}
void gGraphView::mousePressEvent(QMouseEvent *event)
{
int x = event->x();
int y = event->y();
float h, pinned_height = 0, py = 0;
2014-04-05 01:28:07 +00:00
bool done = false;
2014-04-05 01:28:07 +00:00
// first handle pinned graphs.
// Calculate total height of all pinned graphs
for (int i=0, end=m_graphs.size(); i<end; ++i) {
gGraph * g = m_graphs[i];
if (!g || g->isEmpty() || !g->visible() || !g->isPinned()) {
2014-04-05 01:28:07 +00:00
continue;
}
2014-04-05 01:28:07 +00:00
h = g->height() * m_scaleY;
pinned_height += h + graphSpacer;
2014-04-05 01:28:07 +00:00
if (py > height()) {
2014-04-05 01:28:07 +00:00
break;
}
2014-04-05 01:28:07 +00:00
if ((py + h + graphSpacer) >= 0) {
if ((y >= py + h - 1) && (y <= py + h + graphSpacer)) {
2014-04-05 01:28:07 +00:00
this->setCursor(Qt::SplitVCursor);
m_sizer_dragging = true;
m_sizer_index = i;
2014-04-05 01:28:07 +00:00
m_sizer_point.setX(x);
m_sizer_point.setY(y);
done = true;
2014-04-05 01:28:07 +00:00
} else if ((y >= py) && (y < py + h)) {
//qDebug() << "Clicked" << i;
2014-08-17 12:56:05 +00:00
if ((event->button() == Qt::LeftButton) && (x < titleWidth + 20)) {
2014-04-05 01:28:07 +00:00
// clicked on title to drag graph..
// Note: reorder has to be limited to pinned graphs.
m_graph_dragging = true;
2014-04-05 01:28:07 +00:00
m_tooltip->cancel();
timedRedraw(50);
m_graph_index = i;
2014-04-05 01:28:07 +00:00
m_sizer_point.setX(x);
m_sizer_point.setY(py); // point at top of graph..
this->setCursor(Qt::ClosedHandCursor);
//done=true;
2014-08-17 12:56:05 +00:00
} else if ((event->button() == Qt::RightButton) && (x < (titleWidth + gYAxis::Margin))) {
this->setCursor(Qt::ArrowCursor);
pin_action->setText(QObject::tr("Unpin %1 Graph").arg(g->title()));
pin_graph = g;
popout_action->setText(QObject::tr("Popout %1 Graph").arg(g->title()));
popout_graph = g;
2014-08-17 12:56:05 +00:00
populateMenu(g);
context_menu->popup(event->globalPos());
//done=true;
2014-08-12 06:43:10 +00:00
} else if (!g->blockSelect()) {
if (m_metaselect) {
if (m_selected_graph) {
m_selected_graph->m_selecting_area = false;
}
}
// send event to graph..
m_point_clicked = QPoint(event->x(), event->y());
2014-04-05 01:28:07 +00:00
//QMouseEvent e(event->type(),m_point_clicked,event->button(),event->buttons(),event->modifiers());
m_button_down = true;
m_metaselect = event->modifiers() & Qt::AltModifier;
m_horiz_travel = 0;
m_graph_index = i;
m_selected_graph = g;
g->mousePressEvent(event);
2014-04-05 01:28:07 +00:00
}
done = true;
2014-04-05 01:28:07 +00:00
}
}
2014-04-05 01:28:07 +00:00
py += h + graphSpacer;
}
2014-04-05 01:28:07 +00:00
// then handle the remainder...
py = -m_offsetY;
py += pinned_height;
2014-04-05 01:28:07 +00:00
if (!done) {
for (int i=0, end=m_graphs.size(); i<end; ++i) {
gGraph * g = m_graphs[i];
2014-04-05 01:28:07 +00:00
if (!g || g->isEmpty() || !g->visible() || g->isPinned()) { continue; }
h = g->height() * m_scaleY;
2014-04-05 01:28:07 +00:00
if (py > height()) {
break;
}
if ((py + h + graphSpacer) >= 0) {
if ((y >= py + h - 1) && (y <= py + h + graphSpacer)) {
this->setCursor(Qt::SplitVCursor);
m_sizer_dragging = true;
m_sizer_index = i;
m_sizer_point.setX(x);
m_sizer_point.setY(y);
//qDebug() << "Sizer clicked" << i;
//done=true;
} else if ((y >= py) && (y < py + h)) {
//qDebug() << "Clicked" << i;
2014-08-17 12:56:05 +00:00
if ((event->button() == Qt::LeftButton) && (x < (titleWidth + 20))) { // clicked on title to drag graph..
m_graph_dragging = true;
m_tooltip->cancel();
redraw();
m_graph_index = i;
m_sizer_point.setX(x);
m_sizer_point.setY(py); // point at top of graph..
this->setCursor(Qt::ClosedHandCursor);
//done=true;
2014-08-17 12:56:05 +00:00
} else if ((event->button() == Qt::RightButton) && (x < (titleWidth + gYAxis::Margin))) {
this->setCursor(Qt::ArrowCursor);
popout_action->setText(QObject::tr("Popout %1 Graph").arg(g->title()));
popout_graph = g;
2014-08-17 12:56:05 +00:00
pin_action->setText(QObject::tr("Pin %1 Graph").arg(g->title()));
pin_graph = g;
populateMenu(g);
context_menu->popup(event->globalPos());
//done=true;
2014-08-12 06:43:10 +00:00
} else if (!g->blockSelect()) {
if (m_metaselect) {
if (m_selected_graph) {
m_selected_graph->m_selecting_area = false;
}
}
// send event to graph..
m_point_clicked = QPoint(event->x(), event->y());
//QMouseEvent e(event->type(),m_point_clicked,event->button(),event->buttons(),event->modifiers());
m_button_down = true;
m_metaselect = event->modifiers() & Qt::AltModifier;
m_horiz_travel = 0;
m_graph_index = i;
m_selected_graph = g;
g->mousePressEvent(event);
}
}
}
py += h + graphSpacer;
2014-08-17 12:56:05 +00:00
done=true;
}
}
2014-08-17 12:56:05 +00:00
if (!done) {
// if (event->button() == Qt::RightButton) {
// this->setCursor(Qt::ArrowCursor);
// context_menu->popup(event->globalPos());
// done=true;
// }
}
}
void gGraphView::mouseReleaseEvent(QMouseEvent *event)
{
int x = event->x();
int y = event->y();
float h, py = 0, pinned_height = 0;
bool done = false;
// Copy to a local variable to make sure this gets cleared
bool button_down = m_button_down;
m_button_down = false;
2014-04-05 01:28:07 +00:00
// Handle pinned graphs first
for (const auto & g : m_graphs) {
if (!g || g->isEmpty() || !g->visible() || !g->isPinned()) {
continue;
}
h = g->height() * m_scaleY;
pinned_height += h + graphSpacer;
2014-04-05 01:28:07 +00:00
if (py > height()) {
break; // we are done.. can't draw anymore
}
if ((y >= py + h - 1) && (y < (py + h + graphSpacer))) {
this->setCursor(Qt::SplitVCursor);
done = true;
} else if ((y >= py + 1) && (y <= py + h)) {
// if (!m_sizer_dragging && !m_graph_dragging) {
// g->mouseReleaseEvent(event);
// }
if (x >= titleWidth + 10) {
this->setCursor(Qt::ArrowCursor);
} else {
this->setCursor(Qt::OpenHandCursor);
}
done = true;
}
2014-04-05 01:28:07 +00:00
py += h + graphSpacer;
}
// Now do the unpinned ones
py = -m_offsetY;
py += pinned_height;
if (done)
for (const auto & g : m_graphs) {
2014-04-05 01:28:07 +00:00
if (!g || g->isEmpty() || !g->visible() || g->isPinned()) {
continue;
}
2014-04-05 01:28:07 +00:00
h = g->height() * m_scaleY;
2014-04-05 01:28:07 +00:00
if (py > height()) {
break; // we are done.. can't draw anymore
}
2014-04-05 01:28:07 +00:00
if ((y >= py + h - 1) && (y < (py + h + graphSpacer))) {
this->setCursor(Qt::SplitVCursor);
} else if ((y >= py + 1) && (y <= py + h)) {
2014-04-05 01:28:07 +00:00
// if (!m_sizer_dragging && !m_graph_dragging) {
// g->mouseReleaseEvent(event);
// }
if (x >= titleWidth + 10) {
this->setCursor(Qt::ArrowCursor);
} else {
this->setCursor(Qt::OpenHandCursor);
}
}
py += h + graphSpacer;
2014-04-05 01:28:07 +00:00
}
if (m_sizer_dragging) {
m_sizer_dragging = false;
return;
}
if (m_graph_dragging) {
m_graph_dragging = false;
// not sure why the cursor code doesn't catch this..
if (x >= titleWidth + 10) {
this->setCursor(Qt::ArrowCursor);
} else {
this->setCursor(Qt::OpenHandCursor);
}
return;
}
// The graph that got the button press gets the release event
if (button_down) {
// m_button_down = false;
m_metaselect = event->modifiers() & Qt::AltModifier;
2014-07-19 14:09:04 +00:00
saveHistory();
if (m_metaselect) {
m_point_released = event->pos();
} else {
2014-08-07 21:54:32 +00:00
if ((m_graph_index >= 0) && (m_graphs[m_graph_index])) {
m_graphs[m_graph_index]->mouseReleaseEvent(event);
}
}
}
2014-08-17 12:56:05 +00:00
timedRedraw(0);
}
void gGraphView::keyReleaseEvent(QKeyEvent *event)
{
if (m_metaselect && !(event->modifiers() & Qt::AltModifier)) {
QMouseEvent mevent(QEvent::MouseButtonRelease, m_point_released, Qt::LeftButton, Qt::LeftButton, event->modifiers());
2014-07-20 09:26:59 +00:00
if (m_graph_index>=0) {
m_graphs[m_graph_index]->mouseReleaseEvent(&mevent);
2014-07-20 09:26:59 +00:00
}
m_metaselect = false;
timedRedraw(50);
}
if (event->key() == Qt::Key_Escape) {
if (history.size() > 0) {
SelectionHistoryItem h = history.takeFirst();
SetXBounds(h.minx, h.maxx);
2014-07-19 14:09:04 +00:00
// could Forward push this to another list?
} else {
2014-07-19 14:09:04 +00:00
ResetBounds();
}
return;
}
#ifdef BROKEN_OPENGL_BUILD
QWidget::keyReleaseEvent(event);
#elif QT_VERSION < QT_VERSION_CHECK(5,4,0)
QGLWidget::keyReleaseEvent(event);
#else
QOpenGLWidget::keyReleaseEvent(event);
#endif
}
void gGraphView::mouseDoubleClickEvent(QMouseEvent *event)
{
2014-04-05 01:28:07 +00:00
mousePressEvent(event); // signal missing.. a qt change might "fix" this if we are not careful.
int x = event->x();
int y = event->y();
float h, py = 0, pinned_height = 0;
bool done = false;
2014-04-05 01:28:07 +00:00
// Handle pinned graphs first
for (const auto & g : m_graphs) {
if (!g || g->isEmpty() || !g->visible() || !g->isPinned()) {
2014-04-05 01:28:07 +00:00
continue;
}
2014-04-05 01:28:07 +00:00
h = g->height() * m_scaleY;
2014-04-05 01:28:07 +00:00
pinned_height += h + graphSpacer;
if (py > height()) {
break; // we are done.. can't draw anymore
}
2014-04-05 01:28:07 +00:00
if ((py + h + graphSpacer) >= 0) {
if ((y >= py) && (y <= py + h)) {
if (x < titleWidth) {
// What to do when double clicked on the graph title ??
g->mouseDoubleClickEvent(event);
// pin the graph??
g->setPinned(false);
redraw();
2014-04-05 01:28:07 +00:00
} else {
// send event to graph..
g->mouseDoubleClickEvent(event);
2014-04-05 01:28:07 +00:00
}
done = true;
2014-04-05 01:28:07 +00:00
} else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) {
// What to do when double clicked on the resize handle?
done = true;
2014-04-05 01:28:07 +00:00
}
}
py += h;
py += graphSpacer; // do we want the extra spacer down the bottom?
2014-04-05 01:28:07 +00:00
}
py = -m_offsetY;
py += pinned_height;
2014-04-05 01:28:07 +00:00
if (!done) // then handle unpinned graphs
for (const auto & g : m_graphs) {
if (!g || g->isEmpty() || !g->visible() || g->isPinned()) {
continue;
}
h = g->height() * m_scaleY;
if (py > height()) {
break;
}
if ((py + h + graphSpacer) >= 0) {
if ((y >= py) && (y <= py + h)) {
if (x < titleWidth) {
// What to do when double clicked on the graph title ??
g->mouseDoubleClickEvent(event);
g->setPinned(true);
redraw();
} else {
// send event to graph..
g->mouseDoubleClickEvent(event);
}
} else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) {
// What to do when double clicked on the resize handle?
}
}
py += h;
py += graphSpacer; // do we want the extra spacer down the bottom?
}
}
void gGraphView::wheelEvent(QWheelEvent *event)
{
// Hmm.. I could optionalize this to change mousewheel behaviour without affecting the scrollbar now..
if (m_button_down)
return;
if (event->modifiers() == Qt::NoModifier) {
2018-04-22 12:06:48 +00:00
int scrollDampening = AppSetting->scrollDampening();
if (event->orientation() == Qt::Vertical) { // Vertical Scrolling
if (horizScrollTime.elapsed() < scrollDampening) {
return;
}
if (m_scrollbar)
m_scrollbar->SendWheelEvent(event); // Just forwarding the event to scrollbar for now..
m_tooltip->cancel();
vertScrollTime.start();
return;
}
// (This is a total pain in the butt on MacBook touchpads..)
if (vertScrollTime.elapsed() < scrollDampening) {
return;
}
horizScrollTime.start();
}
gGraph *graph = nullptr;
int group = 0;
2014-10-02 07:56:57 +00:00
//int x = event->x();
int y = event->y();
float h, py = 0, pinned_height = 0;
// Find graph hovered over
for (const auto & g : m_graphs) {
if (!g || g->isEmpty() || !g->visible() || !g->isPinned()) {
continue;
}
h = g->height() * m_scaleY;
pinned_height += h + graphSpacer;
if (py > height()) {
break; // we are done.. can't draw anymore
}
if ((py + h + graphSpacer) >= 0) {
if ((y >= py) && (y <= py + h)) {
graph = g;
break;
} else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) {
// What to do when double clicked on the resize handle?
graph = g;
break;
}
}
py += h;
py += graphSpacer; // do we want the extra spacer down the bottom?
}
if (!graph) {
py = -m_offsetY;
py += pinned_height;
for (const auto & g : m_graphs) {
if (!g || g->isEmpty() || !g->visible() || g->isPinned()) {
continue;
}
h = g->height() * m_scaleY;
if (py > height()) {
break;
}
if ((py + h + graphSpacer) >= 0) {
if ((y >= py) && (y <= py + h)) {
graph = g;
break;
} else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) {
// What to do when double clicked on the resize handle?
graph = g;
break;
}
}
py += h;
py += graphSpacer; // do we want the extra spacer down the bottom?
}
}
if (event->modifiers() == Qt::NoModifier) {
if (!graph) {
// just pick any graph then
for (const auto & g : m_graphs) {
if (!g) continue;
if (!g->isEmpty()) {
graph = g;
group = graph->group();
break;
}
}
} else group=graph->group();
if (!graph) { return; }
double xx = (graph->max_x - graph->min_x);
double zoom = 240.0;
int delta = event->delta();
if (delta > 0) {
graph->min_x -= (xx / zoom) * (float)abs(delta);
} else {
graph->min_x += (xx / zoom) * (float)abs(delta);
}
graph->max_x = graph->min_x + xx;
if (graph->min_x < graph->rmin_x) {
graph->min_x = graph->rmin_x;
graph->max_x = graph->rmin_x + xx;
}
if (graph->max_x > graph->rmax_x) {
graph->max_x = graph->rmax_x;
graph->min_x = graph->max_x - xx;
}
saveHistory();
SetXBounds(graph->min_x, graph->max_x, group);
} else if ((event->modifiers() & Qt::ControlModifier)) {
if (graph) graph->wheelEvent(event);
// int x = event->x();
// int y = event->y();
// float py = -m_offsetY;
// float h;
// for (int i = 0; i < m_graphs.size(); i++) {
// gGraph *g = m_graphs[i];
// if (!g || g->isEmpty() || !g->visible()) { continue; }
// h = g->height() * m_scaleY;
// if (py > height()) {
// break;
// }
// if ((py + h + graphSpacer) >= 0) {
// if ((y >= py) && (y <= py + h)) {
// if (x < titleWidth) {
// // What to do when ctrl+wheel is used on the graph title ??
// } else {
// // send event to graph..
// g->wheelEvent(event);
// }
// } else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) {
// // What to do when the wheel is used on the resize handle?
// }
// }
// py += h;
// py += graphSpacer; // do we want the extra spacer down the bottom?
// }
}
}
void gGraphView::getSelectionTimes(qint64 & start, qint64 & end)
{
if (m_graph_index >= 0) {
gGraph *g = m_graphs[m_graph_index];
if (!g) {
start = 0;
end = 0;
return;
}
2014-08-07 21:54:32 +00:00
int x1 = g->m_selection.x() + titleWidth;
int x2 = x1 + g->m_selection.width();
start = g->screenToTime(x1);
end = g->screenToTime(x2);
}
}
void gGraphView::keyPressEvent(QKeyEvent *event)
{
m_metaselect = event->modifiers() & Qt::AltModifier;
if (m_metaselect && ((event->key() == Qt::Key_B) || (event->key() == 8747))) {
if (mainwin->getDaily()->graphView() == this) {
if (m_graph_index >= 0) {
m_metaselect=false;
qint64 start,end;
getSelectionTimes(start,end);
QDateTime d1 = QDateTime::fromMSecsSinceEpoch(start, Qt::UTC);
mainwin->getDaily()->addBookmark(start, end, QString("Bookmark at %1").arg(d1.time().toString("HH:mm:ss")));
m_graphs[m_graph_index]->cancelSelection();
m_graph_index = -1;
timedRedraw(0);
}
event->accept();
return;
}
}
if ((m_metaselect) && (event->key() >= Qt::Key_0) && (event->key() <= Qt::Key_9)) {
2014-10-02 07:56:57 +00:00
//int bk = (int)event->key()-Qt::Key_0;
m_metaselect = false;
2014-08-17 12:56:05 +00:00
timedRedraw(0);
}
if (event->key() == Qt::Key_F3) {
2018-04-22 12:06:48 +00:00
AppSetting->setLineCursorMode(!AppSetting->lineCursorMode());
timedRedraw(0);
}
2014-08-05 11:17:03 +00:00
if ((event->key() == Qt::Key_F1)) {
dumpInfo();
}
if (event->key() == Qt::Key_Tab) {
event->ignore();
return;
}
if (event->key() == Qt::Key_PageUp) {
2014-05-25 07:07:08 +00:00
if (m_scrollbar) {
2018-04-22 12:06:48 +00:00
m_offsetY -= AppSetting->graphHeight() * 3 * m_scaleY;
2014-05-25 07:07:08 +00:00
m_scrollbar->setValue(m_offsetY);
m_offsetY = m_scrollbar->value();
redraw();
}
return;
} else if (event->key() == Qt::Key_PageDown) {
2014-05-25 07:07:08 +00:00
if (m_scrollbar) {
2018-04-22 12:06:48 +00:00
m_offsetY += AppSetting->graphHeight() * 3 * m_scaleY;
2014-05-25 07:07:08 +00:00
if (m_offsetY < 0) { m_offsetY = 0; }
2014-05-25 07:07:08 +00:00
m_scrollbar->setValue(m_offsetY);
m_offsetY = m_scrollbar->value();
redraw();
}
return;
// redraw();
}
gGraph *g = nullptr;
int group = 0;
// Pick the first valid graph in the primary group
for (const auto & gr : m_graphs) {
if (gr->group() == group) {
if (!gr->isEmpty() && gr->visible()) {
g = gr;
break;
}
}
}
if (!g) {
for (const auto & gr : m_graphs) {
if (!gr->isEmpty()) {
g = gr;
group = g->group();
break;
}
2011-09-05 13:26:10 +00:00
}
}
if (!g) { return; }
g->keyPressEvent(event);
if (event->key() == Qt::Key_Left) {
double xx = g->max_x - g->min_x;
double zoom = 8.0;
if (event->modifiers() & Qt::ControlModifier) { zoom /= 4; }
g->min_x -= xx / zoom;;
g->max_x = g->min_x + xx;
2011-09-05 13:26:10 +00:00
if (g->min_x < g->rmin_x) {
g->min_x = g->rmin_x;
g->max_x = g->rmin_x + xx;
2011-09-05 13:26:10 +00:00
}
2014-07-19 14:09:04 +00:00
saveHistory();
SetXBounds(g->min_x, g->max_x, group);
} else if (event->key() == Qt::Key_Right) {
double xx = g->max_x - g->min_x;
double zoom = 8.0;
if (event->modifiers() & Qt::ControlModifier) { zoom /= 4; }
g->min_x += xx / zoom;
g->max_x = g->min_x + xx;
if (g->max_x > g->rmax_x) {
g->max_x = g->rmax_x;
g->min_x = g->rmax_x - xx;
2011-09-05 13:26:10 +00:00
}
2014-07-19 14:09:04 +00:00
saveHistory();
SetXBounds(g->min_x, g->max_x, group);
} else if (event->key() == Qt::Key_Up) {
float zoom = 0.75F;
if (event->modifiers() & Qt::ControlModifier) { zoom /= 1.5; }
g->ZoomX(zoom, 0); // zoom in.
} else if (event->key() == Qt::Key_Down) {
float zoom = 1.33F;
if (event->modifiers() & Qt::ControlModifier) { zoom *= 1.5; }
g->ZoomX(zoom, 0); // Zoom out
2011-09-05 13:26:10 +00:00
}
2011-09-06 07:33:34 +00:00
//qDebug() << "Keypress??";
}
// Sends day object to be distributed to all Graphs Layers objects
void gGraphView::setDay(Day *day)
{
m_day = day;
for (const auto & g : m_graphs) {
if (g) g->setDay(day);
}
2011-12-20 11:12:52 +00:00
ResetBounds(false);
}
bool gGraphView::isEmpty()
{
bool res = true;
for (const auto & graph : m_graphs) {
2014-08-27 14:08:18 +00:00
if (!graph->isSnapshot() && !graph->isEmpty()) {
res = false;
break;
}
}
return res;
}
2011-10-31 11:55:25 +00:00
void gGraphView::refreshTimeout()
{
if (this->isVisible()) {
redraw();
}
}
void gGraphView::timedRedraw(int ms)
{
2014-07-21 09:39:26 +00:00
if (timer->isActive()) {
2014-08-29 04:02:16 +00:00
if (ms == 0) {
2014-07-21 09:39:26 +00:00
timer->stop();
2014-08-29 04:02:16 +00:00
} else {
int m = timer->remainingTime();
if (m > ms) {
timer->stop();
} else return;
}
}
2014-07-21 09:39:26 +00:00
timer->setSingleShot(true);
timer->start(ms);
}
void gGraphView::deselect()
{
for (auto & graph : m_graphs) {
if (graph) graph->deselect();
}
}
void gGraphView::resetLayout()
{
2018-04-22 12:06:48 +00:00
int default_height = AppSetting->graphHeight();
for (auto & graph : m_graphs) {
if (graph) graph->setHeight(default_height);
}
updateScale();
2014-08-29 04:02:16 +00:00
timedRedraw(0);
}
// Reset order of current graphs to new order, remove pinning
void gGraphView::resetGraphOrder(bool pinFirst, const QList<QString> graphOrder) {
// qDebug() << "gGraphView::resetGraphOrder new order" << graphOrder;
QList<gGraph *> new_graphs;
QList<gGraph *> old_graphs = m_graphs;
// Create new_graphs in order specified by graphOrder
for (int i = 0; i < graphOrder.size(); ++i) {
QString nextGraph = graphOrder.at(i);
auto it = m_graphsbyname.find(nextGraph);
if (it == m_graphsbyname.end()) {
qDebug() << "resetGraphOrder could not find" << nextGraph;
continue; // should not happen
}
gGraph * graph = it.value();
new_graphs.append(graph);
int idx = old_graphs.indexOf(graph);
old_graphs.removeAt(idx);
// qDebug() << "resetGraphOrder added to new graphs" << nextGraph;
}
// If we didn't find everything, append anything extra we have
for (int i = 0; i < old_graphs.size(); i++) {
qDebug() << "resetGraphOrder added leftover" << old_graphs.at(i)->name();
new_graphs.append(old_graphs.at(i));
}
m_graphs = new_graphs;
for (auto & graph : m_graphs) {
if (!graph) continue;
if (graph->isSnapshot()) continue;
graph->setPinned(false);
}
if (pinFirst)
m_graphs[0]->setPinned(true);
}
// Reset order of current graphs to match defaults, remove pinning
void gGraphView::resetGraphOrder(bool pinFirst) {
m_graphs = m_default_graphs;
for (auto & graph : m_graphs) {
if (!graph) continue;
if (graph->isSnapshot()) continue;
graph->setPinned(false);
}
if (pinFirst)
m_graphs[0]->setPinned(true);
}
void gGraphView::SaveDefaultSettings() {
m_default_graphs = m_graphs;
2011-10-31 11:55:25 +00:00
}
const quint32 gvmagic = 0x41756728;
const quint16 gvversion = 4;
void gGraphView::SaveSettings(QString title)
{
qDebug() << "Saving" << title << "settings";
2014-07-11 12:09:38 +00:00
QString filename = p_profile->Get("{DataFolder}/") + title.toLower() + ".shg";
QFile f(filename);
f.open(QFile::WriteOnly);
QDataStream out(&f);
out.setVersion(QDataStream::Qt_4_6);
out.setByteOrder(QDataStream::LittleEndian);
out << (quint32)gvmagic;
out << (quint16)gvversion;
out << (qint16)size();
for (auto & graph : m_graphs) {
if (!graph) continue;
if (graph->isSnapshot()) continue;
// qDebug() << "Saving graph" << title << graph->name();
out << graph->name();
out << graph->height();
out << graph->visible();
out << graph->RecMinY();
out << graph->RecMaxY();
out << graph->zoomY();
out << (bool)graph->isPinned();
gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(graph, LT_LineChart));
if (lc) {
out << (quint32)LT_LineChart;
out << lc->m_flags_enabled;
out << lc->m_enabled;
out << lc->m_dot_enabled;
} else {
out << (quint32)LT_Other;
}
}
f.close();
}
// Merge and overwrite two hashes. (We can't use QHash::unite because it doesn't overwrite.)
// This is useful for loading settings, where we want to leave the defaults alone for features
// that don't yet have settings specified.
template <class T> inline void hashMerge(T & a, const T & b)
{
for (auto key : b.keys()) {
a[key] = b[key];
}
}
bool gGraphView::LoadSettings(QString title)
{
2014-07-11 12:09:38 +00:00
QString filename = p_profile->Get("{DataFolder}/") + title.toLower() + ".shg";
QFile f(filename);
2014-08-11 04:52:14 +00:00
if (!f.exists()) {
return false;
}
f.open(QFile::ReadOnly);
QDataStream in(&f);
in.setVersion(QDataStream::Qt_4_6);
in.setByteOrder(QDataStream::LittleEndian);
quint32 t1;
quint16 version;
in >> t1;
if (t1 != gvmagic) {
qDebug() << "gGraphView" << title << "settings magic doesn't match" << t1 << gvmagic;
return false;
}
in >> version;
if (version < gvversion) {
qDebug() << "gGraphView" << title << "settings will be upgraded.";
}
qint16 siz;
in >> siz;
QString name;
float hght;
bool vis;
EventDataType recminy, recmaxy;
2014-04-05 01:28:07 +00:00
bool pinned;
short zoomy = 0;
2014-08-27 13:42:10 +00:00
QList<gGraph *> neworder;
QHash<QString, gGraph *>::iterator gi;
for (int i = 0; i < siz; i++) {
in >> name;
in >> hght;
in >> vis;
in >> recminy;
in >> recmaxy;
//qDebug() << "Loading graph" << title << name;
if (gvversion >= 1) {
in >> zoomy;
}
if (gvversion >= 2) {
2014-04-05 01:28:07 +00:00
in >> pinned;
}
QHash<ChannelID, bool> flags_enabled;
QHash<ChannelID, bool> plots_enabled;
QHash<ChannelID, QHash<quint32, bool> > dot_enabled;
// Warning: Do not break the follow section up!!!
quint32 layertype;
if (gvversion >= 4) {
in >> layertype;
if (layertype == LT_LineChart) {
in >> flags_enabled;
in >> plots_enabled;
in >> dot_enabled;
}
}
gGraph *g = nullptr;
if (version <= 2) {
continue;
// // Names were stored as translated strings, so look up title instead.
// g = nullptr;
// for (int z=0; z<m_graphs.size(); ++z) {
// if (m_graphs[z]->title() == name) {
// g = m_graphs[z];
// break;
// }
// }
} else {
gi = m_graphsbyname.find(name);
if (gi == m_graphsbyname.end()) {
qDebug() << "Graph" << name << "has been renamed or removed";
} else {
g = gi.value();
}
}
if (g) {
neworder.push_back(g);
g->setHeight(hght);
g->setVisible(vis);
g->setRecMinY(recminy);
g->setRecMaxY(recmaxy);
g->setZoomY(zoomy);
2014-04-05 01:28:07 +00:00
g->setPinned(pinned);
if (gvversion >= 4) {
if (layertype == LT_LineChart) {
gLineChart * lc = dynamic_cast<gLineChart *>(findLayer(g, LT_LineChart));
if (lc) {
hashMerge(lc->m_flags_enabled, flags_enabled);
hashMerge(lc->m_enabled, plots_enabled);
hashMerge(lc->m_dot_enabled, dot_enabled);
}
}
}
}
}
if (neworder.size() == m_graphs.size()) {
m_graphs = neworder;
}
f.close();
updateScale();
return true;
}
2011-11-27 22:36:38 +00:00
gGraph *gGraphView::findGraph(QString name)
{
auto it = m_graphsbyname.find(name);
if (it == m_graphsbyname.end()) { return nullptr; }
return it.value();
2011-11-27 22:36:38 +00:00
}
gGraph *gGraphView::findGraphTitle(QString title)
{
for (auto & graph : m_graphs) {
if (graph->title() == title) return graph;
}
return nullptr;
}
2011-12-01 15:57:06 +00:00
int gGraphView::visibleGraphs()
{
int cnt = 0;
for (auto & graph : m_graphs) {
if (graph->visible() && !graph->isEmpty()) { cnt++; }
2011-12-01 15:57:06 +00:00
}
2011-12-01 15:57:06 +00:00
return cnt;
}
2014-05-16 10:44:09 +00:00
void gGraphView::dataChanged()
{
for (auto & graph : m_graphs) {
graph->dataChanged();
}
}
2011-12-20 11:12:52 +00:00
void gGraphView::redraw()
{
#ifdef BROKEN_OPENGL_BUILD
repaint();
#else
update();
#endif
2011-12-20 11:12:52 +00:00
}