From 0899b391756bb1770c00947449ffb6731afe05e8 Mon Sep 17 00:00:00 2001 From: Sean Stangl Date: Wed, 23 Apr 2014 10:20:38 -0700 Subject: [PATCH] Split humongous gGraphView.cpp into class-specific files. Signed-off-by: Mark Watkins --- sleepyhead/Graphs/GLBuffer.cpp | 289 ++++ sleepyhead/Graphs/GLBuffer.h | 100 ++ sleepyhead/Graphs/gFooBar.cpp | 8 +- sleepyhead/Graphs/gFooBar.h | 3 +- sleepyhead/Graphs/gGraph.cpp | 1426 +++++++++++++++++ sleepyhead/Graphs/gGraph.h | 371 +++++ sleepyhead/Graphs/gGraphView.cpp | 2271 +-------------------------- sleepyhead/Graphs/gGraphView.h | 827 +--------- sleepyhead/Graphs/gLineChart.cpp | 13 +- sleepyhead/Graphs/gLineChart.h | 5 +- sleepyhead/Graphs/gVertexBuffer.cpp | 304 ++++ sleepyhead/Graphs/gVertexBuffer.h | 128 ++ sleepyhead/Graphs/gXAxis.cpp | 9 +- sleepyhead/Graphs/gXAxis.h | 3 +- sleepyhead/Graphs/gYAxis.cpp | 10 +- sleepyhead/Graphs/gYAxis.h | 3 +- sleepyhead/Graphs/glcommon.h | 10 + sleepyhead/Graphs/layer.cpp | 268 ++++ sleepyhead/Graphs/layer.h | 273 ++++ sleepyhead/mainwindow.cpp | 2 +- sleepyhead/oximetry.cpp | 1 + sleepyhead/oximetry.h | 2 +- sleepyhead/sleepyhead.pro | 166 +- 23 files changed, 3324 insertions(+), 3168 deletions(-) create mode 100644 sleepyhead/Graphs/GLBuffer.cpp create mode 100644 sleepyhead/Graphs/GLBuffer.h create mode 100644 sleepyhead/Graphs/gGraph.cpp create mode 100644 sleepyhead/Graphs/gGraph.h create mode 100644 sleepyhead/Graphs/gVertexBuffer.cpp create mode 100644 sleepyhead/Graphs/gVertexBuffer.h create mode 100644 sleepyhead/Graphs/layer.cpp create mode 100644 sleepyhead/Graphs/layer.h diff --git a/sleepyhead/Graphs/GLBuffer.cpp b/sleepyhead/Graphs/GLBuffer.cpp new file mode 100644 index 00000000..c3171d75 --- /dev/null +++ b/sleepyhead/Graphs/GLBuffer.cpp @@ -0,0 +1,289 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * Copyright (c) 2011-2014 Mark Watkins + * + * 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 Linux + * distribution for more details. */ + +#include "Graphs/GLBuffer.h" + +#include + +#include "SleepLib/profiles.h" + +extern int lines_drawn_this_frame; +extern int quads_drawn_this_frame; + +GLFloatBuffer::GLFloatBuffer(int max, int type, bool stippled) + : GLBuffer(max, type, stippled) +{ + buffer = new GLfloat[max + 8]; + colors = new GLubyte[max * 4 + (8 * 4)]; +} + +GLFloatBuffer::~GLFloatBuffer() +{ + if (colors) { + delete [] colors; + } + + if (buffer) { + delete [] buffer; + } +} + +void GLFloatBuffer::add(GLfloat x, GLfloat y, QColor &color) +{ +#ifdef ENABLE_THREADED_DRAWING + mutex.lock(); +#endif + if (m_cnt < m_max + 2) { + buffer[m_cnt++] = x; + buffer[m_cnt++] = y; + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + + } else { + qDebug() << "GLBuffer overflow"; + } +#ifdef ENABLE_THREADED_DRAWING + mutex.unlock(); +#endif +} + +void GLFloatBuffer::add(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, QColor &color) +{ + if (m_cnt < m_max + 4) { + qDebug() << "GLFloatBuffer overflow"; + return; + } + +#ifdef ENABLE_THREADED_DRAWING + mutex.lock(); +#endif + buffer[m_cnt++] = x1; + buffer[m_cnt++] = y1; + buffer[m_cnt++] = x2; + buffer[m_cnt++] = y2; + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); +#ifdef ENABLE_THREADED_DRAWING + mutex.unlock(); +#endif +} + +void GLFloatBuffer::add(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, GLfloat x3, GLfloat y3, + GLfloat x4, GLfloat y4, QColor &color) // add with vertex colors +{ + if (m_cnt >= m_max + 8) { + qDebug() << "GLFloatBuffer overflow"; + return; + } + +#ifdef ENABLE_THREADED_DRAWING + mutex.lock(); +#endif + buffer[m_cnt++] = x1; + buffer[m_cnt++] = y1; + buffer[m_cnt++] = x2; + buffer[m_cnt++] = y2; + buffer[m_cnt++] = x3; + buffer[m_cnt++] = y3; + buffer[m_cnt++] = x4; + buffer[m_cnt++] = y4; + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); +#ifdef ENABLE_THREADED_DRAWING + mutex.unlock(); +#endif +} + +void GLFloatBuffer::quadGrLR(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, GLfloat x3, + GLfloat y3, GLfloat x4, GLfloat y4, QColor &color, QColor &color2) // add with vertex colors +{ + if (m_cnt >= m_max + 8) { + qDebug() << "GLFloatBuffer overflow"; + return; + } + +#ifdef ENABLE_THREADED_DRAWING + mutex.lock(); +#endif + buffer[m_cnt++] = x1; + buffer[m_cnt++] = y1; + buffer[m_cnt++] = x2; + buffer[m_cnt++] = y2; + buffer[m_cnt++] = x3; + buffer[m_cnt++] = y3; + buffer[m_cnt++] = x4; + buffer[m_cnt++] = y4; + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + + colors[m_colcnt++] = color2.red(); + colors[m_colcnt++] = color2.green(); + colors[m_colcnt++] = color2.blue(); + colors[m_colcnt++] = color2.alpha(); + colors[m_colcnt++] = color2.red(); + colors[m_colcnt++] = color2.green(); + colors[m_colcnt++] = color2.blue(); + colors[m_colcnt++] = color2.alpha(); +#ifdef ENABLE_THREADED_DRAWING + mutex.unlock(); +#endif +} +void GLFloatBuffer::quadGrTB(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, GLfloat x3, + GLfloat y3, GLfloat x4, GLfloat y4, QColor &color, QColor &color2) +{ + if (m_cnt >= m_max + 8) { + qDebug() << "GLFloatBuffer overflow"; + return; + } + +#ifdef ENABLE_THREADED_DRAWING + mutex.lock(); +#endif + buffer[m_cnt++] = x1; + buffer[m_cnt++] = y1; + buffer[m_cnt++] = x3; + buffer[m_cnt++] = y3; + buffer[m_cnt++] = x2; + buffer[m_cnt++] = y2; + buffer[m_cnt++] = x4; + buffer[m_cnt++] = y4; + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + colors[m_colcnt++] = color.red(); + colors[m_colcnt++] = color.green(); + colors[m_colcnt++] = color.blue(); + colors[m_colcnt++] = color.alpha(); + + colors[m_colcnt++] = color2.red(); + colors[m_colcnt++] = color2.green(); + colors[m_colcnt++] = color2.blue(); + colors[m_colcnt++] = color2.alpha(); + colors[m_colcnt++] = color2.red(); + colors[m_colcnt++] = color2.green(); + colors[m_colcnt++] = color2.blue(); + colors[m_colcnt++] = color2.alpha(); +#ifdef ENABLE_THREADED_DRAWING + mutex.unlock(); +#endif +} + +void GLFloatBuffer::draw() +{ + if (m_cnt <= 0) { + return; + } + + bool antialias = m_forceantialias || (PROFILE.appearance->antiAliasing() && m_antialias); + float size = m_size; + + if (antialias) { + glEnable(GL_BLEND); + glBlendFunc(m_blendfunc1, m_blendfunc2); + + if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { + glEnable(GL_LINE_SMOOTH); + glHint(GL_LINE_SMOOTH_HINT, GL_NICEST); + size += 0.5; + } else if (m_type == GL_POLYGON) { + glEnable(GL_POLYGON_SMOOTH); + glHint(GL_POLYGON_SMOOTH_HINT, GL_NICEST); + } + } + + if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { + glLineWidth(size); + + if (m_stippled) { + glLineStipple(1, 0xAAAA); + glEnable(GL_LINE_STIPPLE); + } + + lines_drawn_this_frame += m_cnt / 2; + } else if (m_type == GL_POINTS) { + glPointSize(size); + } else if (m_type == GL_POLYGON) { + glPolygonMode(GL_BACK, GL_FILL); + lines_drawn_this_frame += m_cnt / 2; + } else if (m_type == GL_QUADS) { + quads_drawn_this_frame += m_cnt / 4; + } + + if (m_scissor) { + glScissor(s1, s2, s3, s4); + glEnable(GL_SCISSOR_TEST); + } + + glVertexPointer(2, GL_FLOAT, 0, buffer); + + glColorPointer(4, GL_UNSIGNED_BYTE, 0, colors); + + //glColor4ub(200,128,95,200); + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_COLOR_ARRAY); + + + glDrawArrays(m_type, 0, m_cnt >> 1); + glDisableClientState(GL_COLOR_ARRAY); + glDisableClientState(GL_VERTEX_ARRAY); + + //qDebug() << "I Drew" << m_cnt << "vertices"; + m_cnt = 0; + m_colcnt = 0; + + if (m_scissor) { + glDisable(GL_SCISSOR_TEST); + m_scissor = false; + } + + if (m_type == GL_POLYGON) { + glPolygonMode(GL_BACK, GL_FILL); + } + + if (antialias) { + if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { + if (m_stippled) { glDisable(GL_LINE_STIPPLE); } + + glDisable(GL_LINE_SMOOTH); + } else if (m_type == GL_POLYGON) { + glDisable(GL_POLYGON_SMOOTH); + } + + glDisable(GL_BLEND); + } +} diff --git a/sleepyhead/Graphs/GLBuffer.h b/sleepyhead/Graphs/GLBuffer.h new file mode 100644 index 00000000..b9d8c9a6 --- /dev/null +++ b/sleepyhead/Graphs/GLBuffer.h @@ -0,0 +1,100 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * Copyright (c) 2011-2014 Mark Watkins + * + * 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 Linux + * distribution for more details. */ + +#ifndef graphs_glbuffer_h +#define graphs_glbuffer_h + +#include +#include +#include + +/*! \class GLBuffer + \brief Base Object to hold an OpenGL draw list + */ +class GLBuffer +{ + public: + GLBuffer(int max = 2048, int type = GL_LINES, bool stippled = false) + : m_max(max), m_type(type), m_cnt(0), m_colcnt(0), m_size(1), + s1(0), s2(0), s3(0), s4(0), + m_scissor(false), + m_antialias(true), + m_forceantialias(false), + m_stippled(stippled), + m_blendfunc1(GL_SRC_ALPHA), + m_blendfunc2(GL_ONE_MINUS_SRC_ALPHA) + { } + virtual ~GLBuffer() {} + + void scissor(GLshort x1, GLshort y1, GLshort x2, GLshort y2) { + s1 = x1; + s2 = y1; + s3 = x2; + s4 = y2; + m_scissor = true; + } + + int Max() const { return m_max; } + int cnt() const { return m_cnt; } + bool full() const { return m_cnt >= m_max; } + float size() const { return m_size; } + int type() const { return m_type; } + + void reset() { m_cnt = 0; } + void setSize(float f) { m_size = f; } + void setAntiAlias(bool b) { m_antialias = b; } + void forceAntiAlias(bool b) { m_forceantialias = b; } + void setColor(QColor col) { m_color = col; } + void setBlendFunc(GLuint b1, GLuint b2) { m_blendfunc1 = b1; m_blendfunc2 = b2; } + + virtual void draw() {} + + protected: + int m_max; + int m_type; // type (GL_LINES, GL_QUADS, etc) + int m_cnt; // cnt + int m_colcnt; + QColor m_color; + float m_size; + int s1, s2, s3, s4; + bool m_scissor; + bool m_antialias; + bool m_forceantialias; + QMutex mutex; + bool m_stippled; + GLuint m_blendfunc1, m_blendfunc2; +}; + +/*! \class GLFloatBuffer + \brief Holds an OpenGL draw list composed of 32bit GLfloat objects and vertex colors + */ +class GLFloatBuffer : public GLBuffer +{ + public: + GLFloatBuffer(int max = 2048, int type = GL_LINES, bool stippled = false); + virtual ~GLFloatBuffer(); + + // Add with vertex color(s). + void add(GLfloat x, GLfloat y, QColor &col); + void add(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, QColor &col); + void add(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, + GLfloat x3, GLfloat y3, GLfloat x4, GLfloat y4, QColor &col); + void quadGrTB(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, + GLfloat x3, GLfloat y3, GLfloat x4, GLfloat y4, QColor &col, QColor &col2); + void quadGrLR(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, + GLfloat x3, GLfloat y3, GLfloat x4, GLfloat y4, QColor &col, QColor &col2); + + virtual void draw(); + + protected: + GLfloat *buffer; + GLubyte *colors; +}; + +#endif // graphs_glbuffer_h diff --git a/sleepyhead/Graphs/gFooBar.cpp b/sleepyhead/Graphs/gFooBar.cpp index 4fa208d1..f852d9bf 100644 --- a/sleepyhead/Graphs/gFooBar.cpp +++ b/sleepyhead/Graphs/gFooBar.cpp @@ -9,9 +9,13 @@ * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ +#include "Graphs/gFooBar.h" + #include -#include "gFooBar.h" -#include "gYAxis.h" + +#include "Graphs/gGraph.h" +#include "Graphs/gVertexBuffer.h" +#include "Graphs/gYAxis.h" gShadowArea::gShadowArea(QColor shadow_color, QColor line_color) : Layer(NoChannel), m_shadow_color(shadow_color), m_line_color(line_color) diff --git a/sleepyhead/Graphs/gFooBar.h b/sleepyhead/Graphs/gFooBar.h index 05187f41..d424be46 100644 --- a/sleepyhead/Graphs/gFooBar.h +++ b/sleepyhead/Graphs/gFooBar.h @@ -12,7 +12,8 @@ #ifndef GFOOBAR_H #define GFOOBAR_H -#include "gGraphView.h" +#include "Graphs/gVertexBuffer.h" +#include "Graphs/layer.h" /*! \class gShadowArea \brief Displays a Shadow for all graph areas not highlighted (used in Event Flags) diff --git a/sleepyhead/Graphs/gGraph.cpp b/sleepyhead/Graphs/gGraph.cpp new file mode 100644 index 00000000..561f281d --- /dev/null +++ b/sleepyhead/Graphs/gGraph.cpp @@ -0,0 +1,1426 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * Copyright (c) 2011-2014 Mark Watkins + * + * 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 Linux + * distribution for more details. */ + +#include "Graphs/gGraph.h" + +#include +#include + +#include "mainwindow.h" +#include "Graphs/gGraphView.h" +#include "Graphs/layer.h" +#include "SleepLib/profiles.h" + +extern QLabel *qstatus2; +extern MainWindow *mainwin; + +// Graph globals. +QFont *defaultfont = nullptr; +QFont *mediumfont = nullptr; +QFont *bigfont = nullptr; +QHash images; +bool fbo_unsupported = false; +QGLFramebufferObject *fbo = nullptr; + +static bool globalsInitialized = false; + +// Graph constants. +static const double zoom_hard_limit = 500.0; + + // Must be called from a thread inside the application. +bool InitGraphGlobals() +{ + if (globalsInitialized) { + return true; + } + + if (!PREF.contains("Fonts_Graph_Name")) { + PREF["Fonts_Graph_Name"] = "Sans Serif"; + PREF["Fonts_Graph_Size"] = 10; + PREF["Fonts_Graph_Bold"] = false; + PREF["Fonts_Graph_Italic"] = false; + } + + if (!PREF.contains("Fonts_Title_Name")) { + PREF["Fonts_Title_Name"] = "Sans Serif"; + PREF["Fonts_Title_Size"] = 14; + PREF["Fonts_Title_Bold"] = true; + PREF["Fonts_Title_Italic"] = false; + } + + if (!PREF.contains("Fonts_Big_Name")) { + PREF["Fonts_Big_Name"] = "Serif"; + PREF["Fonts_Big_Size"] = 35; + PREF["Fonts_Big_Bold"] = false; + PREF["Fonts_Big_Italic"] = false; + } + + defaultfont = new QFont(PREF["Fonts_Graph_Name"].toString(), + PREF["Fonts_Graph_Size"].toInt(), + PREF["Fonts_Graph_Bold"].toBool() ? QFont::Bold : QFont::Normal, + PREF["Fonts_Graph_Italic"].toBool() + ); + mediumfont = new QFont(PREF["Fonts_Title_Name"].toString(), + PREF["Fonts_Title_Size"].toInt(), + PREF["Fonts_Title_Bold"].toBool() ? QFont::Bold : QFont::Normal, + PREF["Fonts_Title_Italic"].toBool() + ); + bigfont = new QFont(PREF["Fonts_Big_Name"].toString(), + PREF["Fonts_Big_Size"].toInt(), + PREF["Fonts_Big_Bold"].toBool() ? QFont::Bold : QFont::Normal, + PREF["Fonts_Big_Italic"].toBool() + ); + + defaultfont->setStyleHint(QFont::AnyStyle, QFont::OpenGLCompatible); + mediumfont->setStyleHint(QFont::AnyStyle, QFont::OpenGLCompatible); + bigfont->setStyleHint(QFont::AnyStyle, QFont::OpenGLCompatible); + + //images["mask"] = new QImage(":/icons/mask.png"); + images["oximeter"] = new QImage(":/icons/cubeoximeter.png"); + images["smiley"] = new QImage(":/icons/smileyface.png"); + //images["sad"] = new QImage(":/icons/sadface.png"); + + images["sheep"] = new QImage(":/icons/sheep.png"); + images["brick"] = new QImage(":/icons/brick.png"); + images["nographs"] = new QImage(":/icons/nographs.png"); + images["nodata"] = new QImage(":/icons/nodata.png"); + + globalsInitialized = true; + return true; +} + +void DestroyGraphGlobals() +{ + if (!globalsInitialized) { + return; + } + + delete defaultfont; + delete bigfont; + delete mediumfont; + + for (QHash::iterator i = images.begin(); i != images.end(); i++) { + delete i.value(); + } + + // Clear the frame buffer object. + if (fbo) { + if (fbo->isBound()) { + fbo->release(); + } + + delete fbo; + fbo = nullptr; + fbo_unsupported = true; // just in case shutdown order gets messed up + } + + globalsInitialized = false; +} + +gGraph::gGraph(gGraphView *graphview, QString title, QString units, int height, short group) + : m_graphview(graphview), + m_title(title), + m_units(units), + m_height(height), + m_visible(true) +{ + m_min_height = 60; + m_width = 0; + + m_layers.clear(); + + f_miny = f_maxy = 0; + rmin_x = rmin_y = 0; + rmax_x = rmax_y = 0; + max_x = max_y = 0; + min_x = min_y = 0; + rec_miny = rec_maxy = 0; + rphysmax_y = rphysmin_y = 0; + m_zoomY = 0; + + if (graphview) { + graphview->addGraph(this, group); + timer = new QTimer(graphview); + connect(timer, SIGNAL(timeout()), SLOT(Timeout())); + } else { + qWarning() << "gGraph created without a gGraphView container.. Naughty programmer!! Bad!!!"; + } + + m_margintop = 14; + m_marginbottom = 5; + m_marginleft = 0; + m_marginright = 15; + m_selecting_area = m_blockzoom = false; + m_pinned = false; + m_lastx23 = 0; + + invalidate_yAxisImage = true; + invalidate_xAxisImage = true; + + m_quad = new gVertexBuffer(64, GL_QUADS); + m_quad->forceAntiAlias(true); + m_enforceMinY = m_enforceMaxY = false; + m_showTitle = true; + m_printing = false; +} +gGraph::~gGraph() +{ + for (int i = 0; i < m_layers.size(); i++) { + if (m_layers[i]->unref()) { + delete m_layers[i]; + } + } + + m_layers.clear(); + delete m_quad; + + timer->stop(); + disconnect(timer, 0, 0, 0); + delete timer; +} +void gGraph::Trigger(int ms) +{ + if (timer->isActive()) { timer->stop(); } + + timer->setSingleShot(true); + timer->start(ms); +} +void gGraph::Timeout() +{ + deselect(); + m_graphview->timedRedraw(0); +} + +void gGraph::deselect() +{ + for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { + (*l)->deselect(); + } +} +bool gGraph::isSelected() +{ + bool res = false; + + for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { + res = (*l)->isSelected(); + + if (res) { break; } + } + + return res; +} + +bool gGraph::isEmpty() +{ + bool empty = true; + + for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { + if (!(*l)->isEmpty()) { + empty = false; + break; + } + } + + return empty; +} + + +float gGraph::printScaleX() { return m_graphview->printScaleX(); } +float gGraph::printScaleY() { return m_graphview->printScaleY(); } + + +void gGraph::drawGLBuf() +{ + + float linesize = 1; + + if (m_printing) { linesize = 4; } //ceil(m_graphview->printScaleY()); + + for (int i = 0; i < m_layers.size(); i++) { + m_layers[i]->drawGLBuf(linesize); + } + + m_quad->draw(); +} +void gGraph::setDay(Day *day) +{ + m_day = day; + + for (int i = 0; i < m_layers.size(); i++) { + m_layers[i]->SetDay(day); + } + + ResetBounds(); +} + +void gGraph::setZoomY(short zoom) +{ + m_zoomY = zoom; + redraw(); +} + + +void gGraph::qglColor(QColor col) +{ + m_graphview->qglColor(col); +} + +void gGraph::renderText(QString text, int x, int y, float angle, QColor color, QFont *font, + bool antialias) +{ + m_graphview->AddTextQue(text, x, y, angle, color, font, antialias); +} + +void gGraph::paint(int originX, int originY, int width, int height) +{ + m_rect = QRect(originX, originY, width, height); + + int fw, font_height; + GetTextExtent("Wg@", fw, font_height); + + if (m_margintop > 0) { m_margintop = font_height + (8 * printScaleY()); } + + //m_marginbottom=5; + + //glColor4f(0,0,0,1); + left = marginLeft(), right = marginRight(), top = marginTop(), bottom = marginBottom(); + int x = 0, y = 0; + + if (m_showTitle) { + int title_x, yh; + /* if (titleImage.isNull()) { + // Render the title to a texture so we don't have to draw the vertical text every time.. + + GetTextExtent("Wy@",x,yh,mediumfont); // This gets a better consistent height. should be cached. + + GetTextExtent(title(),x,y,mediumfont); + + y=yh; + QPixmap tpm=QPixmap(x+4,y+4); + + tpm.fill(Qt::transparent); //empty it + QPainter pmp(&tpm); + + pmp.setRenderHint(QPainter::TextAntialiasing, true); + + QBrush brush2(Qt::black); // text color + pmp.setBrush(brush2); + + pmp.setFont(*mediumfont); + + pmp.drawText(2,y,title()); // draw from the bottom + pmp.end(); + + // convert to QImage and bind to a texture for future use + titleImage=QGLWidget::convertToGLFormat(tpm.toImage().mirrored(false,true)); + titleImageTex=m_graphview->bindTexture(titleImage); + } + y=titleImage.height(); + x=titleImage.width(); //vertical text + + glEnable(GL_BLEND); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + + title_x=y*2; + + glEnable(GL_TEXTURE_2D); + + // Rotate and draw vertical texture containing graph titles + glPushMatrix(); + glTranslatef(marginLeft()+4,originY+height/2+x/2, 0); + glRotatef(-90,0,0,1); + m_graphview->drawTexture(QPoint(0,y/2),titleImageTex); + glPopMatrix(); + + glDisable(GL_TEXTURE_2D); + glDisable(GL_BLEND); + + // All that to replace this little, but -hideously- slow line of text.. + */ + + QFontMetrics fm(*mediumfont); + + yh = fm.height(); + //GetTextExtent("Wy@",x,yh,mediumfont); // This gets a better consistent height. should be cached. + y = yh; + x = fm.width(title()); + //GetTextExtent(title(),x,y,mediumfont); + title_x = yh * 2; + + + renderText(title(), marginLeft() + title_x + 4, originY + height / 2 - y / 2, 90, Qt::black, + mediumfont); + left += title_x; + } else { left = 0; } + + //#define DEBUG_LAYOUT +#ifdef DEBUG_LAYOUT + QColor col = Qt::red; + lines()->add(0, originY, 0, originY + height, col); + lines()->add(left, originY, left, originY + height, col); +#endif + int tmp; + + left = 0; + + for (int i = 0; i < m_layers.size(); i++) { + Layer *ll = m_layers[i]; + + if (!ll->visible()) { continue; } + + tmp = ll->Height() * m_graphview->printScaleY(); + + if (ll->position() == LayerTop) { top += tmp; } + + if (ll->position() == LayerBottom) { bottom += tmp; } + } + + + for (int i = 0; i < m_layers.size(); i++) { + Layer *ll = m_layers[i]; + + if (!ll->visible()) { continue; } + + tmp = ll->Width() * m_graphview->printScaleX(); + + if (ll->position() == LayerLeft) { + ll->m_rect = QRect(originX + left, originY + top, tmp, height - top - bottom); + ll->paint(*this, originX + left, originY + top, tmp, height - top - bottom); + left += tmp; +#ifdef DEBUG_LAYOUT + lines()->add(originX + left - 1, originY, originX + left - 1, originY + height, col); +#endif + } + + if (ll->position() == LayerRight) { + right += tmp; + ll->m_rect = QRect(originX + width - right, originY + top, tmp, height - top - bottom); + ll->paint(*this, originX + width - right, originY + top, tmp, height - top - bottom); +#ifdef DEBUG_LAYOUT + lines()->add(originX + width - right, originY, originX + width - right, originY + height, col); +#endif + } + } + + bottom = marginBottom(); + top = marginTop(); + + for (int i = 0; i < m_layers.size(); i++) { + Layer *ll = m_layers[i]; + + if (!ll->visible()) { continue; } + + tmp = ll->Height() * m_graphview->printScaleY(); + + if (ll->position() == LayerTop) { + ll->m_rect = QRect(originX + left, originY + top, width - left - right, tmp); + ll->paint(*this, originX + left, originY + top, width - left - right, tmp); + top += tmp; + } + + if (ll->position() == LayerBottom) { + bottom += tmp; + ll->m_rect = QRect(originX + left, originY + height - bottom, width - left - right, tmp); + ll->paint(*this, originX + left, originY + height - bottom, width - left - right, tmp); + } + } + + if (isPinned()) { + // Fill the background on pinned graphs + + // m_graphview->quads->add(originX+left,originY+top, originX+width-right,originY+top, originX+width-right,originY+height-bottom, originX+left,originY+height-bottom, 0xffffffff); + glBegin(GL_QUADS); + glColor4f(1.0, 1.0, 1.0, 1.0); // Gradient End + glVertex2i(originX + left, originY + top); + glVertex2i(originX + width - right, originY + top); + glVertex2i(originX + width - right, originY + height - bottom); + glVertex2i(originX + left, originY + height - bottom); + glEnd(); + } + + for (int i = 0; i < m_layers.size(); i++) { + Layer *ll = m_layers[i]; + + if (!ll->visible()) { continue; } + + if (ll->position() == LayerCenter) { + ll->m_rect = QRect(originX + left, originY + top, width - left - right, height - top - bottom); + ll->paint(*this, originX + left, originY + top, width - left - right, height - top - bottom); + } + } + + if (m_selection.width() > 0 && m_selecting_area) { + QColor col(128, 128, 255, 128); + quads()->add(originX + m_selection.x(), originY + top, + originX + m_selection.x() + m_selection.width(), originY + top, col.rgba()); + quads()->add(originX + m_selection.x() + m_selection.width(), originY + height - bottom, + originX + m_selection.x(), originY + height - bottom, col.rgba()); + } +} + +QPixmap gGraph::renderPixmap(int w, int h, bool printing) +{ + QPixmap pm = QPixmap(); + + gGraphView *sg = mainwin->snapshotGraph(); + + if (!sg) { + return QPixmap(); + } + + // Pixmap caching screws up font sizes when printing + sg->setUsePixmapCache(false); + + QFont *_defaultfont = defaultfont; + QFont *_mediumfont = mediumfont; + QFont *_bigfont = bigfont; + + QFont fa = *defaultfont; + QFont fb = *mediumfont; + QFont fc = *bigfont; + + + m_printing = printing; + + if (printing) { + fa.setPixelSize(30); + fb.setPixelSize(35); + fc.setPixelSize(80); + sg->setPrintScaleX(3); + sg->setPrintScaleY(3); + } else { + sg->setPrintScaleX(1); + sg->setPrintScaleY(1); + } + + defaultfont = &fa; + mediumfont = &fb; + bigfont = &fc; + + sg->hideSplitter(); + gGraphView *tgv = m_graphview; + m_graphview = sg; + + sg->setMinimumSize(w, h); + sg->setMaximumSize(w, h); + sg->setFixedSize(w, h); + + float tmp = m_height; + m_height = h; + sg->trashGraphs(); + sg->addGraph(this); + + sg->setScaleY(1.0); + + + sg->makeCurrent(); // has to be current for fbo creation + + float dpr = sg->devicePixelRatio(); + sg->setDevicePixelRatio(1); +#ifdef Q_OS_WIN + + if (pm.isNull()) { + pm = sg->renderPixmap(w, h, false); + } + + if (pm.isNull()) { // Works, but gives shader warnings + pm = QPixmap::fromImage(sg->pbRenderPixmap(w, h)); + } + + if (pm.isNull()) { // crashes on mine, not sure what to do about it + pm = QPixmap::fromImage(sg->fboRenderPixmap(w, h)); + } + +#else + pm = QPixmap::fromImage(sg->fboRenderPixmap(w, h)); + + if (pm.isNull()) { // not sure if this will work with printing + qDebug() << "Had to use PixelBuffer for snapshots\n"; + pm = QPixmap::fromImage(sg->pbRenderPixmap(w, h)); + } + +#endif + sg->setDevicePixelRatio(dpr); + //sg->doneCurrent(); + sg->trashGraphs(); + + m_graphview = tgv; + + m_height = tmp; + + defaultfont = _defaultfont; + mediumfont = _mediumfont; + bigfont = _bigfont; + m_printing = false; + + return pm; +} + +// Sets a new Min & Max X clipping, refreshing the graph and all it's layers. +void gGraph::SetXBounds(qint64 minx, qint64 maxx) +{ + invalidate_xAxisImage = true; + min_x = minx; + max_x = maxx; + + //repaint(); + //m_graphview->redraw(); +} + +int gGraph::flipY(int y) +{ + return m_graphview->height() - y; +} + +void gGraph::ResetBounds() +{ + invalidate_xAxisImage = true; + min_x = MinX(); + max_x = MaxX(); + min_y = MinY(); + max_y = MaxY(); +} + +void gGraph::ToolTip(QString text, int x, int y, int timeout) +{ + if (timeout <= 0) { + timeout = p_profile->general->tooltipTimeout(); + } + + m_graphview->m_tooltip->display(text, x, y, timeout); +} + +// YAxis Autoscaling code +void gGraph::roundY(EventDataType &miny, EventDataType &maxy) +{ + int m, t; + bool ymin_good = false, ymax_good = false; + + if (rec_miny != rec_maxy) { + if (miny > rec_miny) { + miny = rec_miny; + } + + if (maxy < rec_maxy) { + maxy = rec_maxy; + } + + if (miny == rec_miny) { + ymin_good = true; + } + + if (maxy == rec_maxy) { + ymax_good = true; + } + } + + if (maxy == miny) { + m = ceil(maxy / 2.0); + t = m * 2; + + if (maxy == t) { + t += 2; + } + + if (!ymax_good) { + maxy = t; + } + + m = floor(miny / 2.0); + t = m * 2; + + if (miny == t) { + t -= 2; + } + + if (miny >= 0 && t < 0) { + t = 0; + } + + if (!ymin_good) { + miny = t; + } + + return; + } + + if (maxy >= 400) { + m = ceil(maxy / 50.0); + t = m * 50; + + if (!ymax_good) { + maxy = t; + } + + m = floor(miny / 50.0); + + if (!ymin_good) { + miny = m * 50; + } + } else if (maxy >= 5) { + m = ceil(maxy / 5.0); + t = m * 5; + + if (!ymax_good) { + maxy = t; + } + + m = floor(miny / 5.0); + + if (!ymin_good) { + miny = m * 5; + } + } else { + if (maxy == miny && maxy == 0) { + maxy = 0.5; + } else { + //maxy*=4.0; + //miny*=4.0; + if (!ymax_good) { + maxy = ceil(maxy); + } + + if (!ymin_good) { + miny = floor(miny); + } + + //maxy/=4.0; + //miny/=4.0; + } + } + + //if (m_enforceMinY) { miny=f_miny; } + //if (m_enforceMaxY) { maxy=f_maxy; } +} + +void gGraph::AddLayer(Layer *l, LayerPosition position, short width, short height, short order, + bool movable, short x, short y) +{ + l->setLayout(position, width, height, order); + l->setMovable(movable); + l->setPos(x, y); + l->addref(); + m_layers.push_back(l); +} +void gGraph::redraw() +{ + m_graphview->redraw(); +} +void gGraph::timedRedraw(int ms) +{ + m_graphview->timedRedraw(ms); +} + +void gGraph::mouseMoveEvent(QMouseEvent *event) +{ + // qDebug() << m_title << "Move" << event->pos() << m_graphview->pointClicked(); + int y = event->y(); + int x = event->x(); + + bool doredraw = false; + + for (int i = 0; i < m_layers.size(); i++) { + if (m_layers[i]->m_rect.contains(x, y)) + if (m_layers[i]->mouseMoveEvent(event, this)) { + doredraw = true; + } + } + + y -= m_rect.top(); + x -= m_rect.left(); + + int x2 = m_graphview->pointClicked().x() - m_rect.left(); + + int w = m_rect.width() - (left + right); + //int h=m_lastbounds.height()-(bottom+m_marginbottom); + double xx = max_x - min_x; + double xmult = xx / w; + + if (m_graphview->m_selected_graph == this) { // Left Mouse button dragging + if (event->buttons() & Qt::LeftButton) { + //qDebug() << m_title << "Moved" << x << y << left << right << top << bottom << m_width << h; + int a1 = MIN(x, x2); + int a2 = MAX(x, x2); + + if (a1 < left) { a1 = left; } + + if (a2 > left + w) { a2 = left + w; } + + m_selecting_area = true; + m_selection = QRect(a1 - 1, 0, a2 - a1, m_rect.height()); + double w2 = m_rect.width() - right - left; + + if (m_blockzoom) { + xmult = (rmax_x - rmin_x) / w2; + } else { + xmult = (max_x - min_x) / w2; + } + + qint64 a = double(a2 - a1) * xmult; + float d = double(a) / 86400000.0; + int h = a / 3600000; + int m = (a / 60000) % 60; + int s = (a / 1000) % 60; + int ms(a % 1000); + QString str; + + if (d > 1) { + str.sprintf("%1.0f days", d); + } else { + + str.sprintf("%02i:%02i:%02i:%03i", h, m, s, ms); + } + + if (qstatus2) { + qstatus2->setText(str); + } + + doredraw = true; + } else if (event->buttons() & Qt::RightButton) { // Right Mouse button dragging + m_graphview->setPointClicked(event->pos()); + x -= left; + x2 -= left; + + if (!m_blockzoom) { + xx = max_x - min_x; + xmult = xx / double(w); + qint64 j1 = xmult * x; + qint64 j2 = xmult * x2; + qint64 jj = j2 - j1; + min_x += jj; + max_x += jj; + + if (min_x < rmin_x) { + min_x = rmin_x; + max_x = rmin_x + xx; + } + + if (max_x > rmax_x) { + max_x = rmax_x; + min_x = rmax_x - xx; + } + + m_graphview->SetXBounds(min_x, max_x, m_group, false); + doredraw = true; + } else { + qint64 qq = rmax_x - rmin_x; + xx = max_x - min_x; + + if (xx == qq) { xx = 1800000; } + + xmult = qq / double(w); + qint64 j1 = (xmult * x); + min_x = rmin_x + j1 - (xx / 2); + max_x = min_x + xx; + + if (min_x < rmin_x) { + min_x = rmin_x; + max_x = rmin_x + xx; + } + + if (max_x > rmax_x) { + max_x = rmax_x; + min_x = rmax_x - xx; + } + + m_graphview->SetXBounds(min_x, max_x, m_group, false); + doredraw = true; + } + } + } + + //if (!nolayer) { // no mouse button + if (doredraw) { + m_graphview->redraw(); + } + + //} + //if (x>left+m_marginleft && xtop+m_margintop && ypos().y(); + int x = event->pos().x(); + + for (int i = 0; i < m_layers.size(); i++) { + if (m_layers[i]->m_rect.contains(x, y)) + if (m_layers[i]->mousePressEvent(event, this)) { + return; + } + } + + /* + int w=m_lastbounds.width()-(right+m_marginright); + //int h=m_lastbounds.height()-(bottom+m_marginbottom); + //int x2,y2; + double xx=max_x-min_x; + //double xmult=xx/w; + if (x>left+m_marginleft && xtop+m_margintop && ypos().y(); + int x = event->pos().x(); + + for (int i = 0; i < m_layers.size(); i++) { + if (m_layers[i]->m_rect.contains(x, y)) + if (m_layers[i]->mouseReleaseEvent(event, this)) { + return; + } + } + + x -= m_rect.left(); + y -= m_rect.top(); + + + int w = m_rect.width() - left - right; //(m_marginleft+left+right+m_marginright); + int h = m_rect.height() - bottom; //+m_marginbottom); + + int x2 = m_graphview->pointClicked().x() - m_rect.left(); + int y2 = m_graphview->pointClicked().y() - m_rect.top(); + + + //qDebug() << m_title << "Released" << min_x << max_x << x << y << x2 << y2 << left << right << top << bottom << m_width << m_height; + if (m_selecting_area) { + m_selecting_area = false; + m_selection.setWidth(0); + + if (m_graphview->horizTravel() > mouse_movement_threshold) { + x -= left; //+m_marginleft; + y -= top; //+m_margintop; + x2 -= left; //+m_marginleft; + y2 -= top; //+m_margintop; + + if (x < 0) { x = 0; } + + if (x2 < 0) { x2 = 0; } + + if (x > w) { x = w; } + + if (x2 > w) { x2 = w; } + + double xx; + double xmult; + + if (!m_blockzoom) { + xx = max_x - min_x; + xmult = xx / double(w); + qint64 j1 = min_x + xmult * x; + qint64 j2 = min_x + xmult * x2; + qint64 a1 = MIN(j1, j2) + qint64 a2 = MAX(j1, j2) + + //if (a1 rmax_x) { a2 = rmax_x; } + + if (a1 <= rmin_x && a2 <= rmin_x) { + //qDebug() << "Foo??"; + } else { + if (a2 - a1 < zoom_hard_limit) { a2 = a1 + zoom_hard_limit; } + + m_graphview->SetXBounds(a1, a2, m_group); + } + } else { + xx = rmax_x - rmin_x; + xmult = xx / double(w); + qint64 j1 = rmin_x + xmult * x; + qint64 j2 = rmin_x + xmult * x2; + qint64 a1 = MIN(j1, j2) + qint64 a2 = MAX(j1, j2) + + //if (a1 rmax_x) { a2 = rmax_x; } + + if (a1 <= rmin_x && a2 <= rmin_x) { + qDebug() << "Foo2??"; + } else { + if (a2 - a1 < zoom_hard_limit) { a2 = a1 + zoom_hard_limit; } + + m_graphview->SetXBounds(a1, a2, m_group); + } + } + + return; + } else { m_graphview->redraw(); } + } + + if ((m_graphview->horizTravel() < mouse_movement_threshold) && (x > left && x < w + left + && y > top && y < h)) { + // normal click in main area + if (!m_blockzoom) { + double zoom; + + if (event->button() & Qt::RightButton) { + zoom = 1.33; + + if (event->modifiers() & Qt::ControlModifier) { zoom *= 1.5; } + + ZoomX(zoom, x); // Zoom out + return; + } else if (event->button() & Qt::LeftButton) { + zoom = 0.75; + + if (event->modifiers() & Qt::ControlModifier) { zoom /= 1.5; } + + ZoomX(zoom, x); // zoom in. + return; + } + } else { + x -= left; + y -= top; + //w-=m_marginleft+left; + double qq = rmax_x - rmin_x; + double xmult; + + double xx = max_x - min_x; + //if (xx==qq) xx=1800000; + + xmult = qq / double(w); + + if ((xx == qq) || (x == m_lastx23)) { + double zoom = 1; + + if (event->button() & Qt::RightButton) { + zoom = 1.33; + + if (event->modifiers() & Qt::ControlModifier) { zoom *= 1.5; } + } else if (event->button() & Qt::LeftButton) { + zoom = 0.75; + + if (event->modifiers() & Qt::ControlModifier) { zoom /= 1.5; } + } + + xx *= zoom; + + if (xx < qq / zoom_hard_limit) { xx = qq / zoom_hard_limit; } + + if (xx > qq) { xx = qq; } + } + + double j1 = xmult * x; + min_x = rmin_x + j1 - (xx / 2.0); + max_x = min_x + xx; + + if (min_x < rmin_x) { + min_x = rmin_x; + max_x = rmin_x + xx; + } else if (max_x > rmax_x) { + max_x = rmax_x; + min_x = rmax_x - xx; + } + + m_graphview->SetXBounds(min_x, max_x, m_group); + m_lastx23 = x; + } + } + + //m_graphview->redraw(); +} + + +void gGraph::wheelEvent(QWheelEvent *event) +{ + //qDebug() << m_title << "Wheel" << event->x() << event->y() << event->delta(); + //int y=event->pos().y(); + if (event->orientation() == Qt::Horizontal) { + + return; + } + + int x = event->pos().x() - m_graphview->titleWidth; //(left+m_marginleft); + + if (event->delta() > 0) { + ZoomX(0.75, x); + } else { + ZoomX(1.5, x); + } + + int y = event->pos().y(); + x = event->pos().x(); + + for (int i = 0; i < m_layers.size(); i++) { + if (m_layers[i]->m_rect.contains(x, y)) { + m_layers[i]->wheelEvent(event, this); + } + } + +} +void gGraph::mouseDoubleClickEvent(QMouseEvent *event) +{ + //mousePressEvent(event); + //mouseReleaseEvent(event); + int y = event->pos().y(); + int x = event->pos().x(); + + for (int i = 0; i < m_layers.size(); i++) { + if (m_layers[i]->m_rect.contains(x, y)) { + m_layers[i]->mouseDoubleClickEvent(event, this); + } + } + + //int w=m_lastbounds.width()-(m_marginleft+left+right+m_marginright); + //int h=m_lastbounds.height()-(bottom+m_marginbottom); + //int x2=m_graphview->pointClicked().x(),y2=m_graphview->pointClicked().y(); + // if ((m_graphview->horizTravel()left+m_marginleft && xtop+m_margintop && ybutton() & Qt::RightButton) { + // ZoomX(1.66,x); // Zoon out + // return; + // } else if (event->button() & Qt::LeftButton) { + // ZoomX(0.75/2.0,x); // zoom in. + // return; + // } + // } else { + // Propagate the events to graph Layers + // } + //mousePressEvent(event); + //mouseReleaseEvent(event); + //qDebug() << m_title << "Double Clicked" << event->x() << event->y(); +} +void gGraph::keyPressEvent(QKeyEvent *event) +{ + for (QVector::iterator i = m_layers.begin(); i != m_layers.end(); i++) { + (*i)->keyPressEvent(event, this); + } + + //qDebug() << m_title << "Key Pressed.. implement me" << event->key(); +} + +void gGraph::ZoomX(double mult, int origin_px) +{ + + int width = m_rect.width() - left - right; //(m_marginleft+left+right+m_marginright); + + if (origin_px == 0) { origin_px = (width / 2); } + else { origin_px -= left; } + + if (origin_px < 0) { origin_px = 0; } + + if (origin_px > width) { origin_px = width; } + + + // Okay, I want it to zoom in centered on the mouse click area.. + // Find X graph position of mouse click + // find current zoom width + // apply zoom + // center on point found in step 1. + + qint64 min = min_x; + qint64 max = max_x; + + double hardspan = rmax_x - rmin_x; + double span = max - min; + double ww = double(origin_px) / double(width); + double origin = ww * span; + //double center=0.5*span; + //double dist=(origin-center); + + double q = span * mult; + + 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 < rmin_x) { + min = rmin_x; + max = min + q; + } + + if (max > rmax_x) { + max = rmax_x; + min = max - q; + } + + m_graphview->SetXBounds(min, max, m_group); + //updateSelectionTime(max-min); +} + +void gGraph::DrawTextQue() +{ + m_graphview->DrawTextQue(); +} + +// margin recalcs.. +void gGraph::resize(int width, int height) +{ + invalidate_xAxisImage = true; + invalidate_yAxisImage = true; + + Q_UNUSED(width); + Q_UNUSED(height); + //m_height=height; + //m_width=width; +} + +qint64 gGraph::MinX() +{ + qint64 val = 0, tmp; + + for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { + if ((*l)->isEmpty()) { + continue; + } + + tmp = (*l)->Minx(); + + if (!tmp) { + continue; + } + + if (!val || tmp < val) { + val = tmp; + } + } + + if (val) { rmin_x = val; } + + return val; +} +qint64 gGraph::MaxX() +{ + //bool first=true; + qint64 val = 0, tmp; + + for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { + if ((*l)->isEmpty()) { + continue; + } + + tmp = (*l)->Maxx(); + + //if (!tmp) continue; + if (!val || tmp > val) { + val = tmp; + } + } + + if (val) { rmax_x = val; } + + return val; +} + +EventDataType gGraph::MinY() +{ + bool first = true; + EventDataType val = 0, tmp; + + if (m_enforceMinY) { + return rmin_y = f_miny; + } + + for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { + if ((*l)->isEmpty()) { + continue; + } + + tmp = (*l)->Miny(); + + if (tmp == 0 && tmp == (*l)->Maxy()) { + continue; + } + + if (first) { + val = tmp; + first = false; + } else { + if (tmp < val) { + val = tmp; + } + } + } + + return rmin_y = val; +} +EventDataType gGraph::MaxY() +{ + bool first = true; + EventDataType val = 0, tmp; + + if (m_enforceMaxY) { + return rmax_y = f_maxy; + } + + auto iterEnd = m_layers.constEnd(); + for (auto iter = m_layers.constBegin(); iter != iterEnd; ++iter) { + Layer *layer = *iter; + if (layer->isEmpty()) { + continue; + } + + tmp = layer->Maxy(); + if (tmp == 0 && layer->Miny() == 0) { + continue; + } + + if (first) { + val = tmp; + first = false; + } else { + if (tmp > val) { + val = tmp; + } + } + } + + return rmax_y = val; +} + +EventDataType gGraph::physMinY() +{ + bool first = true; + EventDataType val = 0, tmp; + + //if (m_enforceMinY) return rmin_y=f_miny; + + auto iterEnd = m_layers.constEnd(); + for (auto iter = m_layers.constBegin(); iter != iterEnd; ++iter) { + Layer *layer = *iter; + if (layer->isEmpty()) { + continue; + } + + tmp = layer->physMiny(); + if (tmp == 0 && layer->physMaxy() == 0) { + continue; + } + + if (first) { + val = tmp; + first = false; + } else { + if (tmp < val) { + val = tmp; + } + } + } + + return rphysmin_y = val; +} +EventDataType gGraph::physMaxY() +{ + bool first = true; + EventDataType val = 0, tmp; + + // if (m_enforceMaxY) return rmax_y=f_maxy; + + auto iterEnd = m_layers.constEnd(); + for (auto iter = m_layers.constBegin(); iter != iterEnd; ++iter) { + Layer *layer = *iter; + if (layer->isEmpty()) { + continue; + } + + tmp = layer->physMaxy(); + if (tmp == 0 && layer->physMiny() == 0) { + continue; + } + + if (first) { + val = tmp; + first = false; + } else { + if (tmp > val) { + val = tmp; + } + } + } + + return rphysmax_y = val; +} + +void gGraph::SetMinX(qint64 v) +{ + rmin_x = min_x = v; +} + +void gGraph::SetMaxX(qint64 v) +{ + rmax_x = max_x = v; +} + +void gGraph::SetMinY(EventDataType v) +{ + rmin_y = min_y = v; +} + +void gGraph::SetMaxY(EventDataType v) +{ + rmax_y = max_y = v; +} + +gVertexBuffer *gGraph::lines() +{ + return m_graphview->lines; +} + +gVertexBuffer *gGraph::backlines() +{ + return m_graphview->backlines; +} + +gVertexBuffer *gGraph::quads() +{ + return m_graphview->quads; +} + +short gGraph::marginLeft() { return m_marginleft; }//*m_graphview->printScaleX(); } +short gGraph::marginRight() { return m_marginright; } //*m_graphview->printScaleX(); } +short gGraph::marginTop() { return m_margintop; } //*m_graphview->printScaleY(); } +short gGraph::marginBottom() { return m_marginbottom; } //*m_graphview->printScaleY(); } + +Layer *gGraph::getLineChart() +{ + gLineChart *lc; + + for (int i = 0; i < m_layers.size(); i++) { + lc = dynamic_cast(m_layers[i]); + + if (lc) { return lc; } + } + + return nullptr; +} + +void GetTextExtent(QString text, int &width, int &height, QFont *font) +{ +#ifdef ENABLE_THREADED_DRAWING + static QMutex mut; + mut.lock(); +#endif + QFontMetrics fm(*font); + //#ifdef Q_OS_WIN32 + QRect r = fm.tightBoundingRect(text); + width = r.width(); + height = r.height(); + //#else + // width=fm.width(text); + // height=fm.xHeight()+2; // doesn't work properly on windows.. + //#endif +#ifdef ENABLE_THREADED_DRAWING + mut.unlock(); +#endif +} + +int GetXHeight(QFont *font) +{ + QFontMetrics fm(*font); + return fm.xHeight(); +} diff --git a/sleepyhead/Graphs/gGraph.h b/sleepyhead/Graphs/gGraph.h new file mode 100644 index 00000000..50aff602 --- /dev/null +++ b/sleepyhead/Graphs/gGraph.h @@ -0,0 +1,371 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * Copyright (c) 2011-2014 Mark Watkins + * + * 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 Linux + * distribution for more details. */ + +#ifndef graphs_ggraph_h +#define graphs_ggraph_h + +#include +#include +#include +#include +#include +#include + +#include "Graphs/layer.h" + +class gGraphView; + +// Graph globals. +extern QFont *defaultfont; +extern QFont *mediumfont; +extern QFont *bigfont; +extern QHash images; +extern bool fbo_unsupported; +extern QGLFramebufferObject *fbo; + +bool InitGraphGlobals(); +void DestroyGraphGlobals(); + +const int max_fbo_width = 2048; +const int max_fbo_height = 2048; +const int mouse_movement_threshold = 6; + +/*! \class gGraph + \brief Single Graph object, containing multiple layers and Layer layout code + */ +class gGraph : public QObject +{ + Q_OBJECT; + public: + friend class gGraphView; + + /*! \brief Creates a new graph object + \param gGraphView * graphview if not null, links the graph to that object + \param QString title containing the graph Title which is rendered vertically + \param int height containing the opening height for this graph + \param short group containing which graph-link group this graph belongs to + */ + gGraph(gGraphView *graphview = nullptr, QString title = "", QString units = "", + int height = 100, short group = 0); + virtual ~gGraph(); + + //! \brief Tells all Layers to deselect any highlighting + void deselect(); + + //! \brief Returns true if any Layers have anything highlighted + bool isSelected(); + + //! \brief Starts the singleshot Timer running, for ms milliseconds + void Trigger(int ms); + + /*! \fn QPixmap renderPixmap(int width, int height, float fontscale=1.0); + \brief Returns a QPixmap containing a snapshot of the graph rendered at size widthxheight + \param int width Width of graph 'screenshot' + \param int height Height of graph 'screenshot' + \param float fontscale Scaling value to adjust DPI (when used for HighRes printing) + + Note if width or height is more than the OpenGL system allows, it could result in a crash + Keeping them under 2048 is a reasonably safe value. + */ + QPixmap renderPixmap(int width, int height, bool printing = false); + + //! \brief Set Graph visibility status + void setVisible(bool b) { m_visible = b; } + + //! \brief Return Graph visibility status + bool visible() { return m_visible; } + + //! \brief Return height element. This is used by the scaler in gGraphView. + float height() { return m_height; } + + //! \brief Set the height element. (relative to the total of all heights) + void setHeight(float height) { m_height = height; invalidate_yAxisImage = true; } + + int minHeight() { return m_min_height; } + void setMinHeight(int height) { m_min_height = height; } + + int maxHeight() { return m_max_height; } + void setMaxHeight(int height) { m_max_height = height; } + + //! \brief Returns true if the vertical graph title is shown + bool showTitle() { return m_showTitle; } + + //! \brief Set whether or not to render the vertical graph title + void setShowTitle(bool b) { m_showTitle = b; } + + //! \brief Returns printScaleX, used for DPI scaling in report printing + float printScaleX(); + + //! \brief Returns printScaleY, used for DPI scaling in report printing + float printScaleY(); + + //! \brief Returns true if none of the included layers have data attached + bool isEmpty(); + + //! \brief Add Layer l to graph object, allowing you to specify position, + // margin sizes, order, movability status and offsets + void AddLayer(Layer *l, LayerPosition position = LayerCenter, + short pixelsX = 0, short pixelsY = 0, short order = 0, + bool movable = false, short x = 0, short y = 0); + + void qglColor(QColor col); + + //! \brief Queues text for gGraphView object to draw it. + void renderText(QString text, int x, int y, float angle = 0.0, + QColor color = Qt::black, QFont *font = defaultfont, bool antialias = true); + + //! \brief Rounds Y scale values to make them look nice.. + // Applies the Graph Preference min/max settings. + void roundY(EventDataType &miny, EventDataType &maxy); + + //! \brief Process all Layers GLBuffer (Vertex) objects, drawing the actual OpenGL stuff. + void drawGLBuf(); + + //! \brief Returns the Graph's (vertical) title + QString title() { return m_title; } + + //! \brief Sets the Graph's (vertical) title + void setTitle(const QString title) { m_title = title; } + + //! \brief Returns the measurement Units the Y scale is referring to + QString units() { return m_units; } + + //! \brief Sets the measurement Units the Y scale is referring to + void setUnits(const QString units) { m_units = units; } + + //virtual void repaint(); // Repaint individual graph.. + + //! \brief Resets the graphs X & Y dimensions based on the Layer data + virtual void ResetBounds(); + + //! \brief Sets the time range selected for this graph (in milliseconds since 1970 epoch) + virtual void SetXBounds(qint64 minx, qint64 maxx); + + //! \brief Returns the physical Minimum time for all layers contained (in milliseconds since epoch) + virtual qint64 MinX(); + + //! \brief Returns the physical Maximum time for all layers contained (in milliseconds since epoch) + virtual qint64 MaxX(); + + //! \brief Returns the physical Minimum Y scale value for all layers contained + virtual EventDataType MinY(); + + //! \brief Returns the physical Maximum Y scale value for all layers contained + virtual EventDataType MaxY(); + + //! \brief Returns the physical Minimum Y scale value for all layers contained + virtual EventDataType physMinY(); + + //! \brief Returns the physical Maximum Y scale value for all layers contained + virtual EventDataType physMaxY(); + + //! \brief Sets the physical start of this graphs time range (in milliseconds since epoch) + virtual void SetMinX(qint64 v); + + //! \brief Sets the physical end of this graphs time range (in milliseconds since epoch) + virtual void SetMaxX(qint64 v); + + //! \brief Sets the physical Minimum Y scale value used while drawing this graph + virtual void SetMinY(EventDataType v); + + //! \brief Sets the physical Maximum Y scale value used while drawing this graph + virtual void SetMaxY(EventDataType v); + + //! \brief Forces Y Minimum to always select this value + virtual void setForceMinY(EventDataType v) { f_miny = v; m_enforceMinY = true; } + //! \brief Forces Y Maximum to always select this value + virtual void setForceMaxY(EventDataType v) { f_maxy = v; m_enforceMaxY = true; } + + + virtual EventDataType forceMinY() { return rec_miny; } + virtual EventDataType forceMaxY() { return rec_maxy; } + + //! \brief Set recommended Y minimum.. It won't go under this unless the data does. + // It won't go above this. + virtual void setRecMinY(EventDataType v) { rec_miny = v; } + //! \brief Set recommended Y minimum.. It won't go above this unless the data does. + // It won't go under this. + virtual void setRecMaxY(EventDataType v) { rec_maxy = v; } + + //! \brief Returns the recommended Y minimum.. It won't go under this unless the data does. + // It won't go above this. + virtual EventDataType RecMinY() { return rec_miny; } + //! \brief Returns the recommended Y maximum.. It won't go under this unless the data does. + // It won't go above this. + virtual EventDataType RecMaxY() { return rec_maxy; } + + //! \brief Called when main graph area is resized + void resize(int width, int height); // margin recalcs.. + + qint64 max_x, min_x, rmax_x, rmin_x; + EventDataType max_y, min_y, rmax_y, rmin_y, f_miny, f_maxy, rec_miny, rec_maxy; + EventDataType rphysmin_y, rphysmax_y; + + // not sure why there's two.. I can't remember + void setEnforceMinY(bool b) { m_enforceMinY = b; } + void setEnforceMaxY(bool b) { m_enforceMaxY = b; } + + //! \brief Returns whether this graph shows overall timescale, or a zoomed area + bool blockZoom() { return m_blockzoom; } + //! \brief Sets whether this graph shows an overall timescale, or a zoomed area. + void setBlockZoom(bool b) { m_blockzoom = b; } + + //! \brief Flips the GL coordinates from the graphs perspective.. Used in Scissor calculations + int flipY(int y); // flip GL coordinates + + //! \brief Returns the graph-linking group this Graph belongs in + short group() { return m_group; } + + //! \brief Sets the graph-linking group this Graph belongs in + void setGroup(short group) { m_group = group; } + + //! \brief Forces the main gGraphView object to draw all Text Components + void DrawTextQue(); + + //! \brief Sends supplied day object to all Graph layers so they can precalculate stuff + void setDay(Day *day); + + //! \brief Returns the current day object + Day *day() { return m_day; } + + //! \brief The Layer, layout and title drawing code + virtual void paint(int originX, int originY, int width, int height); + + //! \brief Gives the supplied data to the main ToolTip object for display + void ToolTip(QString text, int x, int y, int timeout = 0); + + //! \brief Public version of updateGL(), to redraw all graphs.. Not for normal use + void redraw(); + + //! \brief Asks the main gGraphView to redraw after ms milliseconds + void timedRedraw(int ms); + + //! \brief Sets the margins for the four sides of this graph. + void setMargins(short left, short right, short top, short bottom) { + m_marginleft = left; + m_marginright = right; + m_margintop = top; + m_marginbottom = bottom; + } + + //! \brief Returns this graphs left margin + short marginLeft(); + //! \brief Returns this graphs right margin + short marginRight(); + //! \brief Returns this graphs top margin + short marginTop(); + //! \brief Returns this graphs bottom margin + short marginBottom(); + + //! \brief Returns the main gGraphView objects gVertexBuffer line list. + gVertexBuffer *lines(); + //! \brief Returns the main gGraphView objects gVertexBuffer background line list. + gVertexBuffer *backlines(); + //! \brief Returns the main gGraphView objects gVertexBuffer front line list. + gVertexBuffer *frontlines(); + //! \brief Returns the main gGraphView objects gVertexBuffer quads list. + gVertexBuffer *quads(); + + const inline QRect &rect() const { return m_rect; } + + bool isPinned() { return m_pinned; } + void setPinned(bool b) { m_pinned = b; } + + // //! \brief Returns the main gGraphView objects gVertexBuffer stippled line list. + //GLShortBuffer * stippled(); + + //gVertexBuffer * vlines(); // testing new vertexbuffer + + short left, right, top, bottom; // dirty magin hacks.. + + Layer *getLineChart(); + QTimer *timer; + + // This gets set to true to force a redraw of the yAxis tickers when graphs are resized. + bool invalidate_yAxisImage; + bool invalidate_xAxisImage; + + //! \brief Returns a Vector reference containing all this graphs layers + QVector &layers() { return m_layers; } + + gGraphView *graphView() { return m_graphview; } + short m_marginleft, m_marginright, m_margintop, m_marginbottom; + + short zoomY() { return m_zoomY; } + void setZoomY(short zoom); + + static const short maxZoomY = 2; + + protected: + //! \brief Mouse Wheel events + virtual void wheelEvent(QWheelEvent *event); + + //! \brief Mouse Movement events + virtual void mouseMoveEvent(QMouseEvent *event); + + //! \brief Mouse Button Pressed events + virtual void mousePressEvent(QMouseEvent *event); + + //! \brief Mouse Button Released events + virtual void mouseReleaseEvent(QMouseEvent *event); + + //! \brief Mouse Button Double Clicked events + virtual void mouseDoubleClickEvent(QMouseEvent *event); + + //! \brief Key Pressed event + virtual void keyPressEvent(QKeyEvent *event); + + //! \brief Change the current selected time boundaries by mult, from origin position origin_px + void ZoomX(double mult, int origin_px); + + //! \brief The Main gGraphView object holding this graph + // (this can be pinched temporarily by print code) + gGraphView *m_graphview; + QString m_title; + QString m_units; + + //! \brief Vector containing all this graphs Layers + QVector m_layers; + float m_height, m_width; + + int m_min_height; + int m_max_height; + bool m_visible; + bool m_blockzoom; + QRect m_selection; + bool m_selecting_area; + QPoint m_current; + short m_group; + short m_lastx23; + Day *m_day; + gVertexBuffer *m_quad; + bool m_enforceMinY, m_enforceMaxY; + bool m_showTitle; + bool m_printing; + bool m_pinned; + short m_zoomY; + QRect m_rect; + + protected slots: + //! \brief Deselects any highlights, and schedules a main gGraphView redraw + void Timeout(); +}; + +/*! \brief Gets the width and height parameters for supplied text + \param QString text - The text string in question + \param int & width + \param int & height + \param QFont * font - The selected font used in the size calculations + */ +void GetTextExtent(QString text, int &width, int &height, QFont *font = defaultfont); + +//! \brief Return the height of the letter x for the selected font. +int GetXHeight(QFont *font = defaultfont); + +#endif // graphs_ggraph_h diff --git a/sleepyhead/Graphs/gGraphView.cpp b/sleepyhead/Graphs/gGraphView.cpp index eefcb606..f396bd6e 100644 --- a/sleepyhead/Graphs/gGraphView.cpp +++ b/sleepyhead/Graphs/gGraphView.cpp @@ -9,37 +9,33 @@ * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ -#include -#include -#include "gGraphView.h" -#include "SleepLib/profiles.h" -#include -#include +#include "Graphs/gGraphView.h" + #include -#include +#include #include +#include +#include #include +#include + #if QT_VERSION >= QT_VERSION_CHECK(5,0,0) # include #endif -#include "Graphs/gYAxis.h" -#include "Graphs/gFlagsLine.h" -#include "gLineChart.h" - -#ifdef Q_OS_MAC -# define USE_RENDERTEXT -# include "OpenGL/glu.h" -#else -# include "GL/glu.h" -#endif - #ifdef DEBUG_EFFICIENCY // Only works in 4.8 # include #endif +#include + #include "mainwindow.h" +#include "Graphs/glcommon.h" +#include "Graphs/gLineChart.h" +#include "Graphs/gYAxis.h" +#include "Graphs/gFlagsLine.h" +#include "SleepLib/profiles.h" extern MainWindow *mainwin; @@ -47,696 +43,7 @@ extern MainWindow *mainwin; int lines_drawn_this_frame = 0; int quads_drawn_this_frame = 0; -bool _graph_init = false; - -QFont *defaultfont = nullptr; -QFont *mediumfont = nullptr; -QFont *bigfont = nullptr; - -//MW: ick globals, but I want a system wide framebuffer of decent proprotions.. -bool fbo_unsupported = false; -QGLFramebufferObject *fbo = nullptr; -const int max_fbo_width = 2048; -const int max_fbo_height = 2048; - -bool evil_intel_graphics_chip = false; - extern QLabel *qstatus2; -const int mouse_movement_threshold = 6; - -QHash images; - -// Must be called from a thread inside the application. -void InitGraphs() -{ - if (_graph_init) { - return; - } - - if (!PREF.contains("Fonts_Graph_Name")) { - PREF["Fonts_Graph_Name"] = "Sans Serif"; - PREF["Fonts_Graph_Size"] = 10; - PREF["Fonts_Graph_Bold"] = false; - PREF["Fonts_Graph_Italic"] = false; - } - - if (!PREF.contains("Fonts_Title_Name")) { - PREF["Fonts_Title_Name"] = "Sans Serif"; - PREF["Fonts_Title_Size"] = 14; - PREF["Fonts_Title_Bold"] = true; - PREF["Fonts_Title_Italic"] = false; - } - - if (!PREF.contains("Fonts_Big_Name")) { - PREF["Fonts_Big_Name"] = "Serif"; - PREF["Fonts_Big_Size"] = 35; - PREF["Fonts_Big_Bold"] = false; - PREF["Fonts_Big_Italic"] = false; - } - - defaultfont = new QFont(PREF["Fonts_Graph_Name"].toString(), - PREF["Fonts_Graph_Size"].toInt(), - PREF["Fonts_Graph_Bold"].toBool() ? QFont::Bold : QFont::Normal, - PREF["Fonts_Graph_Italic"].toBool() - ); - mediumfont = new QFont(PREF["Fonts_Title_Name"].toString(), - PREF["Fonts_Title_Size"].toInt(), - PREF["Fonts_Title_Bold"].toBool() ? QFont::Bold : QFont::Normal, - PREF["Fonts_Title_Italic"].toBool() - ); - bigfont = new QFont(PREF["Fonts_Big_Name"].toString(), - PREF["Fonts_Big_Size"].toInt(), - PREF["Fonts_Big_Bold"].toBool() ? QFont::Bold : QFont::Normal, - PREF["Fonts_Big_Italic"].toBool() - ); - - defaultfont->setStyleHint(QFont::AnyStyle, QFont::OpenGLCompatible); - mediumfont->setStyleHint(QFont::AnyStyle, QFont::OpenGLCompatible); - bigfont->setStyleHint(QFont::AnyStyle, QFont::OpenGLCompatible); - - //images["mask"] = new QImage(":/icons/mask.png"); - images["oximeter"] = new QImage(":/icons/cubeoximeter.png"); - images["smiley"] = new QImage(":/icons/smileyface.png"); - //images["sad"] = new QImage(":/icons/sadface.png"); - - images["sheep"] = new QImage(":/icons/sheep.png"); - images["brick"] = new QImage(":/icons/brick.png"); - images["nographs"] = new QImage(":/icons/nographs.png"); - images["nodata"] = new QImage(":/icons/nodata.png"); - - _graph_init = true; -} - -void DoneGraphs() -{ - if (!_graph_init) { - return; - } - - delete defaultfont; - delete bigfont; - delete mediumfont; - - for (QHash::iterator i = images.begin(); i != images.end(); i++) { - delete i.value(); - } - - // Clear the frame buffer object. - if (fbo) { - if (fbo->isBound()) { - fbo->release(); - } - - delete fbo; - fbo = nullptr; - fbo_unsupported = true; // just in case shutdown order gets messed up - } - - _graph_init = false; -} - -void GetTextExtent(QString text, int &width, int &height, QFont *font) -{ -#ifdef ENABLE_THREADED_DRAWING - static QMutex mut; - mut.lock(); -#endif - QFontMetrics fm(*font); - //#ifdef Q_OS_WIN32 - QRect r = fm.tightBoundingRect(text); - width = r.width(); - height = r.height(); - //#else - // width=fm.width(text); - // height=fm.xHeight()+2; // doesn't work properly on windows.. - //#endif -#ifdef ENABLE_THREADED_DRAWING - mut.unlock(); -#endif -} - -int GetXHeight(QFont *font) -{ - QFontMetrics fm(*font); - return fm.xHeight(); -} - -inline quint32 swaporder(quint32 color) -{ - return ((color & 0xFF00FF00) | - ((color & 0xFF0000) >> 16) | - ((color & 0xFF) << 16)); -} - -void gVertexBuffer::setColor(QColor col) -{ - m_color = swaporder(col.rgba()); -} - -void gVertexBuffer::draw() -{ - bool antialias = m_forceantialias || (PROFILE.appearance->antiAliasing() && m_antialias); - - if (m_stippled) { antialias = false; } - - float size = m_size; - - if (antialias) { - glEnable(GL_BLEND); - glBlendFunc(m_blendfunc1, m_blendfunc2); - - if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { - glEnable(GL_LINE_SMOOTH); - glHint(GL_LINE_SMOOTH_HINT, GL_NICEST); - size += 0.5; - } else if (m_type == GL_POLYGON) { - glEnable(GL_POLYGON_SMOOTH); - glHint(GL_POLYGON_SMOOTH_HINT, GL_NICEST); - } - } - - if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { - if (m_stippled) { - glLineStipple(1, m_stipple); - //size=1; - glEnable(GL_LINE_STIPPLE); - } else { - //glLineStipple(1, 0xFFFF); - } - - glLineWidth(size); - - lines_drawn_this_frame += m_cnt / 2; - } else if (m_type == GL_POINTS) { - glPointSize(size); - } else if (m_type == GL_POLYGON) { - glPolygonMode(GL_BACK, GL_FILL); - lines_drawn_this_frame += m_cnt / 2; - } else if (m_type == GL_QUADS) { - quads_drawn_this_frame += m_cnt / 4; - } - - if (m_scissor) { - glScissor(s_x, s_y, s_width, s_height); - glEnable(GL_SCISSOR_TEST); - } - - glEnableClientState(GL_VERTEX_ARRAY); - glEnableClientState(GL_COLOR_ARRAY); - glVertexPointer(2, GL_SHORT, 8, (GLvoid *)buffer); - glColorPointer(4, GL_UNSIGNED_BYTE, 8, ((char *)buffer) + 4); - - glDrawArrays(m_type, 0, m_cnt); - - glDisableClientState(GL_COLOR_ARRAY); - glDisableClientState(GL_VERTEX_ARRAY); - m_cnt = 0; - - if (m_scissor) { - glDisable(GL_SCISSOR_TEST); - m_scissor = false; - } - - if (m_type == GL_POLYGON) { - glPolygonMode(GL_BACK, GL_FILL); - } - - if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { - if (m_stippled) { - glDisable(GL_LINE_STIPPLE); - glLineStipple(1, 0xFFFF); - } - } - - if (antialias) { - if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { - glDisable(GL_LINE_SMOOTH); - } else if (m_type == GL_POLYGON) { - glDisable(GL_POLYGON_SMOOTH); - } - - glDisable(GL_BLEND); - } -} -void gVertexBuffer::add(GLshort x1, GLshort y1, RGBA color) -{ - if (m_cnt < m_max) { - gVertex &v = buffer[m_cnt]; - - v.color = swaporder(color); - v.x = x1; - v.y = y1; - - m_cnt++; - } -} -void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, RGBA color) -{ - if (m_cnt < (m_max - 1)) { - gVertex *v = &buffer[m_cnt]; - - v->x = x1; - v->y = y1; - v->color = swaporder(color); - - v++; - v->x = x2; - v->y = y2; - v->color = swaporder(color); - - m_cnt += 2; - } -} -void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, GLshort y3, - GLshort x4, GLshort y4, RGBA color) -{ - if (m_cnt < (m_max - 3)) { - gVertex *v = &buffer[m_cnt]; - - v->color = swaporder(color); - v->x = x1; - v->y = y1; - v++; - - v->color = swaporder(color); - v->x = x2; - v->y = y2; - - v++; - v->color = swaporder(color); - v->x = x3; - v->y = y3; - - v++; - v->color = swaporder(color); - v->x = x4; - v->y = y4; - - m_cnt += 4; - } -} - -void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, GLshort y3, - GLshort x4, GLshort y4, RGBA color1, RGBA color2) -{ - if (m_cnt < (m_max - 3)) { - gVertex *v = &buffer[m_cnt]; - - v->color = swaporder(color1); - v->x = x1; - v->y = y1; - v++; - - v->color = swaporder(color1); - v->x = x2; - v->y = y2; - - v++; - v->color = swaporder(color2); - v->x = x3; - v->y = y3; - - v++; - v->color = swaporder(color2); - v->x = x4; - v->y = y4; - - m_cnt += 4; - } -} -void gVertexBuffer::unsafe_add(GLshort x1, GLshort y1) -{ - gVertex &v = buffer[m_cnt++]; - - v.color = m_color; - v.x = x1; - v.y = y1; -} -void gVertexBuffer::add(GLshort x1, GLshort y1) -{ - if (m_cnt < m_max) { - gVertex &v = buffer[m_cnt++]; - - v.color = m_color; - v.x = x1; - v.y = y1; - } -} -void gVertexBuffer::unsafe_add(GLshort x1, GLshort y1, GLshort x2, GLshort y2) -{ - gVertex *v = &buffer[m_cnt]; - - v->x = x1; - v->y = y1; - v->color = m_color; - - v++; - v->x = x2; - v->y = y2; - v->color = m_color; - - m_cnt += 2; -} - -void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2) -{ - if (m_cnt < (m_max - 1)) { - gVertex *v = &buffer[m_cnt]; - - v->x = x1; - v->y = y1; - v->color = m_color; - - v++; - v->x = x2; - v->y = y2; - v->color = m_color; - - m_cnt += 2; - } -} -void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, GLshort y3, - GLshort x4, GLshort y4) -{ - if (m_cnt < (m_max - 3)) { - gVertex *v = &buffer[m_cnt]; - - v->color = m_color; - v->x = x1; - v->y = y1; - v++; - - v->color = m_color; - v->x = x2; - v->y = y2; - - v++; - v->color = m_color; - v->x = x3; - v->y = y3; - - v++; - v->color = m_color; - v->x = x4; - v->y = y4; - - m_cnt += 4; - } -} -void gVertexBuffer::unsafe_add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, - GLshort y3, GLshort x4, GLshort y4) -{ - gVertex *v = &buffer[m_cnt]; - - v->color = m_color; - v->x = x1; - v->y = y1; - v++; - - v->color = m_color; - v->x = x2; - v->y = y2; - - v++; - v->color = m_color; - v->x = x3; - v->y = y3; - - v++; - v->color = m_color; - v->x = x4; - v->y = y4; - - m_cnt += 4; -} - -///////////////////////////////////////////////////////////////////// -// GLFloatBuffer - -GLFloatBuffer::GLFloatBuffer(int max, int type, bool stippled) - : GLBuffer(max, type, stippled) -{ - buffer = new GLfloat [max + 8]; - colors = new GLubyte[max * 4 + (8 * 4)]; -} -GLFloatBuffer::~GLFloatBuffer() -{ - if (colors) { delete [] colors; } - - if (buffer) { delete [] buffer; } -} - -void GLFloatBuffer::add(GLfloat x, GLfloat y, QColor &color) -{ - if (m_cnt < m_max + 2) { -#ifdef ENABLE_THREADED_DRAWING - mutex.lock(); -#endif - buffer[m_cnt++] = x; - buffer[m_cnt++] = y; - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); -#ifdef ENABLE_THREADED_DRAWING - mutex.unlock(); -#endif - } else { - qDebug() << "GLBuffer overflow"; - } -} -void GLFloatBuffer::add(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, QColor &color) -{ - if (m_cnt < m_max + 4) { - qDebug() << "GLFloatBuffer overflow"; - return; - } - -#ifdef ENABLE_THREADED_DRAWING - mutex.lock(); -#endif - buffer[m_cnt++] = x1; - buffer[m_cnt++] = y1; - buffer[m_cnt++] = x2; - buffer[m_cnt++] = y2; - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - -#ifdef ENABLE_THREADED_DRAWING - mutex.unlock(); -#endif -} -void GLFloatBuffer::add(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, GLfloat x3, GLfloat y3, - GLfloat x4, GLfloat y4, QColor &color) // add with vertex colors -{ - if (m_cnt >= m_max + 8) { - qDebug() << "GLFloatBuffer overflow"; - return; - } - -#ifdef ENABLE_THREADED_DRAWING - mutex.lock(); -#endif - buffer[m_cnt++] = x1; - buffer[m_cnt++] = y1; - buffer[m_cnt++] = x2; - buffer[m_cnt++] = y2; - buffer[m_cnt++] = x3; - buffer[m_cnt++] = y3; - buffer[m_cnt++] = x4; - buffer[m_cnt++] = y4; - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); -#ifdef ENABLE_THREADED_DRAWING - mutex.unlock(); -#endif -} -void GLFloatBuffer::quadGrLR(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, GLfloat x3, - GLfloat y3, GLfloat x4, GLfloat y4, QColor &color, QColor &color2) // add with vertex colors -{ - if (m_cnt >= m_max + 8) { - qDebug() << "GLFloatBuffer overflow"; - return; - } - -#ifdef ENABLE_THREADED_DRAWING - mutex.lock(); -#endif - buffer[m_cnt++] = x1; - buffer[m_cnt++] = y1; - buffer[m_cnt++] = x2; - buffer[m_cnt++] = y2; - buffer[m_cnt++] = x3; - buffer[m_cnt++] = y3; - buffer[m_cnt++] = x4; - buffer[m_cnt++] = y4; - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - - colors[m_colcnt++] = color2.red(); - colors[m_colcnt++] = color2.green(); - colors[m_colcnt++] = color2.blue(); - colors[m_colcnt++] = color2.alpha(); - colors[m_colcnt++] = color2.red(); - colors[m_colcnt++] = color2.green(); - colors[m_colcnt++] = color2.blue(); - colors[m_colcnt++] = color2.alpha(); -#ifdef ENABLE_THREADED_DRAWING - mutex.unlock(); -#endif -} -void GLFloatBuffer::quadGrTB(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, GLfloat x3, - GLfloat y3, GLfloat x4, GLfloat y4, QColor &color, QColor &color2) -{ - if (m_cnt >= m_max + 8) { - qDebug() << "GLFloatBuffer overflow"; - return; - } - -#ifdef ENABLE_THREADED_DRAWING - mutex.lock(); -#endif - buffer[m_cnt++] = x1; - buffer[m_cnt++] = y1; - buffer[m_cnt++] = x3; - buffer[m_cnt++] = y3; - buffer[m_cnt++] = x2; - buffer[m_cnt++] = y2; - buffer[m_cnt++] = x4; - buffer[m_cnt++] = y4; - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - colors[m_colcnt++] = color.red(); - colors[m_colcnt++] = color.green(); - colors[m_colcnt++] = color.blue(); - colors[m_colcnt++] = color.alpha(); - - colors[m_colcnt++] = color2.red(); - colors[m_colcnt++] = color2.green(); - colors[m_colcnt++] = color2.blue(); - colors[m_colcnt++] = color2.alpha(); - colors[m_colcnt++] = color2.red(); - colors[m_colcnt++] = color2.green(); - colors[m_colcnt++] = color2.blue(); - colors[m_colcnt++] = color2.alpha(); -#ifdef ENABLE_THREADED_DRAWING - mutex.unlock(); -#endif -} - -void GLFloatBuffer::draw() -{ - if (m_cnt <= 0) { return; } - - bool antialias = m_forceantialias || (PROFILE.appearance->antiAliasing() && m_antialias); - float size = m_size; - - if (antialias) { - glEnable(GL_BLEND); - glBlendFunc(m_blendfunc1, m_blendfunc2); - - if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { - glEnable(GL_LINE_SMOOTH); - glHint(GL_LINE_SMOOTH_HINT, GL_NICEST); - size += 0.5; - } else if (m_type == GL_POLYGON) { - glEnable(GL_POLYGON_SMOOTH); - glHint(GL_POLYGON_SMOOTH_HINT, GL_NICEST); - } - } - - if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { - glLineWidth(size); - - if (m_stippled) { - glLineStipple(1, 0xAAAA); - glEnable(GL_LINE_STIPPLE); - } - - lines_drawn_this_frame += m_cnt / 2; - } else if (m_type == GL_POINTS) { - glPointSize(size); - } else if (m_type == GL_POLYGON) { - glPolygonMode(GL_BACK, GL_FILL); - lines_drawn_this_frame += m_cnt / 2; - } else if (m_type == GL_QUADS) { - quads_drawn_this_frame += m_cnt / 4; - } - - if (m_scissor) { - glScissor(s1, s2, s3, s4); - glEnable(GL_SCISSOR_TEST); - } - - glVertexPointer(2, GL_FLOAT, 0, buffer); - - glColorPointer(4, GL_UNSIGNED_BYTE, 0, colors); - - //glColor4ub(200,128,95,200); - glEnableClientState(GL_VERTEX_ARRAY); - glEnableClientState(GL_COLOR_ARRAY); - - - glDrawArrays(m_type, 0, m_cnt >> 1); - glDisableClientState(GL_COLOR_ARRAY); - glDisableClientState(GL_VERTEX_ARRAY); - - //qDebug() << "I Drawed" << m_cnt << "vertices"; - m_cnt = 0; - m_colcnt = 0; - - if (m_scissor) { - glDisable(GL_SCISSOR_TEST); - m_scissor = false; - } - - if (m_type == GL_POLYGON) { - glPolygonMode(GL_BACK, GL_FILL); - } - - if (antialias) { - if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { - if (m_stippled) { glDisable(GL_LINE_STIPPLE); } - - glDisable(GL_LINE_SMOOTH); - } else if (m_type == GL_POLYGON) { - glDisable(GL_POLYGON_SMOOTH); - } - - glDisable(GL_BLEND); - } -} - -////////////////////////////////////////////////////////////////////////////// -// gToolTip implementation gToolTip::gToolTip(gGraphView *graphview) : m_graphview(graphview) @@ -855,264 +162,6 @@ void gToolTip::timerDone() m_graphview->redraw(); } -Layer::~Layer() -{ - for (int i = 0; i < mgl_buffers.size(); i++) { - delete mgl_buffers[i]; - } - - for (int i = 0; i < mv_buffers.size(); i++) { - delete mv_buffers[i]; - } -} -void Layer::drawGLBuf(float linesize) -{ - int type; - float size; - - if (!m_visible) { return; } - - GLBuffer *buf; - gVertexBuffer *vb; - - for (int i = 0; i < mv_buffers.size(); i++) { - vb = mv_buffers[i]; - size = vb->size(); - type = vb->type(); - - if ((linesize > size) && ((type == GL_LINES) || (type == GL_LINE_LOOP))) { - vb->setSize(linesize); - } - - vb->draw(); - vb->setSize(size); - } - - for (int i = 0; i < mgl_buffers.size(); i++) { - buf = mgl_buffers[i]; - size = buf->size(); - type = buf->type(); - - if ((linesize > size) && ((type == GL_LINES) || (type == GL_LINE_LOOP))) { - buf->setSize(linesize); - } - - buf->draw(); - //if ((linesize>size) && ((type==GL_LINES) || (type==GL_LINE_LOOP))) { - buf->setSize(size); - //} - } -} - -void Layer::SetDay(Day *d) -{ - if (d) { - m_day = d; - m_minx = d->first(m_code); - m_maxx = d->last(m_code); - m_miny = d->Min(m_code); - m_maxy = d->Max(m_code); - } else { m_day = nullptr; } - -} - -bool Layer::isEmpty() -{ - //if (m_day && (m_day->count(m_code)>0)) - if (m_day && (m_day->channelExists(m_code))) { - return false; - } - - return true; -} -void Layer::setLayout(LayerPosition position, short width, short height, short order) -{ - m_position = position; - m_width = width; - m_height = height; - m_order = order; -} - -LayerGroup::~LayerGroup() -{ - for (int i = 0; i < layers.size(); i++) { - delete layers[i]; - } -} -bool LayerGroup::isEmpty() -{ - if (!m_day) { - return true; - } - - bool empty = true; - - for (int i = 0; i < layers.size(); i++) { - if (layers[i]->isEmpty()) { - empty = false; - break; - } - } - - return empty; -} -void LayerGroup::drawGLBuf(float linesize) -{ - Layer::drawGLBuf(linesize); - - for (int i = 0; i < layers.size(); i++) { - layers[i]->drawGLBuf(linesize); - } -} - -void LayerGroup::SetDay(Day *d) -{ - m_day = d; - - for (int i = 0; i < layers.size(); i++) { - layers[i]->SetDay(d); - } -} - -void LayerGroup::AddLayer(Layer *l) -{ - layers.push_back(l); - l->addref(); -} - -qint64 LayerGroup::Minx() -{ - bool first = true; - qint64 m = 0, t; - - for (int i = 0; i < layers.size(); i++) { - t = layers[i]->Minx(); - - if (!t) { continue; } - - if (first) { - m = t; - first = false; - } else if (m > t) { m = t; } - } - - return m; -} -qint64 LayerGroup::Maxx() -{ - bool first = true; - qint64 m = 0, t; - - for (int i = 0; i < layers.size(); i++) { - t = layers[i]->Maxx(); - - if (!t) { continue; } - - if (first) { - m = t; - first = false; - } else if (m < t) { m = t; } - } - - return m; -} -EventDataType LayerGroup::Miny() -{ - bool first = true; - EventDataType m = 0, t; - - for (int i = 0; i < layers.size(); i++) { - t = layers[i]->Miny(); - - if (t == layers[i]->Maxy()) { continue; } - - if (first) { - m = t; - first = false; - } else { - if (m > t) { m = t; } - } - } - - return m; -} -EventDataType LayerGroup::Maxy() -{ - bool first = true; - EventDataType m = 0, t; - - for (int i = 0; i < layers.size(); i++) { - t = layers[i]->Maxy(); - - if (t == layers[i]->Miny()) { continue; } - - if (first) { - m = t; - first = false; - } else if (m < t) { m = t; } - } - - return m; -} - -//! \brief Mouse wheel moved somewhere over this layer -bool LayerGroup::wheelEvent(QWheelEvent *event, gGraph *graph) -{ - for (int i = 0; i < layers.size(); i++) - if (layers[i]->wheelEvent(event, graph)) { - return true; - } - - return false; -} - -//! \brief Mouse moved somewhere over this layer -bool LayerGroup::mouseMoveEvent(QMouseEvent *event, gGraph *graph) -{ - for (int i = 0; i < layers.size(); i++) - if (layers[i]->mouseMoveEvent(event, graph)) { return true; } - - return false; -} - -//! \brief Mouse left or right button pressed somewhere on this layer -bool LayerGroup::mousePressEvent(QMouseEvent *event, gGraph *graph) -{ - for (int i = 0; i < layers.size(); i++) - if (layers[i]->mousePressEvent(event, graph)) { return true; } - - return false; -} - -//! \brief Mouse button released that was originally pressed somewhere on this layer -bool LayerGroup::mouseReleaseEvent(QMouseEvent *event, gGraph *graph) -{ - for (int i = 0; i < layers.size(); i++) - if (layers[i]->mouseReleaseEvent(event, graph)) { return true; } - - return false; -} - -//! \brief Mouse button double clicked somewhere on this layer -bool LayerGroup::mouseDoubleClickEvent(QMouseEvent *event, gGraph *graph) -{ - for (int i = 0; i < layers.size(); i++) - if (layers[i]->mouseDoubleClickEvent(event, graph)) { return true; } - - return false; -} - -//! \brief A key was pressed on the keyboard while the graph area was focused. -bool LayerGroup::keyPressEvent(QKeyEvent *event, gGraph *graph) -{ - for (int i = 0; i < layers.size(); i++) - if (layers[i]->keyPressEvent(event, graph)) { return true; } - - return false; -} - -const double zoom_hard_limit = 500.0; - #ifdef ENABLE_THREADED_DRAWING gThread::gThread(gGraphView *g) @@ -1157,348 +206,8 @@ void gThread::run() } } -#endif -gGraph::gGraph(gGraphView *graphview, QString title, QString units, int height, short group) : - m_graphview(graphview), - m_title(title), - m_units(units), - m_height(height), - m_visible(true) -{ - m_min_height = 60; - m_width = 0; +#endif // ENABLE_THREADED_DRAWING - m_layers.clear(); - - f_miny = f_maxy = 0; - rmin_x = rmin_y = 0; - rmax_x = rmax_y = 0; - max_x = max_y = 0; - min_x = min_y = 0; - rec_miny = rec_maxy = 0; - rphysmax_y = rphysmin_y = 0; - m_zoomY = 0; - - if (graphview) { - graphview->addGraph(this, group); - timer = new QTimer(graphview); - connect(timer, SIGNAL(timeout()), SLOT(Timeout())); - } else { - qWarning() << "gGraph created without a gGraphView container.. Naughty programmer!! Bad!!!"; - } - - m_margintop = 14; - m_marginbottom = 5; - m_marginleft = 0; - m_marginright = 15; - m_selecting_area = m_blockzoom = false; - m_pinned = false; - m_lastx23 = 0; - - invalidate_yAxisImage = true; - invalidate_xAxisImage = true; - - m_quad = new gVertexBuffer(64, GL_QUADS); - m_quad->forceAntiAlias(true); - m_enforceMinY = m_enforceMaxY = false; - m_showTitle = true; - m_printing = false; -} -gGraph::~gGraph() -{ - for (int i = 0; i < m_layers.size(); i++) { - if (m_layers[i]->unref()) { - delete m_layers[i]; - } - } - - m_layers.clear(); - delete m_quad; - - timer->stop(); - disconnect(timer, 0, 0, 0); - delete timer; -} -void gGraph::Trigger(int ms) -{ - if (timer->isActive()) { timer->stop(); } - - timer->setSingleShot(true); - timer->start(ms); -} -void gGraph::Timeout() -{ - deselect(); - m_graphview->timedRedraw(0); -} - -void gGraph::deselect() -{ - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - (*l)->deselect(); - } -} -bool gGraph::isSelected() -{ - bool res = false; - - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - res = (*l)->isSelected(); - - if (res) { break; } - } - - return res; -} - -bool gGraph::isEmpty() -{ - bool empty = true; - - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - if (!(*l)->isEmpty()) { - empty = false; - break; - } - } - - return empty; -} - - -float gGraph::printScaleX() { return m_graphview->printScaleX(); } -float gGraph::printScaleY() { return m_graphview->printScaleY(); } - - -void gGraph::drawGLBuf() -{ - - float linesize = 1; - - if (m_printing) { linesize = 4; } //ceil(m_graphview->printScaleY()); - - for (int i = 0; i < m_layers.size(); i++) { - m_layers[i]->drawGLBuf(linesize); - } - - m_quad->draw(); -} -void gGraph::setDay(Day *day) -{ - m_day = day; - - for (int i = 0; i < m_layers.size(); i++) { - m_layers[i]->SetDay(day); - } - - ResetBounds(); -} - -void gGraph::setZoomY(short zoom) -{ - m_zoomY = zoom; - redraw(); -} - - -void gGraph::qglColor(QColor col) -{ - m_graphview->qglColor(col); -} - -void gGraph::renderText(QString text, int x, int y, float angle, QColor color, QFont *font, - bool antialias) -{ - m_graphview->AddTextQue(text, x, y, angle, color, font, antialias); -} - -void gGraph::paint(int originX, int originY, int width, int height) -{ - m_rect = QRect(originX, originY, width, height); - - int fw, font_height; - GetTextExtent("Wg@", fw, font_height); - - if (m_margintop > 0) { m_margintop = font_height + (8 * printScaleY()); } - - //m_marginbottom=5; - - //glColor4f(0,0,0,1); - left = marginLeft(), right = marginRight(), top = marginTop(), bottom = marginBottom(); - int x = 0, y = 0; - - if (m_showTitle) { - int title_x, yh; - /* if (titleImage.isNull()) { - // Render the title to a texture so we don't have to draw the vertical text every time.. - - GetTextExtent("Wy@",x,yh,mediumfont); // This gets a better consistent height. should be cached. - - GetTextExtent(title(),x,y,mediumfont); - - y=yh; - QPixmap tpm=QPixmap(x+4,y+4); - - tpm.fill(Qt::transparent); //empty it - QPainter pmp(&tpm); - - pmp.setRenderHint(QPainter::TextAntialiasing, true); - - QBrush brush2(Qt::black); // text color - pmp.setBrush(brush2); - - pmp.setFont(*mediumfont); - - pmp.drawText(2,y,title()); // draw from the bottom - pmp.end(); - - // convert to QImage and bind to a texture for future use - titleImage=QGLWidget::convertToGLFormat(tpm.toImage().mirrored(false,true)); - titleImageTex=m_graphview->bindTexture(titleImage); - } - y=titleImage.height(); - x=titleImage.width(); //vertical text - - glEnable(GL_BLEND); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - - title_x=y*2; - - glEnable(GL_TEXTURE_2D); - - // Rotate and draw vertical texture containing graph titles - glPushMatrix(); - glTranslatef(marginLeft()+4,originY+height/2+x/2, 0); - glRotatef(-90,0,0,1); - m_graphview->drawTexture(QPoint(0,y/2),titleImageTex); - glPopMatrix(); - - glDisable(GL_TEXTURE_2D); - glDisable(GL_BLEND); - - // All that to replace this little, but -hideously- slow line of text.. - */ - - QFontMetrics fm(*mediumfont); - - yh = fm.height(); - //GetTextExtent("Wy@",x,yh,mediumfont); // This gets a better consistent height. should be cached. - y = yh; - x = fm.width(title()); - //GetTextExtent(title(),x,y,mediumfont); - title_x = yh * 2; - - - renderText(title(), marginLeft() + title_x + 4, originY + height / 2 - y / 2, 90, Qt::black, - mediumfont); - left += title_x; - } else { left = 0; } - - //#define DEBUG_LAYOUT -#ifdef DEBUG_LAYOUT - QColor col = Qt::red; - lines()->add(0, originY, 0, originY + height, col); - lines()->add(left, originY, left, originY + height, col); -#endif - int tmp; - - left = 0; - - for (int i = 0; i < m_layers.size(); i++) { - Layer *ll = m_layers[i]; - - if (!ll->visible()) { continue; } - - tmp = ll->Height() * m_graphview->printScaleY(); - - if (ll->position() == LayerTop) { top += tmp; } - - if (ll->position() == LayerBottom) { bottom += tmp; } - } - - - for (int i = 0; i < m_layers.size(); i++) { - Layer *ll = m_layers[i]; - - if (!ll->visible()) { continue; } - - tmp = ll->Width() * m_graphview->printScaleX(); - - if (ll->position() == LayerLeft) { - ll->m_rect = QRect(originX + left, originY + top, tmp, height - top - bottom); - ll->paint(*this, originX + left, originY + top, tmp, height - top - bottom); - left += tmp; -#ifdef DEBUG_LAYOUT - lines()->add(originX + left - 1, originY, originX + left - 1, originY + height, col); -#endif - } - - if (ll->position() == LayerRight) { - right += tmp; - ll->m_rect = QRect(originX + width - right, originY + top, tmp, height - top - bottom); - ll->paint(*this, originX + width - right, originY + top, tmp, height - top - bottom); -#ifdef DEBUG_LAYOUT - lines()->add(originX + width - right, originY, originX + width - right, originY + height, col); -#endif - } - } - - - bottom = marginBottom(); - top = marginTop(); - - for (int i = 0; i < m_layers.size(); i++) { - Layer *ll = m_layers[i]; - - if (!ll->visible()) { continue; } - - tmp = ll->Height() * m_graphview->printScaleY(); - - if (ll->position() == LayerTop) { - ll->m_rect = QRect(originX + left, originY + top, width - left - right, tmp); - ll->paint(*this, originX + left, originY + top, width - left - right, tmp); - top += tmp; - } - - if (ll->position() == LayerBottom) { - bottom += tmp; - ll->m_rect = QRect(originX + left, originY + height - bottom, width - left - right, tmp); - ll->paint(*this, originX + left, originY + height - bottom, width - left - right, tmp); - } - } - - if (isPinned()) { - // Fill the background on pinned graphs - - // m_graphview->quads->add(originX+left,originY+top, originX+width-right,originY+top, originX+width-right,originY+height-bottom, originX+left,originY+height-bottom, 0xffffffff); - glBegin(GL_QUADS); - glColor4f(1.0, 1.0, 1.0, 1.0); // Gradient End - glVertex2i(originX + left, originY + top); - glVertex2i(originX + width - right, originY + top); - glVertex2i(originX + width - right, originY + height - bottom); - glVertex2i(originX + left, originY + height - bottom); - glEnd(); - } - - for (int i = 0; i < m_layers.size(); i++) { - Layer *ll = m_layers[i]; - - if (!ll->visible()) { continue; } - - if (ll->position() == LayerCenter) { - ll->m_rect = QRect(originX + left, originY + top, width - left - right, height - top - bottom); - ll->paint(*this, originX + left, originY + top, width - left - right, height - top - bottom); - } - } - - if (m_selection.width() > 0 && m_selecting_area) { - QColor col(128, 128, 255, 128); - quads()->add(originX + m_selection.x(), originY + top, - originX + m_selection.x() + m_selection.width(), originY + top, col.rgba()); - quads()->add(originX + m_selection.x() + m_selection.width(), originY + height - bottom, - originX + m_selection.x(), originY + height - bottom, col.rgba()); - } -} void gGraphView::queGraph(gGraph *g, int left, int top, int width, int height) { g->m_rect = QRect(left, top, width, height); @@ -1536,700 +245,6 @@ gGraph *gGraphView::popGraph() return g; } -void gGraph::AddLayer(Layer *l, LayerPosition position, short width, short height, short order, - bool movable, short x, short y) -{ - l->setLayout(position, width, height, order); - l->setMovable(movable); - l->setPos(x, y); - l->addref(); - m_layers.push_back(l); -} -void gGraph::redraw() -{ - m_graphview->redraw(); -} -void gGraph::timedRedraw(int ms) -{ - m_graphview->timedRedraw(ms); -} - -void gGraph::mouseMoveEvent(QMouseEvent *event) -{ - // qDebug() << m_title << "Move" << event->pos() << m_graphview->pointClicked(); - int y = event->y(); - int x = event->x(); - - bool doredraw = false; - - for (int i = 0; i < m_layers.size(); i++) { - if (m_layers[i]->m_rect.contains(x, y)) - if (m_layers[i]->mouseMoveEvent(event, this)) { - doredraw = true; - } - } - - y -= m_rect.top(); - x -= m_rect.left(); - - int x2 = m_graphview->pointClicked().x() - m_rect.left(); - - int w = m_rect.width() - (left + right); - //int h=m_lastbounds.height()-(bottom+m_marginbottom); - double xx = max_x - min_x; - double xmult = xx / w; - - if (m_graphview->m_selected_graph == this) { // Left Mouse button dragging - if (event->buttons() & Qt::LeftButton) { - //qDebug() << m_title << "Moved" << x << y << left << right << top << bottom << m_width << h; - int a1 = MIN(x, x2); - int a2 = MAX(x, x2); - - if (a1 < left) { a1 = left; } - - if (a2 > left + w) { a2 = left + w; } - - m_selecting_area = true; - m_selection = QRect(a1 - 1, 0, a2 - a1, m_rect.height()); - double w2 = m_rect.width() - right - left; - - if (m_blockzoom) { - xmult = (rmax_x - rmin_x) / w2; - } else { - xmult = (max_x - min_x) / w2; - } - - qint64 a = double(a2 - a1) * xmult; - float d = double(a) / 86400000.0; - int h = a / 3600000; - int m = (a / 60000) % 60; - int s = (a / 1000) % 60; - int ms(a % 1000); - QString str; - - if (d > 1) { - str.sprintf("%1.0f days", d); - } else { - - str.sprintf("%02i:%02i:%02i:%03i", h, m, s, ms); - } - - if (qstatus2) { - qstatus2->setText(str); - } - - doredraw = true; - } else if (event->buttons() & Qt::RightButton) { // Right Mouse button dragging - m_graphview->setPointClicked(event->pos()); - x -= left; - x2 -= left; - - if (!m_blockzoom) { - xx = max_x - min_x; - xmult = xx / double(w); - qint64 j1 = xmult * x; - qint64 j2 = xmult * x2; - qint64 jj = j2 - j1; - min_x += jj; - max_x += jj; - - if (min_x < rmin_x) { - min_x = rmin_x; - max_x = rmin_x + xx; - } - - if (max_x > rmax_x) { - max_x = rmax_x; - min_x = rmax_x - xx; - } - - m_graphview->SetXBounds(min_x, max_x, m_group, false); - doredraw = true; - } else { - qint64 qq = rmax_x - rmin_x; - xx = max_x - min_x; - - if (xx == qq) { xx = 1800000; } - - xmult = qq / double(w); - qint64 j1 = (xmult * x); - min_x = rmin_x + j1 - (xx / 2); - max_x = min_x + xx; - - if (min_x < rmin_x) { - min_x = rmin_x; - max_x = rmin_x + xx; - } - - if (max_x > rmax_x) { - max_x = rmax_x; - min_x = rmax_x - xx; - } - - m_graphview->SetXBounds(min_x, max_x, m_group, false); - doredraw = true; - } - } - } - - //if (!nolayer) { // no mouse button - if (doredraw) { - m_graphview->redraw(); - } - - //} - //if (x>left+m_marginleft && xtop+m_margintop && ypos().y(); - int x = event->pos().x(); - - for (int i = 0; i < m_layers.size(); i++) { - if (m_layers[i]->m_rect.contains(x, y)) - if (m_layers[i]->mousePressEvent(event, this)) { - return; - } - } - - /* - int w=m_lastbounds.width()-(right+m_marginright); - //int h=m_lastbounds.height()-(bottom+m_marginbottom); - //int x2,y2; - double xx=max_x-min_x; - //double xmult=xx/w; - if (x>left+m_marginleft && xtop+m_margintop && ypos().y(); - int x = event->pos().x(); - - for (int i = 0; i < m_layers.size(); i++) { - if (m_layers[i]->m_rect.contains(x, y)) - if (m_layers[i]->mouseReleaseEvent(event, this)) { - return; - } - } - - x -= m_rect.left(); - y -= m_rect.top(); - - - int w = m_rect.width() - left - right; //(m_marginleft+left+right+m_marginright); - int h = m_rect.height() - bottom; //+m_marginbottom); - - int x2 = m_graphview->pointClicked().x() - m_rect.left(); - int y2 = m_graphview->pointClicked().y() - m_rect.top(); - - - //qDebug() << m_title << "Released" << min_x << max_x << x << y << x2 << y2 << left << right << top << bottom << m_width << m_height; - if (m_selecting_area) { - m_selecting_area = false; - m_selection.setWidth(0); - - if (m_graphview->horizTravel() > mouse_movement_threshold) { - x -= left; //+m_marginleft; - y -= top; //+m_margintop; - x2 -= left; //+m_marginleft; - y2 -= top; //+m_margintop; - - if (x < 0) { x = 0; } - - if (x2 < 0) { x2 = 0; } - - if (x > w) { x = w; } - - if (x2 > w) { x2 = w; } - - double xx; - double xmult; - - if (!m_blockzoom) { - xx = max_x - min_x; - xmult = xx / double(w); - qint64 j1 = min_x + xmult * x; - qint64 j2 = min_x + xmult * x2; - qint64 a1 = MIN(j1, j2) - qint64 a2 = MAX(j1, j2) - - //if (a1 rmax_x) { a2 = rmax_x; } - - if (a1 <= rmin_x && a2 <= rmin_x) { - //qDebug() << "Foo??"; - } else { - if (a2 - a1 < zoom_hard_limit) { a2 = a1 + zoom_hard_limit; } - - m_graphview->SetXBounds(a1, a2, m_group); - } - } else { - xx = rmax_x - rmin_x; - xmult = xx / double(w); - qint64 j1 = rmin_x + xmult * x; - qint64 j2 = rmin_x + xmult * x2; - qint64 a1 = MIN(j1, j2) - qint64 a2 = MAX(j1, j2) - - //if (a1 rmax_x) { a2 = rmax_x; } - - if (a1 <= rmin_x && a2 <= rmin_x) { - qDebug() << "Foo2??"; - } else { - if (a2 - a1 < zoom_hard_limit) { a2 = a1 + zoom_hard_limit; } - - m_graphview->SetXBounds(a1, a2, m_group); - } - } - - return; - } else { m_graphview->redraw(); } - } - - if ((m_graphview->horizTravel() < mouse_movement_threshold) && (x > left && x < w + left - && y > top && y < h)) { - // normal click in main area - if (!m_blockzoom) { - double zoom; - - if (event->button() & Qt::RightButton) { - zoom = 1.33; - - if (event->modifiers() & Qt::ControlModifier) { zoom *= 1.5; } - - ZoomX(zoom, x); // Zoom out - return; - } else if (event->button() & Qt::LeftButton) { - zoom = 0.75; - - if (event->modifiers() & Qt::ControlModifier) { zoom /= 1.5; } - - ZoomX(zoom, x); // zoom in. - return; - } - } else { - x -= left; - y -= top; - //w-=m_marginleft+left; - double qq = rmax_x - rmin_x; - double xmult; - - double xx = max_x - min_x; - //if (xx==qq) xx=1800000; - - xmult = qq / double(w); - - if ((xx == qq) || (x == m_lastx23)) { - double zoom = 1; - - if (event->button() & Qt::RightButton) { - zoom = 1.33; - - if (event->modifiers() & Qt::ControlModifier) { zoom *= 1.5; } - } else if (event->button() & Qt::LeftButton) { - zoom = 0.75; - - if (event->modifiers() & Qt::ControlModifier) { zoom /= 1.5; } - } - - xx *= zoom; - - if (xx < qq / zoom_hard_limit) { xx = qq / zoom_hard_limit; } - - if (xx > qq) { xx = qq; } - } - - double j1 = xmult * x; - min_x = rmin_x + j1 - (xx / 2.0); - max_x = min_x + xx; - - if (min_x < rmin_x) { - min_x = rmin_x; - max_x = rmin_x + xx; - } else if (max_x > rmax_x) { - max_x = rmax_x; - min_x = rmax_x - xx; - } - - m_graphview->SetXBounds(min_x, max_x, m_group); - m_lastx23 = x; - } - } - - //m_graphview->redraw(); -} - - -void gGraph::wheelEvent(QWheelEvent *event) -{ - //qDebug() << m_title << "Wheel" << event->x() << event->y() << event->delta(); - //int y=event->pos().y(); - if (event->orientation() == Qt::Horizontal) { - - return; - } - - int x = event->pos().x() - m_graphview->titleWidth; //(left+m_marginleft); - - if (event->delta() > 0) { - ZoomX(0.75, x); - } else { - ZoomX(1.5, x); - } - - int y = event->pos().y(); - x = event->pos().x(); - - for (int i = 0; i < m_layers.size(); i++) { - if (m_layers[i]->m_rect.contains(x, y)) { - m_layers[i]->wheelEvent(event, this); - } - } - -} -void gGraph::mouseDoubleClickEvent(QMouseEvent *event) -{ - //mousePressEvent(event); - //mouseReleaseEvent(event); - int y = event->pos().y(); - int x = event->pos().x(); - - for (int i = 0; i < m_layers.size(); i++) { - if (m_layers[i]->m_rect.contains(x, y)) { - m_layers[i]->mouseDoubleClickEvent(event, this); - } - } - - //int w=m_lastbounds.width()-(m_marginleft+left+right+m_marginright); - //int h=m_lastbounds.height()-(bottom+m_marginbottom); - //int x2=m_graphview->pointClicked().x(),y2=m_graphview->pointClicked().y(); - // if ((m_graphview->horizTravel()left+m_marginleft && xtop+m_margintop && ybutton() & Qt::RightButton) { - // ZoomX(1.66,x); // Zoon out - // return; - // } else if (event->button() & Qt::LeftButton) { - // ZoomX(0.75/2.0,x); // zoom in. - // return; - // } - // } else { - // Propagate the events to graph Layers - // } - //mousePressEvent(event); - //mouseReleaseEvent(event); - //qDebug() << m_title << "Double Clicked" << event->x() << event->y(); -} -void gGraph::keyPressEvent(QKeyEvent *event) -{ - for (QVector::iterator i = m_layers.begin(); i != m_layers.end(); i++) { - (*i)->keyPressEvent(event, this); - } - - //qDebug() << m_title << "Key Pressed.. implement me" << event->key(); -} - -void gGraph::ZoomX(double mult, int origin_px) -{ - - int width = m_rect.width() - left - right; //(m_marginleft+left+right+m_marginright); - - if (origin_px == 0) { origin_px = (width / 2); } - else { origin_px -= left; } - - if (origin_px < 0) { origin_px = 0; } - - if (origin_px > width) { origin_px = width; } - - - // Okay, I want it to zoom in centered on the mouse click area.. - // Find X graph position of mouse click - // find current zoom width - // apply zoom - // center on point found in step 1. - - qint64 min = min_x; - qint64 max = max_x; - - double hardspan = rmax_x - rmin_x; - double span = max - min; - double ww = double(origin_px) / double(width); - double origin = ww * span; - //double center=0.5*span; - //double dist=(origin-center); - - double q = span * mult; - - 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 < rmin_x) { - min = rmin_x; - max = min + q; - } - - if (max > rmax_x) { - max = rmax_x; - min = max - q; - } - - m_graphview->SetXBounds(min, max, m_group); - //updateSelectionTime(max-min); -} - -void gGraph::DrawTextQue() -{ - m_graphview->DrawTextQue(); -} - -// margin recalcs.. -void gGraph::resize(int width, int height) -{ - invalidate_xAxisImage = true; - invalidate_yAxisImage = true; - - Q_UNUSED(width); - Q_UNUSED(height); - //m_height=height; - //m_width=width; -} - -qint64 gGraph::MinX() -{ - qint64 val = 0, tmp; - - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - if ((*l)->isEmpty()) { - continue; - } - - tmp = (*l)->Minx(); - - if (!tmp) { - continue; - } - - if (!val || tmp < val) { - val = tmp; - } - } - - if (val) { rmin_x = val; } - - return val; -} -qint64 gGraph::MaxX() -{ - //bool first=true; - qint64 val = 0, tmp; - - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - if ((*l)->isEmpty()) { - continue; - } - - tmp = (*l)->Maxx(); - - //if (!tmp) continue; - if (!val || tmp > val) { - val = tmp; - } - } - - if (val) { rmax_x = val; } - - return val; -} - -EventDataType gGraph::MinY() -{ - bool first = true; - EventDataType val = 0, tmp; - - if (m_enforceMinY) { - return rmin_y = f_miny; - } - - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - if ((*l)->isEmpty()) { - continue; - } - - tmp = (*l)->Miny(); - - if (tmp == 0 && tmp == (*l)->Maxy()) { - continue; - } - - if (first) { - val = tmp; - first = false; - } else { - if (tmp < val) { - val = tmp; - } - } - } - - return rmin_y = val; -} -EventDataType gGraph::MaxY() -{ - bool first = true; - EventDataType val = 0, tmp; - - if (m_enforceMaxY) { - return rmax_y = f_maxy; - } - - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - if ((*l)->isEmpty()) { - continue; - } - - tmp = (*l)->Maxy(); - - if (tmp == 0 && tmp == (*l)->Miny()) { - continue; - } - - if (first) { - val = tmp; - first = false; - } else { - if (tmp > val) { - val = tmp; - } - } - } - - return rmax_y = val; -} - -EventDataType gGraph::physMinY() -{ - bool first = true; - EventDataType val = 0, tmp; - - //if (m_enforceMinY) return rmin_y=f_miny; - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - if ((*l)->isEmpty()) { - continue; - } - - tmp = (*l)->physMiny(); - - if (tmp == 0 && tmp == (*l)->physMaxy()) { - continue; - } - - if (first) { - val = tmp; - first = false; - } else { - if (tmp < val) { - val = tmp; - } - } - } - - return rphysmin_y = val; -} -EventDataType gGraph::physMaxY() -{ - bool first = true; - EventDataType val = 0, tmp; - - // if (m_enforceMaxY) return rmax_y=f_maxy; - for (QVector::iterator l = m_layers.begin(); l != m_layers.end(); l++) { - if ((*l)->isEmpty()) { - continue; - } - - tmp = (*l)->physMaxy(); - - if (tmp == 0 && tmp == (*l)->physMiny()) { - continue; - } - - if (first) { - val = tmp; - first = false; - } else { - if (tmp > val) { - val = tmp; - } - } - } - - return rphysmax_y = val; -} -void gGraph::SetMinX(qint64 v) -{ - rmin_x = min_x = v; -} -void gGraph::SetMaxX(qint64 v) -{ - rmax_x = max_x = v; -} -void gGraph::SetMinY(EventDataType v) -{ - rmin_y = min_y = v; -} -void gGraph::SetMaxY(EventDataType v) -{ - rmax_y = max_y = v; -} -gVertexBuffer *gGraph::lines() -{ - return m_graphview->lines; -} -gVertexBuffer *gGraph::backlines() -{ - return m_graphview->backlines; -} -gVertexBuffer *gGraph::quads() -{ - return m_graphview->quads; -} -//GLShortBuffer * gGraph::stippled() -//{ -// return m_graphview->stippled; -//} -//gVertexBuffer * gGraph::vlines() -//{ return m_graphview->vlines; } // testing new vertexbuffer - -short gGraph::marginLeft() { return m_marginleft; }//*m_graphview->printScaleX(); } -short gGraph::marginRight() { return m_marginright; } //*m_graphview->printScaleX(); } -short gGraph::marginTop() { return m_margintop; } //*m_graphview->printScaleY(); } -short gGraph::marginBottom() { return m_marginbottom; } //*m_graphview->printScaleY(); } - -Layer *gGraph::getLineChart() -{ - gLineChart *lc; - - for (int i = 0; i < m_layers.size(); i++) { - lc = dynamic_cast(m_layers[i]); - - if (lc) { return lc; } - } - - return nullptr; -} - // Render all qued text via QPainter method void gGraphView::DrawTextQue(QPainter &painter) { @@ -2299,14 +314,11 @@ QImage gGraphView::pbRenderPixmap(int w, int h) QPainter painter(&pm); DrawTextQue(painter); painter.end(); - } return pm; - } - QImage gGraphView::fboRenderPixmap(int w, int h) { QImage pm = QImage(); @@ -2361,241 +373,8 @@ QImage gGraphView::fboRenderPixmap(int w, int h) return pm; } -QPixmap gGraph::renderPixmap(int w, int h, bool printing) -{ - QPixmap pm = QPixmap(); - - gGraphView *sg = mainwin->snapshotGraph(); - - if (!sg) { - return QPixmap(); - } - - // Pixmap caching screws up font sizes when printing - sg->setUsePixmapCache(false); - - QFont *_defaultfont = defaultfont; - QFont *_mediumfont = mediumfont; - QFont *_bigfont = bigfont; - - QFont fa = *defaultfont; - QFont fb = *mediumfont; - QFont fc = *bigfont; - - - m_printing = printing; - - if (printing) { - fa.setPixelSize(30); - fb.setPixelSize(35); - fc.setPixelSize(80); - sg->setPrintScaleX(3); - sg->setPrintScaleY(3); - } else { - sg->setPrintScaleX(1); - sg->setPrintScaleY(1); - } - - defaultfont = &fa; - mediumfont = &fb; - bigfont = &fc; - - sg->hideSplitter(); - gGraphView *tgv = m_graphview; - m_graphview = sg; - - sg->setMinimumSize(w, h); - sg->setMaximumSize(w, h); - sg->setFixedSize(w, h); - - float tmp = m_height; - m_height = h; - sg->trashGraphs(); - sg->addGraph(this); - - sg->setScaleY(1.0); - - - sg->makeCurrent(); // has to be current for fbo creation - - float dpr = sg->devicePixelRatio(); - sg->setDevicePixelRatio(1); -#ifdef Q_OS_WIN - - if (pm.isNull()) { - pm = sg->renderPixmap(w, h, false); - } - - if (pm.isNull()) { // Works, but gives shader warnings - pm = QPixmap::fromImage(sg->pbRenderPixmap(w, h)); - } - - if (pm.isNull()) { // crashes on mine, not sure what to do about it - pm = QPixmap::fromImage(sg->fboRenderPixmap(w, h)); - } - -#else - pm = QPixmap::fromImage(sg->fboRenderPixmap(w, h)); - - if (pm.isNull()) { // not sure if this will work with printing - qDebug() << "Had to use PixelBuffer for snapshots\n"; - pm = QPixmap::fromImage(sg->pbRenderPixmap(w, h)); - } - -#endif - sg->setDevicePixelRatio(dpr); - //sg->doneCurrent(); - sg->trashGraphs(); - - m_graphview = tgv; - - m_height = tmp; - - defaultfont = _defaultfont; - mediumfont = _mediumfont; - bigfont = _bigfont; - m_printing = false; - - return pm; -} - -// Sets a new Min & Max X clipping, refreshing the graph and all it's layers. -void gGraph::SetXBounds(qint64 minx, qint64 maxx) -{ - invalidate_xAxisImage = true; - min_x = minx; - max_x = maxx; - - //repaint(); - //m_graphview->redraw(); -} -int gGraph::flipY(int y) -{ - return m_graphview->height() - y; -} - -void gGraph::ResetBounds() -{ - invalidate_xAxisImage = true; - min_x = MinX(); - max_x = MaxX(); - min_y = MinY(); - max_y = MaxY(); -} -void gGraph::ToolTip(QString text, int x, int y, int timeout) -{ - if (timeout <= 0) { - timeout = p_profile->general->tooltipTimeout(); - } - - m_graphview->m_tooltip->display(text, x, y, timeout); -} - -// YAxis Autoscaling code -void gGraph::roundY(EventDataType &miny, EventDataType &maxy) -{ - int m, t; - bool ymin_good = false, ymax_good = false; - - if (rec_miny != rec_maxy) { - if (miny > rec_miny) { - miny = rec_miny; - } - - if (maxy < rec_maxy) { - maxy = rec_maxy; - } - - if (miny == rec_miny) { - ymin_good = true; - } - - if (maxy == rec_maxy) { - ymax_good = true; - } - } - - if (maxy == miny) { - m = ceil(maxy / 2.0); - t = m * 2; - - if (maxy == t) { - t += 2; - } - - if (!ymax_good) { - maxy = t; - } - - m = floor(miny / 2.0); - t = m * 2; - - if (miny == t) { - t -= 2; - } - - if (miny >= 0 && t < 0) { - t = 0; - } - - if (!ymin_good) { - miny = t; - } - - return; - } - - if (maxy >= 400) { - m = ceil(maxy / 50.0); - t = m * 50; - - if (!ymax_good) { - maxy = t; - } - - m = floor(miny / 50.0); - - if (!ymin_good) { - miny = m * 50; - } - } else if (maxy >= 5) { - m = ceil(maxy / 5.0); - t = m * 5; - - if (!ymax_good) { - maxy = t; - } - - m = floor(miny / 5.0); - - if (!ymin_good) { - miny = m * 5; - } - } else { - if (maxy == miny && maxy == 0) { - maxy = 0.5; - } else { - //maxy*=4.0; - //miny*=4.0; - if (!ymax_good) { - maxy = ceil(maxy); - } - - if (!ymin_good) { - miny = floor(miny); - } - - //maxy/=4.0; - //miny/=4.0; - } - } - - //if (m_enforceMinY) { miny=f_miny; } - //if (m_enforceMaxY) { maxy=f_maxy; } -} - -gGraphView::gGraphView(QWidget *parent, gGraphView *shared) : - QGLWidget(QGLFormat(QGL::Rgba | QGL::DoubleBuffer | QGL::NoOverlay), parent, shared), +gGraphView::gGraphView(QWidget *parent, gGraphView *shared) + : QGLWidget(QGLFormat(QGL::Rgba | QGL::DoubleBuffer | QGL::NoOverlay), parent, shared), m_offsetY(0), m_offsetX(0), m_scaleY(1.0), m_scrollbar(nullptr) { m_shared = shared; @@ -2615,7 +394,7 @@ gGraphView::gGraphView(QWidget *parent, gGraphView *shared) : this->setMouseTracking(true); m_emptytext = QObject::tr("No Data"); - InitGraphs(); + InitGraphGlobals(); // FIXME: sstangl: handle error return. #ifdef ENABLE_THREADED_DRAWING m_idealthreads = QThread::idealThreadCount(); @@ -3872,11 +1651,13 @@ void gGraphView::paintGL() } 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(pixmap_cache.count(), 'f', - 0) + " strings " + QString::number(pixmap_cache_size / 1024.0, 'f', 1) + "Kb"; + 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(pixmap_cache.count(), 'f', 0) + + " strings " + QString::number(pixmap_cache_size / 1024.0, 'f', 1) + + "Kb"; + int w, h; GetTextExtent(ss, w, h); // this uses tightBoundingRect, which is different on Mac than it is on Windows & Linux. @@ -4714,7 +2495,7 @@ bool gGraphView::isEmpty() bool res = true; for (int i = 0; i < m_graphs.size(); i++) { - if (!m_graphs[i]->isEmpty()) { + if (!m_graphs.at(i)->isEmpty()) { res = false; break; } diff --git a/sleepyhead/Graphs/gGraphView.h b/sleepyhead/Graphs/gGraphView.h index 136b7435..30bfc468 100644 --- a/sleepyhead/Graphs/gGraphView.h +++ b/sleepyhead/Graphs/gGraphView.h @@ -22,270 +22,16 @@ #include #include +#include #include #include -#define MIN(a,b) (((a)<(b)) ? (a) : (b)); -#define MAX(a,b) (((a)<(b)) ? (b) : (a)); - enum FlagType { FT_Bar, FT_Dot, FT_Span }; -//! \brief Initialize the Graph Fonts -void InitGraphs(); -//! \brief Destroy the Graph Fonts -void DoneGraphs(); - -extern QFont *defaultfont; -extern QFont *mediumfont; -extern QFont *bigfont; - -extern QHash images; - -/*! \brief Gets the width and height parameters for supplied text - \param QString text - The text string in question - \param int & width - \param int & height - \param QFont * font - The selected font used in the size calculations - */ -void GetTextExtent(QString text, int &width, int &height, QFont *font = defaultfont); - -//! \brief Return the height of the letter x for the selected font. -int GetXHeight(QFont *font = defaultfont); - class gGraphView; -class gGraph; const int textque_max = 512; -typedef quint32 RGBA; -/*union RGBA { - struct { - GLubyte red; - GLubyte green; - GLubyte blue; - GLubyte alpha; - } bytes; - quint32 value; -}; */ - -#ifdef BUILD_WITH_MSVC -__declspec(align(1)) -#endif -struct gVertex { - gVertex(GLshort _x, GLshort _y, GLuint _c) { x = _x; y = _y; color = _c; } - GLshort x; - GLshort y; - RGBA color; -} -#ifndef BUILD_WITH_MSVC -__attribute__((packed)) -#endif -; - -class gVertexBuffer -{ - public: - gVertexBuffer(int max = 2048, int type = GL_LINES) - : m_max(max), m_type(type), m_cnt(0), m_size(1), - m_scissor(false), m_antialias(false), m_forceantialias(false), m_stippled(false), - buffer(nullptr), - s_x(0), s_y(0), s_width(0), s_height(0), - m_color(0), - m_stipple(0xffff), - m_blendfunc1(GL_SRC_ALPHA), - m_blendfunc2(GL_ONE_MINUS_SRC_ALPHA) { - // FIXME: Really should not allocate in constructor. - buffer = (gVertex *)calloc(max, sizeof(gVertex)); - } - - ~gVertexBuffer() { - if (buffer) { - free(buffer); - } - } - - void add(GLshort x1, GLshort y1, RGBA color); - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, RGBA color); - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, - GLshort x3, GLshort y3, GLshort x4, GLshort y4, RGBA color); - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, - GLshort x3, GLshort y3, GLshort x4, GLshort y4, RGBA color, RGBA color2); - - void add(GLshort x1, GLshort y1); - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2); - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, - GLshort x3, GLshort y3, GLshort x4, GLshort y4); - - void unsafe_add(GLshort x1, GLshort y1); - void unsafe_add(GLshort x1, GLshort y1, GLshort x2, GLshort y2); - void unsafe_add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, - GLshort x3, GLshort y3, GLshort x4, GLshort y4); - - void draw(); - - void scissor(GLshort x, GLshort y, GLshort width, GLshort height) { - s_x = x; - s_y = y; - s_width = width; - s_height = height; - m_scissor = true; - } - - int Max() const { return m_max; } - int cnt() const { return m_cnt; } - GLuint type() const { return m_type; } - float size() const { return m_size; } - bool full() const { return m_cnt >= m_max; } - - void reset() { m_cnt = 0; } - void forceAntiAlias(bool b) { m_forceantialias = b; } - void setSize(float f) { m_size = f; } - void setAntiAlias(bool b) { m_antialias = b; } - void setStipple(GLshort stipple) { m_stipple = stipple; } - void setStippleOn(bool b) { m_stippled = b; } - void setBlendFunc(GLuint b1, GLuint b2) { m_blendfunc1 = b1; m_blendfunc2 = b2; } - void setColor(QColor col); - - protected: - //! \brief Maximum number of gVertex points contained in buffer - int m_max; - //! \brief Indicates type of GL vertex information (GL_LINES, GL_QUADS, etc) - GLuint m_type; - //! \brief Count of Vertex points used this draw cycle. - int m_cnt; - //! \brief Line/Point thickness - float m_size; - - bool m_scissor; - bool m_antialias; - bool m_forceantialias; - bool m_stippled; - - //! \brief Contains list of Vertex & Color points - gVertex *buffer; - //! \brief GL Scissor parameters - GLshort s_x, s_y, s_width, s_height; - //! \brief Current drawing color - GLuint m_color; - //! \brief Stipple bitfield - GLshort m_stipple; - //! \brief Source GL Blend Function - GLuint m_blendfunc1; - //! \brief Destination GL Blend Function - GLuint m_blendfunc2; -}; - -/*! \class GLBuffer - \brief Base Object to hold an OpenGL draw list - */ -class GLBuffer -{ - public: - GLBuffer(int max = 2048, int type = GL_LINES, bool stippled = false) - : m_max(max), m_type(type), m_cnt(0), m_colcnt(0), m_size(1), - s1(0), s2(0), s3(0), s4(0), - m_scissor(false), - m_antialias(true), - m_forceantialias(false), - m_stippled(stippled), - m_blendfunc1(GL_SRC_ALPHA), - m_blendfunc2(GL_ONE_MINUS_SRC_ALPHA) - { } - virtual ~GLBuffer() {} - - void scissor(GLshort x1, GLshort y1, GLshort x2, GLshort y2) { - s1 = x1; - s2 = y1; - s3 = x2; - s4 = y2; - m_scissor = true; - } - - int Max() const { return m_max; } - int cnt() const { return m_cnt; } - bool full() const { return m_cnt >= m_max; } - float size() const { return m_size; } - int type() const { return m_type; } - - void reset() { m_cnt = 0; } - void setSize(float f) { m_size = f; } - void setAntiAlias(bool b) { m_antialias = b; } - void forceAntiAlias(bool b) { m_forceantialias = b; } - void setColor(QColor col) { m_color = col; } - void setBlendFunc(GLuint b1, GLuint b2) { m_blendfunc1 = b1; m_blendfunc2 = b2; } - - virtual void draw() {} - - protected: - int m_max; - int m_type; // type (GL_LINES, GL_QUADS, etc) - int m_cnt; // cnt - int m_colcnt; - QColor m_color; - float m_size; - int s1, s2, s3, s4; - bool m_scissor; - bool m_antialias; - bool m_forceantialias; - QMutex mutex; - bool m_stippled; - GLuint m_blendfunc1, m_blendfunc2; -}; - -/* ! \class GLShortBuffer - \brief Holds an OpenGL draw list composed of 16bit integers and vertex colors -class GLShortBuffer:public GLBuffer -{ -public: - GLShortBuffer(int max=2048,int type=GL_LINES, bool stippled=false); - virtual ~GLShortBuffer(); - - // use one or the other.. can't use both - // color free version is faster - void add(GLshort x, GLshort y); - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2); - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, GLshort y3, GLshort x4, GLshort y4); - - // color per vertex version - void add(GLshort x, GLshort y,QColor & col); // add with vertex color - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2,QColor & col); // add with vertex colors - void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2,GLshort x3, GLshort y3, GLshort x4, GLshort y4,QColor & col); // add with vertex colors - - virtual void draw(); - - //inline GLshort & operator [](int i) { return buffer[i]; } -protected: - GLshort * buffer; - GLubyte * colors; -}; - */ - -/*! \class GLFloatBuffer - \brief Holds an OpenGL draw list composed of 32bit GLfloat objects and vertex colors - */ -class GLFloatBuffer : public GLBuffer -{ - public: - GLFloatBuffer(int max = 2048, int type = GL_LINES, bool stippled = false); - virtual ~GLFloatBuffer(); - - // Add with vertex color(s). - void add(GLfloat x, GLfloat y, QColor &col); - void add(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, QColor &col); - void add(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, - GLfloat x3, GLfloat y3, GLfloat x4, GLfloat y4, QColor &col); - void quadGrTB(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, - GLfloat x3, GLfloat y3, GLfloat x4, GLfloat y4, QColor &col, QColor &col2); - void quadGrLR(GLfloat x1, GLfloat y1, GLfloat x2, GLfloat y2, - GLfloat x3, GLfloat y3, GLfloat x4, GLfloat y4, QColor &col, QColor &col2); - - virtual void draw(); - - protected: - GLfloat *buffer; - GLubyte *colors; -}; - /*! \struct TextQue \brief Holds a single item of text for the drawing queue */ @@ -321,253 +67,6 @@ class MyScrollBar : public QScrollBar } }; -enum LayerPosition { LayerLeft, LayerRight, LayerTop, LayerBottom, LayerCenter, LayerOverlay }; - -/*! \class Layer - \brief The base component for all individual Graph layers - */ -class Layer -{ - friend class gGraph; - friend class LayerGroup; - - public: - Layer(ChannelID code) - : m_refcount(0), - m_day(nullptr), - m_visible(true), - m_movable(false), - m_minx(0), m_maxx(0), - m_miny(0), m_maxy(0), - m_physminy(0), m_physmaxy(0), - m_code(code), - m_width(0), m_height(0), - m_X(0), m_Y(0), - m_order(0), - m_position(LayerCenter) - { } - - virtual ~Layer(); - - //! \brief This gets called on day selection, allowing this layer to precalculate any drawing data - virtual void SetDay(Day *d); - - //! \brief Set the ChannelID used in this layer - virtual void SetCode(ChannelID c) { m_code = c; } - //! \brief Return the ChannelID used in this layer - const ChannelID &code() { return m_code; } - - //! \brief returns true if this layer contains no data. - virtual bool isEmpty(); - - //! \brief Override and returns true if there are any highlighted components - virtual bool isSelected() { return false; } - - //! \brief Deselect any highlighted components - virtual void deselect() { } - - //! \brief Return this layers physical minimum date boundary - virtual qint64 Minx() { return m_day ? m_day->first() : m_minx; } - - //! \brief Return this layers physical maximum date boundary - virtual qint64 Maxx() { return m_day ? m_day->last() : m_maxx; } - - //! \brief Return this layers physical minimum Yaxis value - virtual EventDataType Miny() { return m_miny; } - - //! \brief Return this layers physical maximum Yaxis value - virtual EventDataType Maxy() { return m_maxy; } - - //! \brief Return this layers physical minimum Yaxis value - virtual EventDataType physMiny() { return m_physminy; } - - //! \brief Return this layers physical maximum Yaxis value - virtual EventDataType physMaxy() { return m_physmaxy; } - - //! \brief Set this layers physical minimum date boundary - virtual void setMinX(qint64 val) { m_minx = val; } - - //! \brief Set this layers physical maximum date boundary - virtual void setMaxX(qint64 val) { m_maxx = val; } - - //! \brief Set this layers physical minimum Yaxis value - virtual void setMinY(EventDataType val) { m_miny = val; } - - //! \brief Set this layers physical maximum Yaxis value - virtual void setMaxY(EventDataType val) { m_maxy = val; } - - //! \brief Set this layers Visibility status - void setVisible(bool b) { m_visible = b; } - - //! \brief Return this layers Visibility status - bool visible() const { return m_visible; } - - //! \brief Set this layers Moveability status (not really used yet) - void setMovable(bool b) { m_movable = b; } - - //! \brief Return this layers Moveability status (not really used yet) - bool movable() const { return m_movable; } - - /*! \brief Override this for the drawing code, using GLBuffer components for drawing - \param gGraph & gv Graph Object that holds this layer - \param int left - \param int top - \param int width - \param int height - */ - virtual void paint(gGraph &gv, int left, int top, int width, int height) = 0; - - //! \brief Set the layout position and order for this layer. - void setLayout(LayerPosition position, short width, short height, short order); - - void setPos(short x, short y) { m_X = x; m_Y = y; } - - int Width() { return m_width; } - int Height() { return m_height; } - - //! \brief Return this Layers Layout Position. - LayerPosition position() { return m_position; } - //void X() { return m_X; } - //void Y() { return m_Y; } - - //! \brief Draw all this layers custom GLBuffers (ie. the actual OpenGL Vertices) - virtual void drawGLBuf(float linesize); - - //! \brief not sure why I needed the reference counting stuff. - short m_refcount; - void addref() { m_refcount++; } - bool unref() { - m_refcount--; - return (m_refcount <= 0); - } - - protected: - //! \brief Add a GLBuffer (vertex) object customized to this layer - void addGLBuf(GLBuffer *buf) { mgl_buffers.push_back(buf); } - void addVertexBuffer(gVertexBuffer *buf) { mv_buffers.push_back(buf); } - - //QRect bounds; // bounds, relative to top of individual graph. - Day *m_day; - bool m_visible; - bool m_movable; - qint64 m_minx, m_maxx; - EventDataType m_miny, m_maxy; - EventDataType m_physminy, m_physmaxy; - ChannelID m_code; - short m_width; // reserved x pixels needed for this layer. 0==Depends on position.. - short m_height; // reserved y pixels needed for this layer. both 0 == expand to all free area. - short m_X; // offset for repositionable layers.. - short m_Y; - short m_order; // order for positioning.. - LayerPosition m_position; - QRect m_rect; - - //! \brief A vector containing all this layers custom drawing buffers - QVector mgl_buffers; - QVector mv_buffers; - - //! \brief Mouse wheel moved somewhere over this layer - virtual bool wheelEvent(QWheelEvent *event, gGraph *graph) { - Q_UNUSED(event); - Q_UNUSED(graph); - return false; - } - //! \brief Mouse moved somewhere over this layer - virtual bool mouseMoveEvent(QMouseEvent *event, gGraph *graph) { - Q_UNUSED(event); - Q_UNUSED(graph); - return false; - } - //! \brief Mouse left or right button pressed somewhere on this layer - virtual bool mousePressEvent(QMouseEvent *event, gGraph *graph) { - Q_UNUSED(event); - Q_UNUSED(graph); - return false; - } - //! \brief Mouse button released that was originally pressed somewhere on this layer - virtual bool mouseReleaseEvent(QMouseEvent *event, gGraph *graph) { - Q_UNUSED(event); - Q_UNUSED(graph); - return false; - } - //! \brief Mouse button double clicked somewhere on this layer - virtual bool mouseDoubleClickEvent(QMouseEvent *event, gGraph *graph) { - Q_UNUSED(event); - Q_UNUSED(graph); - return false; - } - //! \brief A key was pressed on the keyboard while the graph area was focused. - virtual bool keyPressEvent(QKeyEvent *event, gGraph *graph) { - Q_UNUSED(event); - Q_UNUSED(graph); - return false; - } -}; - -/*! \class LayerGroup - \brief Contains a list of graph Layer obejcts - */ -class LayerGroup : public Layer -{ - public: - LayerGroup() - : Layer(NoChannel) - { } - - virtual ~LayerGroup(); - - //! \brief Add Layer to this Layer Group - virtual void AddLayer(Layer *l); - - //! \brief Returns the minimum time value for all Layers contained in this group (milliseconds since epoch) - virtual qint64 Minx(); - - //! \brief Returns the maximum time value for all Layers contained in this group (milliseconds since epoch) - virtual qint64 Maxx(); - - //! \brief Returns the minimum Y-axis value for all Layers contained in this group - virtual EventDataType Miny(); - - //! \brief Returns the maximum Y-axis value for all Layers contained in this group - virtual EventDataType Maxy(); - - //! \brief Check all layers contained and return true if none contain data - virtual bool isEmpty(); - - //! \brief Calls SetDay for all Layers contained in this object - virtual void SetDay(Day *d); - - //! \brief Calls drawGLBuf for all Layers contained in this object - virtual void drawGLBuf(float linesize); - - //! \brief Return the list of Layers this object holds - QVector &getLayers() { return layers; } - - protected: - //! \brief Contains all Layer objects in this group - QVector layers; - - //! \brief Mouse wheel moved somewhere over this LayerGroup - virtual bool wheelEvent(QWheelEvent *event, gGraph *graph); - - //! \brief Mouse moved somewhere over this LayerGroup - virtual bool mouseMoveEvent(QMouseEvent *event, gGraph *graph); - - //! \brief Mouse left or right button pressed somewhere on this LayerGroup - virtual bool mousePressEvent(QMouseEvent *event, gGraph *graph); - - //! \brief Mouse button released that was originally pressed somewhere on this LayerGroup - virtual bool mouseReleaseEvent(QMouseEvent *event, gGraph *graph); - - //! \brief Mouse button double clicked somewhere on this layerGroup - virtual bool mouseDoubleClickEvent(QMouseEvent *event, gGraph *graph); - - //! \brief A key was pressed on the keyboard while the graph area was focused. - virtual bool keyPressEvent(QKeyEvent *event, gGraph *graph); -}; - -class gGraph; - /*! \class gThread \brief Part of the Threaded drawing code @@ -634,330 +133,6 @@ class gToolTip : public QObject void timerDone(); }; -/*! \class gGraph - \brief Single Graph object, containing multiple layers and Layer layout code - */ -class gGraph : public QObject -{ - Q_OBJECT - public: - friend class gGraphView; - - /*! \brief Creates a new graph object - \param gGraphView * graphview if not null, links the graph to that object - \param QString title containing the graph Title which is rendered vertically - \param int height containing the opening height for this graph - \param short group containing which graph-link group this graph belongs to - */ - gGraph(gGraphView *graphview = nullptr, QString title = "", QString units = "", - int height = 100, short group = 0); - virtual ~gGraph(); - - //! \brief Tells all Layers to deselect any highlighting - void deselect(); - - //! \brief Returns true if any Layers have anything highlighted - bool isSelected(); - - //! \brief Starts the singleshot Timer running, for ms milliseconds - void Trigger(int ms); - - /*! \fn QPixmap renderPixmap(int width, int height, float fontscale=1.0); - \brief Returns a QPixmap containing a snapshot of the graph rendered at size widthxheight - \param int width Width of graph 'screenshot' - \param int height Height of graph 'screenshot' - \param float fontscale Scaling value to adjust DPI (when used for HighRes printing) - - Note if width or height is more than the OpenGL system allows, it could result in a crash - Keeping them under 2048 is a reasonably safe value. - */ - QPixmap renderPixmap(int width, int height, bool printing = false); - - //! \brief Set Graph visibility status - void setVisible(bool b) { m_visible = b; } - - //! \brief Return Graph visibility status - bool visible() { return m_visible; } - - //! \brief Return height element. This is used by the scaler in gGraphView. - float height() { return m_height; } - - //! \brief Set the height element. (relative to the total of all heights) - void setHeight(float height) { m_height = height; invalidate_yAxisImage = true; } - - int minHeight() { return m_min_height; } - void setMinHeight(int height) { m_min_height = height; } - - int maxHeight() { return m_max_height; } - void setMaxHeight(int height) { m_max_height = height; } - - //! \brief Returns true if the vertical graph title is shown - bool showTitle() { return m_showTitle; } - - //! \brief Set whether or not to render the vertical graph title - void setShowTitle(bool b) { m_showTitle = b; } - - //! \brief Returns printScaleX, used for DPI scaling in report printing - float printScaleX(); - - //! \brief Returns printScaleY, used for DPI scaling in report printing - float printScaleY(); - - //! \brief Returns true if none of the included layers have data attached - bool isEmpty(); - - //! \brief Add Layer l to graph object, allowing you to specify position, - // margin sizes, order, movability status and offsets - void AddLayer(Layer *l, LayerPosition position = LayerCenter, - short pixelsX = 0, short pixelsY = 0, short order = 0, - bool movable = false, short x = 0, short y = 0); - - - void qglColor(QColor col); - - //! \brief Queues text for gGraphView object to draw it. - void renderText(QString text, int x, int y, float angle = 0.0, - QColor color = Qt::black, QFont *font = defaultfont, bool antialias = true); - - //! \brief Rounds Y scale values to make them look nice.. - // Applies the Graph Preference min/max settings. - void roundY(EventDataType &miny, EventDataType &maxy); - - //! \brief Process all Layers GLBuffer (Vertex) objects, drawing the actual OpenGL stuff. - void drawGLBuf(); - - //! \brief Returns the Graph's (vertical) title - QString title() { return m_title; } - - //! \brief Sets the Graph's (vertical) title - void setTitle(const QString title) { m_title = title; } - - //! \brief Returns the measurement Units the Y scale is referring to - QString units() { return m_units; } - - //! \brief Sets the measurement Units the Y scale is referring to - void setUnits(const QString units) { m_units = units; } - - //virtual void repaint(); // Repaint individual graph.. - - //! \brief Resets the graphs X & Y dimensions based on the Layer data - virtual void ResetBounds(); - - //! \brief Sets the time range selected for this graph (in milliseconds since 1970 epoch) - virtual void SetXBounds(qint64 minx, qint64 maxx); - - //! \brief Returns the physical Minimum time for all layers contained (in milliseconds since epoch) - virtual qint64 MinX(); - - //! \brief Returns the physical Maximum time for all layers contained (in milliseconds since epoch) - virtual qint64 MaxX(); - - //! \brief Returns the physical Minimum Y scale value for all layers contained - virtual EventDataType MinY(); - - //! \brief Returns the physical Maximum Y scale value for all layers contained - virtual EventDataType MaxY(); - - //! \brief Returns the physical Minimum Y scale value for all layers contained - virtual EventDataType physMinY(); - - //! \brief Returns the physical Maximum Y scale value for all layers contained - virtual EventDataType physMaxY(); - - //! \brief Sets the physical start of this graphs time range (in milliseconds since epoch) - virtual void SetMinX(qint64 v); - - //! \brief Sets the physical end of this graphs time range (in milliseconds since epoch) - virtual void SetMaxX(qint64 v); - - //! \brief Sets the physical Minimum Y scale value used while drawing this graph - virtual void SetMinY(EventDataType v); - - //! \brief Sets the physical Maximum Y scale value used while drawing this graph - virtual void SetMaxY(EventDataType v); - - //! \brief Forces Y Minimum to always select this value - virtual void setForceMinY(EventDataType v) { f_miny = v; m_enforceMinY = true; } - //! \brief Forces Y Maximum to always select this value - virtual void setForceMaxY(EventDataType v) { f_maxy = v; m_enforceMaxY = true; } - - - virtual EventDataType forceMinY() { return rec_miny; } - virtual EventDataType forceMaxY() { return rec_maxy; } - - //! \brief Set recommended Y minimum.. It won't go under this unless the data does. - // It won't go above this. - virtual void setRecMinY(EventDataType v) { rec_miny = v; } - //! \brief Set recommended Y minimum.. It won't go above this unless the data does. - // It won't go under this. - virtual void setRecMaxY(EventDataType v) { rec_maxy = v; } - - //! \brief Returns the recommended Y minimum.. It won't go under this unless the data does. - // It won't go above this. - virtual EventDataType RecMinY() { return rec_miny; } - //! \brief Returns the recommended Y maximum.. It won't go under this unless the data does. - // It won't go above this. - virtual EventDataType RecMaxY() { return rec_maxy; } - - //! \brief Called when main graph area is resized - void resize(int width, int height); // margin recalcs.. - - qint64 max_x, min_x, rmax_x, rmin_x; - EventDataType max_y, min_y, rmax_y, rmin_y, f_miny, f_maxy, rec_miny, rec_maxy; - EventDataType rphysmin_y, rphysmax_y; - - // not sure why there's two.. I can't remember - void setEnforceMinY(bool b) { m_enforceMinY = b; } - void setEnforceMaxY(bool b) { m_enforceMaxY = b; } - - //! \brief Returns whether this graph shows overall timescale, or a zoomed area - bool blockZoom() { return m_blockzoom; } - //! \brief Sets whether this graph shows an overall timescale, or a zoomed area. - void setBlockZoom(bool b) { m_blockzoom = b; } - - //! \brief Flips the GL coordinates from the graphs perspective.. Used in Scissor calculations - int flipY(int y); // flip GL coordinates - - //! \brief Returns the graph-linking group this Graph belongs in - short group() { return m_group; } - - //! \brief Sets the graph-linking group this Graph belongs in - void setGroup(short group) { m_group = group; } - - //! \brief Forces the main gGraphView object to draw all Text Components - void DrawTextQue(); - - //! \brief Sends supplied day object to all Graph layers so they can precalculate stuff - void setDay(Day *day); - - //! \brief Returns the current day object - Day *day() { return m_day; } - - //! \brief The Layer, layout and title drawing code - virtual void paint(int originX, int originY, int width, int height); - - //! \brief Gives the supplied data to the main ToolTip object for display - void ToolTip(QString text, int x, int y, int timeout = 0); - - //! \brief Public version of updateGL(), to redraw all graphs.. Not for normal use - void redraw(); - - //! \brief Asks the main gGraphView to redraw after ms milliseconds - void timedRedraw(int ms); - - //! \brief Sets the margins for the four sides of this graph. - void setMargins(short left, short right, short top, short bottom) { - m_marginleft = left; - m_marginright = right; - m_margintop = top; - m_marginbottom = bottom; - } - - //! \brief Returns this graphs left margin - short marginLeft(); - //! \brief Returns this graphs right margin - short marginRight(); - //! \brief Returns this graphs top margin - short marginTop(); - //! \brief Returns this graphs bottom margin - short marginBottom(); - - //! \brief Returns the main gGraphView objects gVertexBuffer line list. - gVertexBuffer *lines(); - //! \brief Returns the main gGraphView objects gVertexBuffer background line list. - gVertexBuffer *backlines(); - //! \brief Returns the main gGraphView objects gVertexBuffer front line list. - gVertexBuffer *frontlines(); - //! \brief Returns the main gGraphView objects gVertexBuffer quads list. - gVertexBuffer *quads(); - - const inline QRect &rect() const { return m_rect; } - - bool isPinned() { return m_pinned; } - void setPinned(bool b) { m_pinned = b; } - - // //! \brief Returns the main gGraphView objects gVertexBuffer stippled line list. - //GLShortBuffer * stippled(); - - //gVertexBuffer * vlines(); // testing new vertexbuffer - - short left, right, top, bottom; // dirty magin hacks.. - - Layer *getLineChart(); - QTimer *timer; - - // This gets set to true to force a redraw of the yAxis tickers when graphs are resized. - bool invalidate_yAxisImage; - bool invalidate_xAxisImage; - - //! \brief Returns a Vector reference containing all this graphs layers - QVector &layers() { return m_layers; } - - gGraphView *graphView() { return m_graphview; } - short m_marginleft, m_marginright, m_margintop, m_marginbottom; - - short zoomY() { return m_zoomY; } - void setZoomY(short zoom); - - static const short maxZoomY = 2; - - protected: - //void invalidate(); - - //! \brief Mouse Wheel events - virtual void wheelEvent(QWheelEvent *event); - - //! \brief Mouse Movement events - virtual void mouseMoveEvent(QMouseEvent *event); - - //! \brief Mouse Button Pressed events - virtual void mousePressEvent(QMouseEvent *event); - - //! \brief Mouse Button Released events - virtual void mouseReleaseEvent(QMouseEvent *event); - - //! \brief Mouse Button Double Clicked events - virtual void mouseDoubleClickEvent(QMouseEvent *event); - - //! \brief Key Pressed event - virtual void keyPressEvent(QKeyEvent *event); - - //! \brief Change the current selected time boundaries by mult, from origin position origin_px - void ZoomX(double mult, int origin_px); - - //! \brief The Main gGraphView object holding this graph - // (this can be pinched temporarily by print code) - gGraphView *m_graphview; - QString m_title; - QString m_units; - - //! \brief Vector containing all this graphs Layers - QVector m_layers; - float m_height, m_width; - - int m_min_height; - int m_max_height; - bool m_visible; - bool m_blockzoom; - QRect m_selection; - bool m_selecting_area; - QPoint m_current; - short m_group; - short m_lastx23; - Day *m_day; - gVertexBuffer *m_quad; - bool m_enforceMinY, m_enforceMaxY; - bool m_showTitle; - bool m_printing; - bool m_pinned; - short m_zoomY; - QRect m_rect; - - protected slots: - //! \brief Deselects any highlights, and schedules a main gGraphView redraw - void Timeout(); -}; - /*! \struct myPixmapCache \brief My version of Pixmap cache with texture binding support diff --git a/sleepyhead/Graphs/gLineChart.cpp b/sleepyhead/Graphs/gLineChart.cpp index 989447dc..43e59180 100644 --- a/sleepyhead/Graphs/gLineChart.cpp +++ b/sleepyhead/Graphs/gLineChart.cpp @@ -9,12 +9,17 @@ * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ -#include +#include "Graphs/gLineChart.h" + #include #include -#include -#include "gLineChart.h" -#include "glcommon.h" + +#include + +#include "Graphs/glcommon.h" +#include "Graphs/gGraph.h" +#include "Graphs/gGraphView.h" +#include "SleepLib/profiles.h" #define EXTRA_ASSERTS 1 gLineChart::gLineChart(ChannelID code, QColor col, bool square_plot, bool disable_accel) diff --git a/sleepyhead/Graphs/gLineChart.h b/sleepyhead/Graphs/gLineChart.h index c47de6a0..cf34b85c 100644 --- a/sleepyhead/Graphs/gLineChart.h +++ b/sleepyhead/Graphs/gLineChart.h @@ -13,10 +13,11 @@ #define GLINECHART_H #include + +#include "Graphs/gVertexBuffer.h" +#include "Graphs/layer.h" #include "SleepLib/event.h" #include "SleepLib/day.h" -#include "gGraphView.h" -//#include "graphlayer.h" /*! \class AHIChart \brief Another graph calculating the AHI/hour, this one looks at all the sessions for a day. Currently Unused. diff --git a/sleepyhead/Graphs/gVertexBuffer.cpp b/sleepyhead/Graphs/gVertexBuffer.cpp new file mode 100644 index 00000000..a516e1d9 --- /dev/null +++ b/sleepyhead/Graphs/gVertexBuffer.cpp @@ -0,0 +1,304 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * Copyright (c) 2011-2014 Mark Watkins + * + * 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 Linux + * distribution for more details. */ + +#include "Graphs/gVertexBuffer.h" + +#include "SleepLib/profiles.h" + +extern int lines_drawn_this_frame; +extern int quads_drawn_this_frame; + +inline quint32 swaporder(quint32 color) +{ + return ((color & 0xFF00FF00) | + ((color & 0xFF0000) >> 16) | + ((color & 0xFF) << 16)); +} + +void gVertexBuffer::setColor(QColor col) +{ + m_color = swaporder(col.rgba()); +} + +void gVertexBuffer::draw() +{ + bool antialias = m_forceantialias || (PROFILE.appearance->antiAliasing() && m_antialias); + + if (m_stippled) { antialias = false; } + + float size = m_size; + + if (antialias) { + glEnable(GL_BLEND); + glBlendFunc(m_blendfunc1, m_blendfunc2); + + if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { + glEnable(GL_LINE_SMOOTH); + glHint(GL_LINE_SMOOTH_HINT, GL_NICEST); + size += 0.5; + } else if (m_type == GL_POLYGON) { + glEnable(GL_POLYGON_SMOOTH); + glHint(GL_POLYGON_SMOOTH_HINT, GL_NICEST); + } + } + + if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { + if (m_stippled) { + glLineStipple(1, m_stipple); + //size=1; + glEnable(GL_LINE_STIPPLE); + } else { + //glLineStipple(1, 0xFFFF); + } + + glLineWidth(size); + + lines_drawn_this_frame += m_cnt / 2; + } else if (m_type == GL_POINTS) { + glPointSize(size); + } else if (m_type == GL_POLYGON) { + glPolygonMode(GL_BACK, GL_FILL); + lines_drawn_this_frame += m_cnt / 2; + } else if (m_type == GL_QUADS) { + quads_drawn_this_frame += m_cnt / 4; + } + + if (m_scissor) { + glScissor(s_x, s_y, s_width, s_height); + glEnable(GL_SCISSOR_TEST); + } + + glEnableClientState(GL_VERTEX_ARRAY); + glEnableClientState(GL_COLOR_ARRAY); + glVertexPointer(2, GL_SHORT, 8, (GLvoid *)buffer); + glColorPointer(4, GL_UNSIGNED_BYTE, 8, ((char *)buffer) + 4); + + glDrawArrays(m_type, 0, m_cnt); + + glDisableClientState(GL_COLOR_ARRAY); + glDisableClientState(GL_VERTEX_ARRAY); + m_cnt = 0; + + if (m_scissor) { + glDisable(GL_SCISSOR_TEST); + m_scissor = false; + } + + if (m_type == GL_POLYGON) { + glPolygonMode(GL_BACK, GL_FILL); + } + + if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { + if (m_stippled) { + glDisable(GL_LINE_STIPPLE); + glLineStipple(1, 0xFFFF); + } + } + + if (antialias) { + if (m_type == GL_LINES || m_type == GL_LINE_LOOP) { + glDisable(GL_LINE_SMOOTH); + } else if (m_type == GL_POLYGON) { + glDisable(GL_POLYGON_SMOOTH); + } + + glDisable(GL_BLEND); + } +} +void gVertexBuffer::add(GLshort x1, GLshort y1, RGBA color) +{ + if (m_cnt < m_max) { + gVertex &v = buffer[m_cnt]; + + v.color = swaporder(color); + v.x = x1; + v.y = y1; + + m_cnt++; + } +} +void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, RGBA color) +{ + if (m_cnt < (m_max - 1)) { + gVertex *v = &buffer[m_cnt]; + + v->x = x1; + v->y = y1; + v->color = swaporder(color); + + v++; + v->x = x2; + v->y = y2; + v->color = swaporder(color); + + m_cnt += 2; + } +} +void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, GLshort y3, + GLshort x4, GLshort y4, RGBA color) +{ + if (m_cnt < (m_max - 3)) { + gVertex *v = &buffer[m_cnt]; + + v->color = swaporder(color); + v->x = x1; + v->y = y1; + v++; + + v->color = swaporder(color); + v->x = x2; + v->y = y2; + + v++; + v->color = swaporder(color); + v->x = x3; + v->y = y3; + + v++; + v->color = swaporder(color); + v->x = x4; + v->y = y4; + + m_cnt += 4; + } +} + +void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, GLshort y3, + GLshort x4, GLshort y4, RGBA color1, RGBA color2) +{ + if (m_cnt < (m_max - 3)) { + gVertex *v = &buffer[m_cnt]; + + v->color = swaporder(color1); + v->x = x1; + v->y = y1; + v++; + + v->color = swaporder(color1); + v->x = x2; + v->y = y2; + + v++; + v->color = swaporder(color2); + v->x = x3; + v->y = y3; + + v++; + v->color = swaporder(color2); + v->x = x4; + v->y = y4; + + m_cnt += 4; + } +} +void gVertexBuffer::unsafe_add(GLshort x1, GLshort y1) +{ + gVertex &v = buffer[m_cnt++]; + + v.color = m_color; + v.x = x1; + v.y = y1; +} +void gVertexBuffer::add(GLshort x1, GLshort y1) +{ + if (m_cnt < m_max) { + gVertex &v = buffer[m_cnt++]; + + v.color = m_color; + v.x = x1; + v.y = y1; + } +} +void gVertexBuffer::unsafe_add(GLshort x1, GLshort y1, GLshort x2, GLshort y2) +{ + gVertex *v = &buffer[m_cnt]; + + v->x = x1; + v->y = y1; + v->color = m_color; + + v++; + v->x = x2; + v->y = y2; + v->color = m_color; + + m_cnt += 2; +} + +void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2) +{ + if (m_cnt < (m_max - 1)) { + gVertex *v = &buffer[m_cnt]; + + v->x = x1; + v->y = y1; + v->color = m_color; + + v++; + v->x = x2; + v->y = y2; + v->color = m_color; + + m_cnt += 2; + } +} +void gVertexBuffer::add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, GLshort y3, + GLshort x4, GLshort y4) +{ + if (m_cnt < (m_max - 3)) { + gVertex *v = &buffer[m_cnt]; + + v->color = m_color; + v->x = x1; + v->y = y1; + v++; + + v->color = m_color; + v->x = x2; + v->y = y2; + + v++; + v->color = m_color; + v->x = x3; + v->y = y3; + + v++; + v->color = m_color; + v->x = x4; + v->y = y4; + + m_cnt += 4; + } +} +void gVertexBuffer::unsafe_add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, GLshort x3, + GLshort y3, GLshort x4, GLshort y4) +{ + gVertex *v = &buffer[m_cnt]; + + v->color = m_color; + v->x = x1; + v->y = y1; + v++; + + v->color = m_color; + v->x = x2; + v->y = y2; + + v++; + v->color = m_color; + v->x = x3; + v->y = y3; + + v++; + v->color = m_color; + v->x = x4; + v->y = y4; + + m_cnt += 4; +} diff --git a/sleepyhead/Graphs/gVertexBuffer.h b/sleepyhead/Graphs/gVertexBuffer.h new file mode 100644 index 00000000..2759739e --- /dev/null +++ b/sleepyhead/Graphs/gVertexBuffer.h @@ -0,0 +1,128 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * Copyright (c) 2011-2014 Mark Watkins + * + * 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 Linux + * distribution for more details. */ + +#ifndef graphs_gvertexbuffer_h +#define graphs_gvertexbuffer_h + +#include +#include + +#include "Graphs/glcommon.h" + +typedef quint32 RGBA; + +#ifdef BUILD_WITH_MSVC +__declspec(align(1)) +#endif +struct gVertex { + gVertex(GLshort _x, GLshort _y, GLuint _c) { x = _x; y = _y; color = _c; } + GLshort x; + GLshort y; + RGBA color; +} +#ifndef BUILD_WITH_MSVC +__attribute__((packed)) +#endif +; + +class gVertexBuffer +{ + public: + gVertexBuffer(int max = 2048, int type = GL_LINES) + : m_max(max), m_type(type), m_cnt(0), m_size(1), + m_scissor(false), m_antialias(false), m_forceantialias(false), m_stippled(false), + buffer(nullptr), + s_x(0), s_y(0), s_width(0), s_height(0), + m_color(0), + m_stipple(0xffff), + m_blendfunc1(GL_SRC_ALPHA), + m_blendfunc2(GL_ONE_MINUS_SRC_ALPHA) + { + // FIXME: sstangl: Really should not allocate in constructor. + buffer = (gVertex *)calloc(max, sizeof(gVertex)); + } + + ~gVertexBuffer() { + if (buffer) { + free(buffer); + } + } + + void add(GLshort x1, GLshort y1, RGBA color); + void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, RGBA color); + void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, + GLshort x3, GLshort y3, GLshort x4, GLshort y4, RGBA color); + void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, + GLshort x3, GLshort y3, GLshort x4, GLshort y4, RGBA color, RGBA color2); + + void add(GLshort x1, GLshort y1); + void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2); + void add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, + GLshort x3, GLshort y3, GLshort x4, GLshort y4); + + void unsafe_add(GLshort x1, GLshort y1); + void unsafe_add(GLshort x1, GLshort y1, GLshort x2, GLshort y2); + void unsafe_add(GLshort x1, GLshort y1, GLshort x2, GLshort y2, + GLshort x3, GLshort y3, GLshort x4, GLshort y4); + + void draw(); + + void scissor(GLshort x, GLshort y, GLshort width, GLshort height) { + s_x = x; + s_y = y; + s_width = width; + s_height = height; + m_scissor = true; + } + + int Max() const { return m_max; } + int cnt() const { return m_cnt; } + GLuint type() const { return m_type; } + float size() const { return m_size; } + bool full() const { return m_cnt >= m_max; } + + void reset() { m_cnt = 0; } + void forceAntiAlias(bool b) { m_forceantialias = b; } + void setSize(float f) { m_size = f; } + void setAntiAlias(bool b) { m_antialias = b; } + void setStipple(GLshort stipple) { m_stipple = stipple; } + void setStippleOn(bool b) { m_stippled = b; } + void setBlendFunc(GLuint b1, GLuint b2) { m_blendfunc1 = b1; m_blendfunc2 = b2; } + void setColor(QColor col); + + protected: + //! \brief Maximum number of gVertex points contained in buffer + int m_max; + //! \brief Indicates type of GL vertex information (GL_LINES, GL_QUADS, etc) + GLuint m_type; + //! \brief Count of Vertex points used this draw cycle. + int m_cnt; + //! \brief Line/Point thickness + float m_size; + + bool m_scissor; + bool m_antialias; + bool m_forceantialias; + bool m_stippled; + + //! \brief Contains list of Vertex & Color points + gVertex *buffer; + //! \brief GL Scissor parameters + GLshort s_x, s_y, s_width, s_height; + //! \brief Current drawing color + GLuint m_color; + //! \brief Stipple bitfield + GLshort m_stipple; + //! \brief Source GL Blend Function + GLuint m_blendfunc1; + //! \brief Destination GL Blend Function + GLuint m_blendfunc2; +}; + +#endif // graphs_gvertexbuffer_h diff --git a/sleepyhead/Graphs/gXAxis.cpp b/sleepyhead/Graphs/gXAxis.cpp index d768f632..71599701 100644 --- a/sleepyhead/Graphs/gXAxis.cpp +++ b/sleepyhead/Graphs/gXAxis.cpp @@ -9,10 +9,15 @@ * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ -#include +#include "Graphs/gXAxis.h" + #include -#include "gXAxis.h" +#include + +#include "Graphs/glcommon.h" +#include "Graphs/gGraph.h" +#include "Graphs/gGraphView.h" const quint64 divisors[] = { 15552000000ULL, 7776000000ULL, 5184000000ULL, 2419200000ULL, 1814400000ULL, 1209600000L, 604800000L, 259200000L, diff --git a/sleepyhead/Graphs/gXAxis.h b/sleepyhead/Graphs/gXAxis.h index 96efac68..9c629d2a 100644 --- a/sleepyhead/Graphs/gXAxis.h +++ b/sleepyhead/Graphs/gXAxis.h @@ -11,7 +11,8 @@ #ifndef GXAXIS_H #define GXAXIS_H -#include "gGraphView.h" + +#include "Graphs/layer.h" /*! \class gXAxis \brief Draws the XTicker timescales underneath graphs */ diff --git a/sleepyhead/Graphs/gYAxis.cpp b/sleepyhead/Graphs/gYAxis.cpp index 144ae5d1..af276a2f 100644 --- a/sleepyhead/Graphs/gYAxis.cpp +++ b/sleepyhead/Graphs/gYAxis.cpp @@ -9,9 +9,15 @@ * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ -#include +#include "Graphs/gYAxis.h" + #include -#include "gYAxis.h" + +#include + +#include "Graphs/glcommon.h" +#include "Graphs/gGraph.h" +#include "Graphs/gGraphView.h" #include "SleepLib/profiles.h" gXGrid::gXGrid(QColor col) diff --git a/sleepyhead/Graphs/gYAxis.h b/sleepyhead/Graphs/gYAxis.h index bf2de33c..8a23be9e 100644 --- a/sleepyhead/Graphs/gYAxis.h +++ b/sleepyhead/Graphs/gYAxis.h @@ -12,8 +12,7 @@ #ifndef GYAXIS_H #define GYAXIS_H -#include "gGraphView.h" - +#include "Graphs/layer.h" /*! \class gXGrid \brief Draws the horizintal major/minor grids over graphs diff --git a/sleepyhead/Graphs/glcommon.h b/sleepyhead/Graphs/glcommon.h index 104756c7..af95cdb8 100644 --- a/sleepyhead/Graphs/glcommon.h +++ b/sleepyhead/Graphs/glcommon.h @@ -15,6 +15,16 @@ #include #include +#ifdef Q_OS_MAC +# define USE_RENDERTEXT +# include "OpenGL/glu.h" +#else +# include "GL/glu.h" +#endif + +#define MIN(a,b) (((a)<(b)) ? (a) : (b)); +#define MAX(a,b) (((a)<(b)) ? (b) : (a)); + const QColor COLOR_Black = Qt::black; const QColor COLOR_LightGreen = QColor("light green"); const QColor COLOR_DarkGreen = Qt::darkGreen; diff --git a/sleepyhead/Graphs/layer.cpp b/sleepyhead/Graphs/layer.cpp new file mode 100644 index 00000000..bf52431b --- /dev/null +++ b/sleepyhead/Graphs/layer.cpp @@ -0,0 +1,268 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * Copyright (c) 2011-2014 Mark Watkins + * + * 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 Linux + * distribution for more details. */ + +#include "Graphs/layer.h" + +Layer::~Layer() +{ + for (int i = 0; i < mgl_buffers.size(); i++) { + delete mgl_buffers[i]; + } + + for (int i = 0; i < mv_buffers.size(); i++) { + delete mv_buffers[i]; + } +} +void Layer::drawGLBuf(float linesize) +{ + int type; + float size; + + if (!m_visible) { return; } + + GLBuffer *buf; + gVertexBuffer *vb; + + for (int i = 0; i < mv_buffers.size(); i++) { + vb = mv_buffers[i]; + size = vb->size(); + type = vb->type(); + + if ((linesize > size) && ((type == GL_LINES) || (type == GL_LINE_LOOP))) { + vb->setSize(linesize); + } + + vb->draw(); + vb->setSize(size); + } + + for (int i = 0; i < mgl_buffers.size(); i++) { + buf = mgl_buffers[i]; + size = buf->size(); + type = buf->type(); + + if ((linesize > size) && ((type == GL_LINES) || (type == GL_LINE_LOOP))) { + buf->setSize(linesize); + } + + buf->draw(); + //if ((linesize>size) && ((type==GL_LINES) || (type==GL_LINE_LOOP))) { + buf->setSize(size); + //} + } +} + +void Layer::SetDay(Day *d) +{ + if (d) { + m_day = d; + m_minx = d->first(m_code); + m_maxx = d->last(m_code); + m_miny = d->Min(m_code); + m_maxy = d->Max(m_code); + } else { m_day = nullptr; } + +} + +bool Layer::isEmpty() +{ + //if (m_day && (m_day->count(m_code)>0)) + if (m_day && (m_day->channelExists(m_code))) { + return false; + } + + return true; +} +void Layer::setLayout(LayerPosition position, short width, short height, short order) +{ + m_position = position; + m_width = width; + m_height = height; + m_order = order; +} + +LayerGroup::~LayerGroup() +{ + for (int i = 0; i < layers.size(); i++) { + delete layers[i]; + } +} +bool LayerGroup::isEmpty() +{ + if (!m_day) { + return true; + } + + bool empty = true; + + for (int i = 0; i < layers.size(); i++) { + if (layers[i]->isEmpty()) { + empty = false; + break; + } + } + + return empty; +} +void LayerGroup::drawGLBuf(float linesize) +{ + Layer::drawGLBuf(linesize); + + for (int i = 0; i < layers.size(); i++) { + layers[i]->drawGLBuf(linesize); + } +} + +void LayerGroup::SetDay(Day *d) +{ + m_day = d; + + for (int i = 0; i < layers.size(); i++) { + layers[i]->SetDay(d); + } +} + +void LayerGroup::AddLayer(Layer *l) +{ + layers.push_back(l); + l->addref(); +} + +qint64 LayerGroup::Minx() +{ + bool first = true; + qint64 m = 0, t; + + for (int i = 0; i < layers.size(); i++) { + t = layers[i]->Minx(); + + if (!t) { continue; } + + if (first) { + m = t; + first = false; + } else if (m > t) { m = t; } + } + + return m; +} +qint64 LayerGroup::Maxx() +{ + bool first = true; + qint64 m = 0, t; + + for (int i = 0; i < layers.size(); i++) { + t = layers[i]->Maxx(); + + if (!t) { continue; } + + if (first) { + m = t; + first = false; + } else if (m < t) { m = t; } + } + + return m; +} + +EventDataType LayerGroup::Miny() +{ + bool first = true; + EventDataType m = 0, t; + + for (int i = 0; i < layers.size(); i++) { + t = layers[i]->Miny(); + + if (t == layers[i]->Maxy()) { continue; } + + if (first) { + m = t; + first = false; + } else { + if (m > t) { m = t; } + } + } + + return m; +} + +EventDataType LayerGroup::Maxy() +{ + bool first = true; + EventDataType m = 0, t; + + for (int i = 0; i < layers.size(); i++) { + t = layers[i]->Maxy(); + + if (t == layers[i]->Miny()) { continue; } + + if (first) { + m = t; + first = false; + } else if (m < t) { m = t; } + } + + return m; +} + +//! \brief Mouse wheel moved somewhere over this layer +bool LayerGroup::wheelEvent(QWheelEvent *event, gGraph *graph) +{ + for (int i = 0; i < layers.size(); i++) + if (layers[i]->wheelEvent(event, graph)) { + return true; + } + + return false; +} + +//! \brief Mouse moved somewhere over this layer +bool LayerGroup::mouseMoveEvent(QMouseEvent *event, gGraph *graph) +{ + for (int i = 0; i < layers.size(); i++) + if (layers[i]->mouseMoveEvent(event, graph)) { return true; } + + return false; +} + +//! \brief Mouse left or right button pressed somewhere on this layer +bool LayerGroup::mousePressEvent(QMouseEvent *event, gGraph *graph) +{ + for (int i = 0; i < layers.size(); i++) + if (layers[i]->mousePressEvent(event, graph)) { return true; } + + return false; +} + +//! \brief Mouse button released that was originally pressed somewhere on this layer +bool LayerGroup::mouseReleaseEvent(QMouseEvent *event, gGraph *graph) +{ + for (int i = 0; i < layers.size(); i++) + if (layers[i]->mouseReleaseEvent(event, graph)) { return true; } + + return false; +} + +//! \brief Mouse button double clicked somewhere on this layer +bool LayerGroup::mouseDoubleClickEvent(QMouseEvent *event, gGraph *graph) +{ + for (int i = 0; i < layers.size(); i++) + if (layers[i]->mouseDoubleClickEvent(event, graph)) { return true; } + + return false; +} + +//! \brief A key was pressed on the keyboard while the graph area was focused. +bool LayerGroup::keyPressEvent(QKeyEvent *event, gGraph *graph) +{ + for (int i = 0; i < layers.size(); i++) + if (layers[i]->keyPressEvent(event, graph)) { return true; } + + return false; +} diff --git a/sleepyhead/Graphs/layer.h b/sleepyhead/Graphs/layer.h new file mode 100644 index 00000000..1008664d --- /dev/null +++ b/sleepyhead/Graphs/layer.h @@ -0,0 +1,273 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- + * vim: set ts=8 sts=4 et sw=4 tw=99: + * + * Copyright (c) 2011-2014 Mark Watkins + * + * 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 Linux + * distribution for more details. */ + +#ifndef graphs_layer_h +#define graphs_layer_h + +#include +#include +#include +#include +#include + +#include "Graphs/GLBuffer.h" +#include "Graphs/gVertexBuffer.h" +#include "SleepLib/common.h" +#include "SleepLib/day.h" +#include "SleepLib/machine_common.h" + +class gGraph; +class LayerGroup; + +enum LayerPosition { LayerLeft, LayerRight, LayerTop, LayerBottom, LayerCenter, LayerOverlay }; + +/*! \class Layer + \brief The base component for all individual Graph layers + */ +class Layer +{ + friend class gGraph; + friend class LayerGroup; + + public: + Layer(ChannelID code) + : m_refcount(0), + m_day(nullptr), + m_visible(true), + m_movable(false), + m_minx(0), m_maxx(0), + m_miny(0), m_maxy(0), + m_physminy(0), m_physmaxy(0), + m_code(code), + m_width(0), m_height(0), + m_X(0), m_Y(0), + m_order(0), + m_position(LayerCenter) + { } + + virtual ~Layer(); + + //! \brief This gets called on day selection, allowing this layer to precalculate any drawing data + virtual void SetDay(Day *d); + + //! \brief Set the ChannelID used in this layer + virtual void SetCode(ChannelID c) { m_code = c; } + //! \brief Return the ChannelID used in this layer + const ChannelID &code() { return m_code; } + + //! \brief returns true if this layer contains no data. + virtual bool isEmpty(); + + //! \brief Override and returns true if there are any highlighted components + virtual bool isSelected() { return false; } + + //! \brief Deselect any highlighted components + virtual void deselect() { } + + //! \brief Return this layers physical minimum date boundary + virtual qint64 Minx() { return m_day ? m_day->first() : m_minx; } + + //! \brief Return this layers physical maximum date boundary + virtual qint64 Maxx() { return m_day ? m_day->last() : m_maxx; } + + //! \brief Return this layers physical minimum Yaxis value + virtual EventDataType Miny() { return m_miny; } + + //! \brief Return this layers physical maximum Yaxis value + virtual EventDataType Maxy() { return m_maxy; } + + //! \brief Return this layers physical minimum Yaxis value + virtual EventDataType physMiny() { return m_physminy; } + + //! \brief Return this layers physical maximum Yaxis value + virtual EventDataType physMaxy() { return m_physmaxy; } + + //! \brief Set this layers physical minimum date boundary + virtual void setMinX(qint64 val) { m_minx = val; } + + //! \brief Set this layers physical maximum date boundary + virtual void setMaxX(qint64 val) { m_maxx = val; } + + //! \brief Set this layers physical minimum Yaxis value + virtual void setMinY(EventDataType val) { m_miny = val; } + + //! \brief Set this layers physical maximum Yaxis value + virtual void setMaxY(EventDataType val) { m_maxy = val; } + + //! \brief Set this layers Visibility status + void setVisible(bool b) { m_visible = b; } + + //! \brief Return this layers Visibility status + bool visible() const { return m_visible; } + + //! \brief Set this layers Moveability status (not really used yet) + void setMovable(bool b) { m_movable = b; } + + //! \brief Return this layers Moveability status (not really used yet) + bool movable() const { return m_movable; } + + /*! \brief Override this for the drawing code, using GLBuffer components for drawing + \param gGraph & gv Graph Object that holds this layer + \param int left + \param int top + \param int width + \param int height + */ + virtual void paint(gGraph &gv, int left, int top, int width, int height) = 0; + + //! \brief Set the layout position and order for this layer. + void setLayout(LayerPosition position, short width, short height, short order); + + void setPos(short x, short y) { m_X = x; m_Y = y; } + + int Width() { return m_width; } + int Height() { return m_height; } + + //! \brief Return this Layers Layout Position. + LayerPosition position() { return m_position; } + //void X() { return m_X; } + //void Y() { return m_Y; } + + //! \brief Draw all this layers custom GLBuffers (ie. the actual OpenGL Vertices) + virtual void drawGLBuf(float linesize); + + //! \brief not sure why I needed the reference counting stuff. + short m_refcount; + void addref() { m_refcount++; } + bool unref() { + m_refcount--; + return (m_refcount <= 0); + } + + protected: + //! \brief Add a GLBuffer (vertex) object customized to this layer + void addGLBuf(GLBuffer *buf) { mgl_buffers.push_back(buf); } + void addVertexBuffer(gVertexBuffer *buf) { mv_buffers.push_back(buf); } + + //QRect bounds; // bounds, relative to top of individual graph. + Day *m_day; + bool m_visible; + bool m_movable; + qint64 m_minx, m_maxx; + EventDataType m_miny, m_maxy; + EventDataType m_physminy, m_physmaxy; + ChannelID m_code; + short m_width; // reserved x pixels needed for this layer. 0==Depends on position.. + short m_height; // reserved y pixels needed for this layer. both 0 == expand to all free area. + short m_X; // offset for repositionable layers.. + short m_Y; + short m_order; // order for positioning.. + LayerPosition m_position; + QRect m_rect; + + //! \brief A vector containing all this layers custom drawing buffers + QVector mgl_buffers; + QVector mv_buffers; + + //! \brief Mouse wheel moved somewhere over this layer + virtual bool wheelEvent(QWheelEvent *event, gGraph *graph) { + Q_UNUSED(event); + Q_UNUSED(graph); + return false; + } + //! \brief Mouse moved somewhere over this layer + virtual bool mouseMoveEvent(QMouseEvent *event, gGraph *graph) { + Q_UNUSED(event); + Q_UNUSED(graph); + return false; + } + //! \brief Mouse left or right button pressed somewhere on this layer + virtual bool mousePressEvent(QMouseEvent *event, gGraph *graph) { + Q_UNUSED(event); + Q_UNUSED(graph); + return false; + } + //! \brief Mouse button released that was originally pressed somewhere on this layer + virtual bool mouseReleaseEvent(QMouseEvent *event, gGraph *graph) { + Q_UNUSED(event); + Q_UNUSED(graph); + return false; + } + //! \brief Mouse button double clicked somewhere on this layer + virtual bool mouseDoubleClickEvent(QMouseEvent *event, gGraph *graph) { + Q_UNUSED(event); + Q_UNUSED(graph); + return false; + } + //! \brief A key was pressed on the keyboard while the graph area was focused. + virtual bool keyPressEvent(QKeyEvent *event, gGraph *graph) { + Q_UNUSED(event); + Q_UNUSED(graph); + return false; + } +}; + +/*! \class LayerGroup + \brief Contains a list of graph Layer objects + */ +class LayerGroup : public Layer +{ + public: + LayerGroup() + : Layer(NoChannel) + { } + + virtual ~LayerGroup(); + + //! \brief Add Layer to this Layer Group + virtual void AddLayer(Layer *l); + + //! \brief Returns the minimum time value for all Layers contained in this group (milliseconds since epoch) + virtual qint64 Minx(); + + //! \brief Returns the maximum time value for all Layers contained in this group (milliseconds since epoch) + virtual qint64 Maxx(); + + //! \brief Returns the minimum Y-axis value for all Layers contained in this group + virtual EventDataType Miny(); + + //! \brief Returns the maximum Y-axis value for all Layers contained in this group + virtual EventDataType Maxy(); + + //! \brief Check all layers contained and return true if none contain data + virtual bool isEmpty(); + + //! \brief Calls SetDay for all Layers contained in this object + virtual void SetDay(Day *d); + + //! \brief Calls drawGLBuf for all Layers contained in this object + virtual void drawGLBuf(float linesize); + + //! \brief Return the list of Layers this object holds + QVector &getLayers() { return layers; } + + protected: + //! \brief Contains all Layer objects in this group + QVector layers; + + //! \brief Mouse wheel moved somewhere over this LayerGroup + virtual bool wheelEvent(QWheelEvent *event, gGraph *graph); + + //! \brief Mouse moved somewhere over this LayerGroup + virtual bool mouseMoveEvent(QMouseEvent *event, gGraph *graph); + + //! \brief Mouse left or right button pressed somewhere on this LayerGroup + virtual bool mousePressEvent(QMouseEvent *event, gGraph *graph); + + //! \brief Mouse button released that was originally pressed somewhere on this LayerGroup + virtual bool mouseReleaseEvent(QMouseEvent *event, gGraph *graph); + + //! \brief Mouse button double clicked somewhere on this layerGroup + virtual bool mouseDoubleClickEvent(QMouseEvent *event, gGraph *graph); + + //! \brief A key was pressed on the keyboard while the graph area was focused. + virtual bool keyPressEvent(QKeyEvent *event, gGraph *graph); +}; + +#endif // graphs_layer_h diff --git a/sleepyhead/mainwindow.cpp b/sleepyhead/mainwindow.cpp index c10076a9..6c41ed11 100644 --- a/sleepyhead/mainwindow.cpp +++ b/sleepyhead/mainwindow.cpp @@ -234,7 +234,7 @@ MainWindow::~MainWindow() } // Trash anything allocated by the Graph objects - DoneGraphs(); + DestroyGraphGlobals(); // Shutdown and Save the current User profile Profiles::Done(); diff --git a/sleepyhead/oximetry.cpp b/sleepyhead/oximetry.cpp index c64101bc..8dfd098f 100644 --- a/sleepyhead/oximetry.cpp +++ b/sleepyhead/oximetry.cpp @@ -28,6 +28,7 @@ #include "SleepLib/loader_plugins/cms50_loader.h" #include "SleepLib/event.h" #include "SleepLib/calcs.h" +#include "Graphs/gFooBar.h" #include "Graphs/gXAxis.h" #include "Graphs/gSummaryChart.h" #include "Graphs/gLineChart.h" diff --git a/sleepyhead/oximetry.h b/sleepyhead/oximetry.h index 034664ac..0bfac21d 100644 --- a/sleepyhead/oximetry.h +++ b/sleepyhead/oximetry.h @@ -26,7 +26,7 @@ #include "SleepLib/session.h" #include "Graphs/gLineChart.h" -#include "Graphs/gFooBar.h" +#include "Graphs/gGraphView.h" //! \brief Oximeters current mode enum SerialOxMode { SO_OFF, SO_IMPORT, SO_LIVE, SO_WAIT }; diff --git a/sleepyhead/sleepyhead.pro b/sleepyhead/sleepyhead.pro index 0006c02d..5db9cf88 100644 --- a/sleepyhead/sleepyhead.pro +++ b/sleepyhead/sleepyhead.pro @@ -73,102 +73,110 @@ if (win32-msvc2008|win32-msvc2010|win32-msvc2012):!equals(TEMPLATE_PREFIX, "vc") #include(SleepLib2/sleeplib.pri) -SOURCES += main.cpp\ +SOURCES += \ + common_gui.cpp \ + daily.cpp \ + exportcsv.cpp \ + main.cpp \ + mainwindow.cpp \ + newprofile.cpp \ + oximetry.cpp \ + overview.cpp \ + preferencesdialog.cpp \ + profileselect.cpp \ + reports.cpp \ + sessionbar.cpp \ + summary.cpp \ + updateparser.cpp \ + UpdaterWindow.cpp \ + Graphs/gFlagsLine.cpp \ + Graphs/gFooBar.cpp \ + Graphs/gGraph.cpp \ + Graphs/gGraphView.cpp \ + Graphs/GLBuffer.cpp \ + Graphs/glcommon.cpp \ + Graphs/gLineChart.cpp \ + Graphs/gLineOverlay.cpp \ + Graphs/gSegmentChart.cpp \ + Graphs/gspacer.cpp \ + Graphs/gStatsLine.cpp \ + Graphs/gSummaryChart.cpp \ + Graphs/gVertexBuffer.cpp \ + Graphs/gXAxis.cpp \ + Graphs/gYAxis.cpp \ + Graphs/layer.cpp \ + SleepLib/calcs.cpp \ + SleepLib/common.cpp \ + SleepLib/day.cpp \ + SleepLib/event.cpp \ SleepLib/machine.cpp \ SleepLib/machine_loader.cpp \ SleepLib/preferences.cpp \ SleepLib/profiles.cpp \ - SleepLib/loader_plugins/cms50_loader.cpp \ - SleepLib/loader_plugins/prs1_loader.cpp \ - SleepLib/loader_plugins/zeo_loader.cpp \ - SleepLib/loader_plugins/resmed_loader.cpp \ - daily.cpp \ - oximetry.cpp \ - overview.cpp \ - mainwindow.cpp \ - SleepLib/event.cpp \ - SleepLib/session.cpp \ - SleepLib/day.cpp \ - Graphs/gLineChart.cpp \ - Graphs/gLineOverlay.cpp \ - Graphs/gFooBar.cpp \ - Graphs/gXAxis.cpp \ - Graphs/gYAxis.cpp \ - Graphs/gFlagsLine.cpp \ - Graphs/glcommon.cpp \ - Graphs/gSegmentChart.cpp \ - preferencesdialog.cpp \ - Graphs/gGraphView.cpp \ - Graphs/gStatsLine.cpp \ - Graphs/gSummaryChart.cpp \ SleepLib/schema.cpp \ - profileselect.cpp \ - newprofile.cpp \ - exportcsv.cpp \ - common_gui.cpp \ - SleepLib/loader_plugins/intellipap_loader.cpp \ - SleepLib/calcs.cpp \ - updateparser.cpp \ - UpdaterWindow.cpp \ - SleepLib/common.cpp \ + SleepLib/session.cpp \ + SleepLib/loader_plugins/cms50_loader.cpp \ SleepLib/loader_plugins/icon_loader.cpp \ + SleepLib/loader_plugins/intellipap_loader.cpp \ SleepLib/loader_plugins/mseries_loader.cpp \ - reports.cpp \ - summary.cpp \ - sessionbar.cpp \ - Graphs/gspacer.cpp \ - SleepLib/loader_plugins/somnopose_loader.cpp + SleepLib/loader_plugins/prs1_loader.cpp \ + SleepLib/loader_plugins/resmed_loader.cpp \ + SleepLib/loader_plugins/somnopose_loader.cpp \ + SleepLib/loader_plugins/zeo_loader.cpp HEADERS += \ + common_gui.h \ + daily.h \ + exportcsv.h \ + mainwindow.h \ + newprofile.h \ + oximetry.h \ + overview.h \ + preferencesdialog.h \ + profileselect.h \ + reports.h \ + sessionbar.h \ + summary.h \ + updateparser.h \ + UpdaterWindow.h \ + version.h \ + Graphs/gFlagsLine.h \ + Graphs/gFooBar.h \ + Graphs/gGraph.h \ + Graphs/gGraphView.h \ + Graphs/GLBuffer.h \ + Graphs/glcommon.h \ + Graphs/gLineChart.h \ + Graphs/gLineOverlay.h \ + Graphs/gSegmentChart.h\ + Graphs/gspacer.h \ + Graphs/gStatsLine.h \ + Graphs/gSummaryChart.h \ + Graphs/gVertexBuffer.h \ + Graphs/gXAxis.h \ + Graphs/gYAxis.h \ + Graphs/layer.h \ + SleepLib/calcs.h \ + SleepLib/common.h \ + SleepLib/day.h \ + SleepLib/event.h \ SleepLib/machine.h \ + SleepLib/machine_common.h \ SleepLib/machine_loader.h \ SleepLib/preferences.h \ SleepLib/profiles.h \ - SleepLib/loader_plugins/cms50_loader.h \ - SleepLib/loader_plugins/prs1_loader.h \ - SleepLib/loader_plugins/zeo_loader.h \ - oximetry.h \ - daily.h \ - overview.h \ - mainwindow.h \ - SleepLib/event.h \ - SleepLib/machine_common.h \ - SleepLib/session.h \ - SleepLib/day.h \ - Graphs/gLineChart.h \ - Graphs/gLineOverlay.h \ - Graphs/gFooBar.h \ - Graphs/gXAxis.h \ - Graphs/gYAxis.h \ - Graphs/gFlagsLine.h \ - Graphs/glcommon.h \ - Graphs/gSegmentChart.h\ - SleepLib/loader_plugins/resmed_loader.h \ - preferencesdialog.h \ - Graphs/gGraphView.h \ - Graphs/gStatsLine.h \ - Graphs/gSummaryChart.h \ SleepLib/schema.h \ - profileselect.h \ - newprofile.h \ - exportcsv.h \ - common_gui.h \ - SleepLib/loader_plugins/intellipap_loader.h \ - SleepLib/calcs.h \ - version.h \ - updateparser.h \ - UpdaterWindow.h \ - SleepLib/common.h \ + SleepLib/session.h \ + SleepLib/loader_plugins/cms50_loader.h \ SleepLib/loader_plugins/icon_loader.h \ + SleepLib/loader_plugins/intellipap_loader.h \ SleepLib/loader_plugins/mseries_loader.h \ - reports.h \ - summary.h \ - sessionbar.h \ - Graphs/gspacer.h \ - SleepLib/loader_plugins/somnopose_loader.h + SleepLib/loader_plugins/prs1_loader.h \ + SleepLib/loader_plugins/resmed_loader.h \ + SleepLib/loader_plugins/somnopose_loader.h \ + SleepLib/loader_plugins/zeo_loader.h - -FORMS += \ +FORMS += \ daily.ui \ overview.ui \ mainwindow.ui \