diff --git a/sleepyhead/Graphs/GLBuffer.cpp b/sleepyhead/Graphs/GLBuffer.cpp deleted file mode 100644 index c3171d75..00000000 --- a/sleepyhead/Graphs/GLBuffer.cpp +++ /dev/null @@ -1,289 +0,0 @@ -/* -*- 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 deleted file mode 100644 index b9d8c9a6..00000000 --- a/sleepyhead/Graphs/GLBuffer.h +++ /dev/null @@ -1,100 +0,0 @@ -/* -*- 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/gFlagsLine.cpp b/sleepyhead/Graphs/gFlagsLine.cpp index ad93da9a..1ea6b806 100644 --- a/sleepyhead/Graphs/gFlagsLine.cpp +++ b/sleepyhead/Graphs/gFlagsLine.cpp @@ -32,10 +32,6 @@ bool gFlagsLabelArea::mouseMoveEvent(QMouseEvent *event, gGraph *graph) gFlagsGroup::gFlagsGroup() { - addVertexBuffer(quads = new gVertexBuffer(512, GL_QUADS)); - addVertexBuffer(lines = new gVertexBuffer(20, GL_LINE_LOOP)); - quads->setAntiAlias(true); - lines->setAntiAlias(false); m_barh = 0; m_empty = true; } @@ -91,7 +87,7 @@ void gFlagsGroup::SetDay(Day *d) m_barh = 0; } -void gFlagsGroup::paint(gGraph &g, int left, int top, int width, int height) +void gFlagsGroup::paint(QPainter &painter, gGraph &g, int left, int top, int width, int height) { if (!m_visible) { return; } @@ -108,23 +104,19 @@ void gFlagsGroup::paint(gGraph &g, int left, int top, int width, int height) if (i & 1) { barcol = COLOR_ALT_BG1; } else { barcol = COLOR_ALT_BG2; } - quads->add(left, linetop, left, linetop + m_barh, left + width - 1, linetop + m_barh, - left + width - 1, linetop, barcol.rgba()); + painter.fillRect(left, linetop, width-1, m_barh, QBrush(barcol)); // Paint the actual flags lvisible[i]->m_rect = QRect(left, linetop, width, m_barh); - lvisible[i]->paint(g, left, linetop, width, m_barh); + lvisible[i]->paint(painter, g, left, linetop, width, m_barh); linetop += m_barh; } - gVertexBuffer *outlines = g.lines(); - outlines->add(left - 1, top, left - 1, top + height, COLOR_Outline.rgba()); - outlines->add(left - 1, top + height, left + width, top + height, COLOR_Outline.rgba()); - outlines->add(left + width, top + height, left + width, top, COLOR_Outline.rgba()); - outlines->add(left + width, top, left - 1, top, COLOR_Outline.rgba()); - - //lines->add(left-1, top, left-1, top+height); - //lines->add(left+width, top+height, left+width, top); + painter.setPen(COLOR_Outline); + painter.drawLine(left - 1, top, left - 1, top + height); + painter.drawLine(left - 1, top + height, left + width, top + height); + painter.drawLine(left + width, top + height, left + width, top); + painter.drawLine(left + width, top, left - 1, top); } bool gFlagsGroup::mouseMoveEvent(QMouseEvent *event, gGraph *graph) @@ -162,25 +154,16 @@ gFlagsLine::gFlagsLine(ChannelID code, QColor flag_color, QString label, bool al : Layer(code), m_label(label), m_always_visible(always_visible), m_flt(flt), m_flag_color(flag_color) { - addVertexBuffer(quads = new gVertexBuffer(2048, GL_QUADS)); - //addGLBuf(lines=new GLBuffer(flag_color,1024,GL_LINES)); - quads->setAntiAlias(true); - //lines->setAntiAlias(true); - //GetTextExtent(m_label,m_lx,m_ly); - //m_static.setText(m_label);; } gFlagsLine::~gFlagsLine() { - //delete lines; - //delete quads; } -void gFlagsLine::paint(gGraph &w, int left, int top, int width, int height) +void gFlagsLine::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { if (!m_visible) { return; } if (!m_day) { return; } - lines = w.lines(); double minx; double maxx; @@ -207,9 +190,7 @@ void gFlagsLine::paint(gGraph &w, int left, int top, int width, int height) float bartop = top + 2; float bottom = top + height - 2; - bool verts_exceeded = false; qint64 X, X2, L; - lines->setColor(schema::channel[m_code].defaultColor()); qint64 start; quint32 *tptr; @@ -220,6 +201,11 @@ void gFlagsLine::paint(gGraph &w, int left, int top, int width, int height) qint64 clockdrift = qint64(PROFILE.cpap->clockDrift()) * 1000L; qint64 drift = 0; + QVector vlines; + + QColor color=schema::channel[m_code].defaultColor(); + QBrush brush(color); + for (QList::iterator s = m_day->begin(); s != m_day->end(); s++) { if (!(*s)->enabled()) { continue; @@ -266,17 +252,6 @@ void gFlagsLine::paint(gGraph &w, int left, int top, int width, int height) // Draw Event Flag Bars /////////////////////////////////////////////////////////////////////////// - // Check bounds outside of loop is faster.. - // This will have to be reverted if multithreaded drawing is ever brought back - - int rem = lines->Max() - lines->cnt(); - - if ((np << 1) > rem) { - qDebug() << "gFlagsLine would overfill lines for" << schema::channel[m_code].label(); - np = rem >> 1; - verts_exceeded = true; - } - for (int i = 0; i < np; i++) { X = start + *tptr++; @@ -285,22 +260,12 @@ void gFlagsLine::paint(gGraph &w, int left, int top, int width, int height) } x1 = (X - minx) * xmult + left; - lines->add(x1, bartop, x1, bottom); - - //if (lines->full()) { verts_exceeded=true; break; } + vlines.append(QLine(x1, bartop, x1, bottom)); } } else if (m_flt == FT_Span) { /////////////////////////////////////////////////////////////////////////// // Draw Event Flag Spans /////////////////////////////////////////////////////////////////////////// - quads->setColor(m_flag_color); - int rem = quads->Max() - quads->cnt(); - - if ((np << 2) > rem) { - qDebug() << "gFlagsLine would overfill quads for" << schema::channel[m_code].label(); - np = rem >> 2; - verts_exceeded = true; - } for (; dptr < eptr; dptr++) { X = start + * tptr++; @@ -315,21 +280,14 @@ void gFlagsLine::paint(gGraph &w, int left, int top, int width, int height) x1 = double(X - minx) * xmult + left; x2 = double(X2 - minx) * xmult + left; - quads->add(x2, bartop, x1, bartop, x1, bottom, x2, bottom); - //if (quads->full()) { verts_exceeded=true; break; } - + painter.fillRect(x2, bartop, x1-x2, bottom-bartop, brush); } } - - if (verts_exceeded) { break; } } - - if (verts_exceeded) { break; } } - if (verts_exceeded) { - qWarning() << "maxverts exceeded in gFlagsLine::plot()"; - } + painter.setPen(color); + painter.drawLines(vlines); } bool gFlagsLine::mouseMoveEvent(QMouseEvent *event, gGraph *graph) diff --git a/sleepyhead/Graphs/gFlagsLine.h b/sleepyhead/Graphs/gFlagsLine.h index fc99614d..f3cb469e 100644 --- a/sleepyhead/Graphs/gFlagsLine.h +++ b/sleepyhead/Graphs/gFlagsLine.h @@ -24,8 +24,9 @@ class gFlagsLabelArea: public gSpacer { public: gFlagsLabelArea(gFlagsGroup *group); - virtual void paint(gGraph &w, int left, int top, int width, int height) { + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { Q_UNUSED(w) + Q_UNUSED(painter) Q_UNUSED(left) Q_UNUSED(top) Q_UNUSED(width) @@ -58,7 +59,7 @@ class gFlagsLine: public Layer virtual ~gFlagsLine(); //! \brief Drawing code to add the flags and span markers to the Vertex buffers. - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); //! \brief Returns true if should always show this flag, even if it's empty bool isAlwaysVisible() { return m_always_visible; } @@ -82,8 +83,6 @@ class gFlagsLine: public Layer int total_lines, line_num; FlagType m_flt; QColor m_flag_color; - gVertexBuffer *quads; - gVertexBuffer *lines; int m_lx, m_ly; }; @@ -99,7 +98,7 @@ class gFlagsGroup: public LayerGroup virtual ~gFlagsGroup(); //! Draw filled rectangles behind Event Flag's, and an outlines around them all, Calls the individual paint for each gFlagLine - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); //! Returns the first time represented by all gFlagLine layers, in milliseconds since epoch virtual qint64 Minx(); @@ -124,7 +123,6 @@ class gFlagsGroup: public LayerGroup protected: virtual bool mouseMoveEvent(QMouseEvent *event, gGraph *graph); - gVertexBuffer *quads, *lines; QVector lvisible; float m_barh; bool m_empty; diff --git a/sleepyhead/Graphs/gFooBar.cpp b/sleepyhead/Graphs/gFooBar.cpp index f852d9bf..9de6c96b 100644 --- a/sleepyhead/Graphs/gFooBar.cpp +++ b/sleepyhead/Graphs/gFooBar.cpp @@ -14,22 +14,16 @@ #include #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) { - addVertexBuffer(quads = new gVertexBuffer(20, GL_QUADS)); - addVertexBuffer(lines = new gVertexBuffer(20, GL_LINES)); - quads->forceAntiAlias(true); - lines->setAntiAlias(true); - lines->setSize(2); } gShadowArea::~gShadowArea() { } -void gShadowArea::paint(gGraph &w, int left, int top, int width, int height) +void gShadowArea::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { if (!m_visible) { return; } @@ -42,19 +36,16 @@ void gShadowArea::paint(gGraph &w, int left, int top, int width, int height) int start_px = left - 1; int end_px = left + width; - //float h=top; - double rmx = w.rmax_x - w.rmin_x; double px = ((1.0 / rmx) * (w.min_x - w.rmin_x)) * width; double py = ((1.0 / rmx) * (w.max_x - w.rmin_x)) * width; - quads->add(start_px, top, start_px, top + height, start_px + px, top + height, start_px + px, top, - m_shadow_color.rgba()); - quads->add(start_px + py, top, start_px + py, top + height, end_px, top + height, end_px, top, - m_shadow_color.rgba()); + painter.fillRect(start_px, top, px, height, QBrush(m_shadow_color)); + painter.fillRect(start_px + py, top, end_px-start_px-py, height, QBrush(m_shadow_color)); - lines->add(start_px + px, top, start_px + py, top, m_line_color.rgba()); - lines->add(start_px + px, top + height + 1, start_px + py, top + height + 1, m_line_color.rgba()); + painter.setPen(m_line_color); + painter.drawLine(start_px + px, top, start_px + py, top); + painter.drawLine(start_px + px, top + height + 1, start_px + py, top + height + 1); } gFooBar::gFooBar(int offset, QColor handle_color, QColor line_color) @@ -64,12 +55,13 @@ gFooBar::gFooBar(int offset, QColor handle_color, QColor line_color) gFooBar::~gFooBar() { } -void gFooBar::paint(gGraph &w, int left, int top, int width, int height) +void gFooBar::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { - Q_UNUSED(top); - Q_UNUSED(left); - Q_UNUSED(width); - Q_UNUSED(height); + Q_UNUSED(top) + Q_UNUSED(left) + Q_UNUSED(width) + Q_UNUSED(height) + Q_UNUSED(painter) if (!m_visible) { return; } diff --git a/sleepyhead/Graphs/gFooBar.h b/sleepyhead/Graphs/gFooBar.h index d424be46..a5469e1c 100644 --- a/sleepyhead/Graphs/gFooBar.h +++ b/sleepyhead/Graphs/gFooBar.h @@ -12,7 +12,6 @@ #ifndef GFOOBAR_H #define GFOOBAR_H -#include "Graphs/gVertexBuffer.h" #include "Graphs/layer.h" /*! \class gShadowArea @@ -23,12 +22,10 @@ class gShadowArea: public Layer public: gShadowArea(QColor shadow_color = QColor(40, 40, 40, 40), QColor line_color = Qt::blue); virtual ~gShadowArea(); - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); protected: QColor m_shadow_color; QColor m_line_color; - gVertexBuffer *quads; - gVertexBuffer *lines; }; /*! \class gFooBar @@ -41,7 +38,7 @@ class gFooBar: public Layer gFooBar(int offset = 10, QColor handle_color = QColor("orange"), QColor line_color = QColor("dark grey")); virtual ~gFooBar(); - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); static const int Margin = 15; protected: int m_offset; diff --git a/sleepyhead/Graphs/gGraph.cpp b/sleepyhead/Graphs/gGraph.cpp index 89711b3a..5c5dbc2b 100644 --- a/sleepyhead/Graphs/gGraph.cpp +++ b/sleepyhead/Graphs/gGraph.cpp @@ -26,8 +26,6 @@ QFont *defaultfont = nullptr; QFont *mediumfont = nullptr; QFont *bigfont = nullptr; QHash images; -bool fbo_unsupported = false; -QGLFramebufferObject *fbo = nullptr; static bool globalsInitialized = false; @@ -110,17 +108,6 @@ void DestroyGraphGlobals() 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; } @@ -164,8 +151,6 @@ gGraph::gGraph(gGraphView *graphview, QString title, QString units, int height, 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; @@ -179,7 +164,6 @@ gGraph::~gGraph() } m_layers.clear(); - delete m_quad; timer->stop(); disconnect(timer, 0, 0, 0); @@ -236,19 +220,18 @@ float gGraph::printScaleX() { return m_graphview->printScaleX(); } float gGraph::printScaleY() { return m_graphview->printScaleY(); } -void gGraph::drawGLBuf() -{ +//void gGraph::drawGLBuf() +//{ - float linesize = 1; +// float linesize = 1; - if (m_printing) { linesize = 4; } //ceil(m_graphview->printScaleY()); +// if (m_printing) { linesize = 4; } //ceil(m_graphview->printScaleY()); - for (int i = 0; i < m_layers.size(); i++) { - m_layers[i]->drawGLBuf(linesize); - } +// 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; @@ -272,13 +255,12 @@ 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) +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) +void gGraph::paint(QPainter &painter, int originX, int originY, int width, int height) { m_rect = QRect(originX, originY, width, height); @@ -295,55 +277,6 @@ void gGraph::paint(int originX, int originY, int width, int height) 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); @@ -354,17 +287,17 @@ void gGraph::paint(int originX, int originY, int width, int height) //GetTextExtent(title(),x,y,mediumfont); title_x = yh * 2; - - renderText(title(), marginLeft() + title_x + 4, originY + height / 2 - y / 2, 90, Qt::black, - mediumfont); + QString & txt = title(); + graphView()->AddTextQue(txt, 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); + painter.setPen(col); + painter.drawLine(0, originY, 0, originY + height); + painter.drawLine(left, originY, left, originY + height); #endif int tmp; @@ -392,19 +325,23 @@ void gGraph::paint(int originX, int originY, int width, int height) 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); + ll->paint(painter, *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); + QColor col = Qt::red; + painter.setPen(col); + painter.drawLine(originX + left - 1, originY, originX + left - 1, originY + height); #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); + ll->paint(painter, *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); + QColor col = Qt::red; + painter.setPen(col); + painter.drawLine(originX + width - right, originY, originX + width - right, originY + height); #endif } } @@ -421,28 +358,20 @@ void gGraph::paint(int originX, int originY, int width, int height) 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); + ll->paint(painter, *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); + ll->paint(painter, *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(); + painter.fillRect(originX + left, originY + top, width - right, height - bottom - top, QBrush(QColor(Qt::white))); } for (int i = 0; i < m_layers.size(); i++) { @@ -452,23 +381,22 @@ void gGraph::paint(int originX, int originY, int width, int height) 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); + ll->paint(painter, *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()); + painter.fillRect(originX + m_selection.x(), originY + top, m_selection.width(), height - bottom - top,QBrush(col)); +// 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) { @@ -524,29 +452,14 @@ QPixmap gGraph::renderPixmap(int w, int h, bool printing) float dpr = sg->devicePixelRatio(); sg->setDevicePixelRatio(1); -#ifdef Q_OS_WIN - if (pm.isNull()) { - pm = sg->renderPixmap(w, h, false); - } + QPixmap pm(w,h); - if (pm.isNull()) { // Works, but gives shader warnings - pm = QPixmap::fromImage(sg->pbRenderPixmap(w, h)); - } + QPainter painter(&pm); + painter.fillRect(0,0,w,h,QBrush(QColor(Qt::white))); + sg->renderGraphs(painter); + painter.end(); - 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(); @@ -1154,9 +1067,9 @@ void gGraph::ZoomX(double mult, int origin_px) //updateSelectionTime(max-min); } -void gGraph::DrawTextQue() +void gGraph::DrawTextQue(QPainter &painter) { - m_graphview->DrawTextQue(); + m_graphview->DrawTextQue(painter); } // margin recalcs.. @@ -1367,26 +1280,6 @@ 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; diff --git a/sleepyhead/Graphs/gGraph.h b/sleepyhead/Graphs/gGraph.h index 50aff602..3ae41dbc 100644 --- a/sleepyhead/Graphs/gGraph.h +++ b/sleepyhead/Graphs/gGraph.h @@ -6,16 +6,16 @@ * 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 #include "Graphs/layer.h" @@ -26,14 +26,10 @@ 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 @@ -41,7 +37,7 @@ const int mouse_movement_threshold = 6; */ class gGraph : public QObject { - Q_OBJECT; + Q_OBJECT public: friend class gGraphView; @@ -124,11 +120,11 @@ class gGraph : public QObject // 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 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; } + inline QString & title() { return m_title; } //! \brief Sets the Graph's (vertical) title void setTitle(const QString title) { m_title = title; } @@ -226,7 +222,7 @@ class gGraph : public QObject void setGroup(short group) { m_group = group; } //! \brief Forces the main gGraphView object to draw all Text Components - void DrawTextQue(); + void DrawTextQue(QPainter &painter); //! \brief Sends supplied day object to all Graph layers so they can precalculate stuff void setDay(Day *day); @@ -235,7 +231,7 @@ class gGraph : public QObject Day *day() { return m_day; } //! \brief The Layer, layout and title drawing code - virtual void paint(int originX, int originY, int width, int height); + virtual void paint(QPainter &painter, 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); @@ -255,33 +251,19 @@ class gGraph : public QObject } //! \brief Returns this graphs left margin - short marginLeft(); + inline short marginLeft() { return m_marginleft; } //! \brief Returns this graphs right margin - short marginRight(); + inline short marginRight() { return m_marginright; } //! \brief Returns this graphs top margin - short marginTop(); + inline short marginTop() { return m_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(); + inline short marginBottom() { return m_marginbottom; } 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(); @@ -344,7 +326,6 @@ class gGraph : public QObject short m_group; short m_lastx23; Day *m_day; - gVertexBuffer *m_quad; bool m_enforceMinY, m_enforceMaxY; bool m_showTitle; bool m_printing; diff --git a/sleepyhead/Graphs/gGraphView.cpp b/sleepyhead/Graphs/gGraphView.cpp index 3ec022d3..0b55d91f 100644 --- a/sleepyhead/Graphs/gGraphView.cpp +++ b/sleepyhead/Graphs/gGraphView.cpp @@ -18,6 +18,7 @@ #include #include #include +#include #if QT_VERSION >= QT_VERSION_CHECK(5,0,0) # include @@ -38,11 +39,6 @@ #include "SleepLib/profiles.h" extern MainWindow *mainwin; - -// for profiling purposes, a count of lines drawn in a single frame -int lines_drawn_this_frame = 0; -int quads_drawn_this_frame = 0; - extern QLabel *qstatus2; gToolTip::gToolTip(gGraphView *graphview) @@ -101,15 +97,13 @@ void gToolTip::cancel() timer->stop(); } -void gToolTip::paint() //actually paints it. +void gToolTip::paint(QPainter &painter) //actually paints it. { if (!m_visible) { return; } int x = m_pos.x(); int y = m_pos.y(); - QPainter painter(m_graphview); - QRect rect(x, y, 0, 0); painter.setFont(*defaultfont); @@ -138,9 +132,6 @@ void gToolTip::paint() //actually paints it. rect.setHeight(h); } - lines_drawn_this_frame += 4; - quads_drawn_this_frame += 1; - QBrush brush(QColor(255, 255, 128, 230)); brush.setStyle(Qt::SolidPattern); painter.setBrush(brush); @@ -152,8 +143,6 @@ void gToolTip::paint() //actually paints it. painter.setFont(*defaultfont); painter.drawText(rect, Qt::AlignCenter, m_text); - - painter.end(); } void gToolTip::timerDone() @@ -219,14 +208,15 @@ void gGraphView::queGraph(gGraph *g, int left, int top, int width, int height) dl_mutex.unlock(); #endif } + void gGraphView::trashGraphs() { - //for (int i=0;i= QT_VERSION_CHECK(5,0,0) - int dpr = devicePixelRatio(); -#endif - - for (int i = 0; i < m_textque_items; i++) { - TextQue &q = m_textque[i]; - painter.setBrush(q.color); - painter.setRenderHint(QPainter::TextAntialiasing, q.antialias); - QFont font = *q.font; -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - int fs = font.pointSize(); - - if (fs > 0) { - font.setPointSize(fs * dpr); - } else { - font.setPixelSize(font.pixelSize()*dpr); - } - -#endif - painter.setFont(font); - - if (q.angle == 0) { // normal text - -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - painter.drawText(q.x * dpr, q.y * dpr, q.text); -#else - painter.drawText(q.x, q.y, q.text); -#endif - } else { // rotated text - w = painter.fontMetrics().width(q.text); - h = painter.fontMetrics().xHeight() + 2; - - painter.translate(q.x, q.y); - painter.rotate(-q.angle); - painter.drawText(floor(-w / 2.0), floor(-h / 2.0), q.text); - painter.rotate(+q.angle); - painter.translate(-q.x, -q.y); - } - - q.text.clear(); - } - - m_textque_items = 0; -} - -QImage gGraphView::pbRenderPixmap(int w, int h) -{ - QImage pm = QImage(); - QGLFormat pbufferFormat = format(); - QGLPixelBuffer pbuffer(w, h, pbufferFormat, this); - - if (pbuffer.isValid()) { - pbuffer.makeCurrent(); - initializeGL(); - resizeGL(w, h); - glClearColor(255, 255, 255, 255); - glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); - - renderGraphs(); - glFlush(); - pm = pbuffer.toImage(); - pbuffer.doneCurrent(); - QPainter painter(&pm); - DrawTextQue(painter); - painter.end(); - } - - return pm; -} - -QImage gGraphView::fboRenderPixmap(int w, int h) -{ - QImage pm = QImage(); - - if (fbo_unsupported) { - return pm; - } - -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - float dpr = devicePixelRatio(); - w *= dpr; - h *= dpr; -#endif - - if ((w > max_fbo_width) || (h > max_fbo_height)) { - qWarning() << - "gGraphView::fboRenderPixmap called with dimensiopns exceeding maximum frame buffer object size"; - return pm; - } - - if (!fbo) { - fbo = new QGLFramebufferObject(max_fbo_width, max_fbo_height, - QGLFramebufferObject::Depth); //NoAttachment); - } - - if (fbo && fbo->isValid()) { - makeCurrent(); - - if (fbo->bind()) { - initializeGL(); - resizeGL(w, h); - glClearColor(255, 255, 255, 255); - glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); - - renderGraphs(); // render graphs sans text - glFlush(); - fbo->release(); - pm = fbo->toImage().copy(0, max_fbo_height - h, w, h); - doneCurrent(); - - QPainter painter(&pm); - DrawTextQue(painter); //Just use this on mac to - painter.end(); - - } - } else { - delete fbo; - fbo = nullptr; - fbo_unsupported = true; - } - - return pm; -} - gGraphView::gGraphView(QWidget *parent, gGraphView *shared) - : QGLWidget(QGLFormat(QGL::Rgba | QGL::DoubleBuffer | QGL::NoOverlay), parent, shared), + : QGLWidget(QGLFormat(QGL::Rgba | QGL::DoubleBuffer), parent, shared), m_offsetY(0), m_offsetX(0), m_scaleY(1.0), m_scrollbar(nullptr) { m_shared = shared; m_sizer_index = m_graph_index = 0; - m_textque_items = 0; m_button_down = m_graph_dragging = m_sizer_dragging = false; m_lastypos = m_lastxpos = 0; m_horiz_travel = 0; - pixmap_cache_size = 0; m_minx = m_maxx = 0; m_day = nullptr; m_selected_graph = nullptr; @@ -409,19 +269,6 @@ gGraphView::gGraphView(QWidget *parent, gGraphView *shared) //gt->start(); }*/ - lines = new gVertexBuffer(100000, GL_LINES); // big fat shared line list - backlines = new gVertexBuffer(10000, GL_LINES); // big fat shared line list - quads = new gVertexBuffer(1024, GL_QUADS); // big fat shared line list - quads->forceAntiAlias(true); - frontlines = new gVertexBuffer(20000, GL_LINES); - - //vlines=new gVertexBuffer(20000,GL_LINES); - - //stippled->setSize(1.5); - //stippled->forceAntiAlias(false); - //lines->setSize(1.5); - //backlines->setSize(1.5); - setFocusPolicy(Qt::StrongFocus); m_showsplitter = true; timer = new QTimer(this); @@ -444,6 +291,7 @@ gGraphView::gGraphView(QWidget *parent, gGraphView *shared) m_blockUpdates = false; use_pixmap_cache = true; + // pixmapcache.setCacheLimit(10240*2); #if QT_VERSION >= QT_VERSION_CHECK(5,0,0) m_dpr = this->windowHandle()->devicePixelRatio(); #else @@ -474,22 +322,9 @@ gGraphView::~gGraphView() m_graphs[i]=NULL; } - QHash::iterator it; - - for (it = pixmap_cache.begin(); it != pixmap_cache.end(); it++) { - delete(*it); - } - - pixmap_cache.clear(); - delete m_tooltip; m_graphs.clear(); - delete frontlines; - delete lines; - delete backlines; - delete quads; - if (m_scrollbar) { this->disconnect(m_scrollbar, SIGNAL(sliderMoved(int)), 0, 0); } @@ -501,184 +336,121 @@ gGraphView::~gGraphView() bool gGraphView::usePixmapCache() { //use_pixmap_cache is an overide setting - return use_pixmap_cache & PROFILE.appearance->usePixmapCaching(); + return PROFILE.appearance->usePixmapCaching(); } - -void gGraphView::DrawTextQue() +#define CACHE_DRAWTEXT +#ifndef CACHE_DRAWTEXT +// Render all qued text via QPainter method +void gGraphView::DrawTextQue(QPainter &painter) { - const qint64 expire_after_ms = 4000; // expire string pixmaps after this many milliseconds - //const qint64 under_limit_cache_bonus=30000; // If under the limit, give a bonus to the millisecond timeout. - const qint32 max_pixmap_cache = 4 * - 1048576; // Maximum size of pixmap cache (it can grow over this, but only temporarily) - - quint64 ti = 0, exptime = 0; int w, h; - QHash::iterator it; - QPainter painter; - // Purge the Pixmap cache of any old text strings - if (usePixmapCache()) { - // Current time in milliseconds since epoch. - ti = QDateTime::currentDateTime().toMSecsSinceEpoch(); + // not sure if global antialiasing would be better.. + //painter.setRenderHint(QPainter::TextAntialiasing, PROFILE.appearance->antiAliasing()); + int m_textque_items = m_textque.size(); + for (int i = 0; i < m_textque_items; ++i) { + TextQue &q = m_textque[i]; + painter.setPen(q.color); + painter.setRenderHint(QPainter::TextAntialiasing, q.antialias); + QFont font = *q.font; + painter.setFont(font); - if (pixmap_cache_size > - max_pixmap_cache) { // comment this if block out to only cleanup when past the maximum cache size - // Expire any strings not used - QList expire; + if (q.angle == 0) { // normal text - exptime = expire_after_ms; + painter.drawText(q.x, q.y, q.text); + } else { // rotated text + w = painter.fontMetrics().width(q.text); + h = painter.fontMetrics().xHeight() + 2; - // Uncomment the next line to allow better use of pixmap cache memory - //if (pixmap_cache_size < max_pixmap_cache) exptime+=under_limit_cache_bonus; - - for (it = pixmap_cache.begin(); it != pixmap_cache.end(); it++) { - if ((*it)->last_used < (ti - exptime)) { - expire.push_back(it.key()); - } - } - - - // TODO: Force expiry if over an upper memory threshold.. doesn't appear to be necessary. - - for (int i = 0; i < expire.count(); i++) { - const QString key = expire.at(i); - // unbind the texture - myPixmapCache *pc = pixmap_cache[key]; - deleteTexture(pc->textureID); - QImage &pm = pc->image; - pixmap_cache_size -= pm.width() * pm.height() * (pm.depth() / 8); - // free the pixmap - //delete pc->pixmap; - - // free the myPixmapCache object - delete pc; - - // pull the dead record from the cache. - pixmap_cache.remove(key); - } + painter.translate(q.x, q.y); + painter.rotate(-q.angle); + painter.drawText(floor(-w / 2.0), floor(-h / 2.0), q.text); + painter.rotate(+q.angle); + painter.translate(-q.x, -q.y); } + strings_drawn_this_frame++; + q.text.clear(); } - //glPushAttrib(GL_COLOR_BUFFER_BIT); - painter.begin(this); - - int buf = 4; - -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - float dpr = devicePixelRatio(); - buf *= dpr; -#endif + m_textque.clear(); +} +#else +// Render graphs with QPainter or pixmap caching, depending on preferences +void gGraphView::DrawTextQue(QPainter &painter) +{ // process the text drawing queue - for (int i = 0; i < m_textque_items; i++) { + int m_textque_items = m_textque.size(); + + int h,w; + + + for (int i = 0; i < m_textque_items; ++i) { TextQue &q = m_textque[i]; // can do antialiased text via texture cache fine on mac if (usePixmapCache()) { // Generate the pixmap cache "key" - QString hstr = QString("%4:%5:%6%7").arg(q.text).arg(q.color.name()).arg(q.font->key()).arg( - q.antialias); + QString hstr = QString("%1:%2:%3"). + arg(q.text). + arg(q.color.name()). + arg(q.font->pointSize()); - QImage pm; - - it = pixmap_cache.find(hstr); - myPixmapCache *pc = nullptr; - - if (it != pixmap_cache.end()) { - pc = (*it); - - } else { - // not found.. create the image and store it in a cache - - //This is much slower than other text rendering methods, but caching more than makes up for the speed decrease. - pc = new myPixmapCache; - pc->last_used = ti; // set the last_used value. + QPixmap pm; + const int buf = 8; + if (!QPixmapCache::find(hstr, &pm)) { QFontMetrics fm(*q.font); - QRect rect = fm.boundingRect(q.text); + QRect rect=fm.tightBoundingRect(q.text); w = fm.width(q.text); - h = fm.height(); - -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - w *= dpr; - h *= dpr; -#endif - - rect.setWidth(w); - rect.setHeight(h); - - pm = QImage(w + buf, h + buf, QImage::Format_ARGB32_Premultiplied); + h = fm.height()+buf; + pm=QPixmap(w, h); pm.fill(Qt::transparent); QPainter imgpainter(&pm); - QBrush b(q.color); - imgpainter.setBrush(b); + imgpainter.setPen(q.color); - QFont font = *q.font; -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - int fs = font.pointSize(); - - if (fs > 0) { - font.setPointSize(fs * dpr); - } else { - font.setPixelSize(font.pixelSize()*dpr); - } - -#endif - imgpainter.setFont(font); + imgpainter.setFont(*q.font); imgpainter.setRenderHint(QPainter::TextAntialiasing, q.antialias); - imgpainter.drawText(buf / 2, h, q.text); + imgpainter.drawText(0, h-buf, q.text); imgpainter.end(); - pc->image = pm; - pixmap_cache_size += pm.width() * pm.height() * (pm.depth() / 8); - pixmap_cache[hstr] = pc; - + QPixmapCache::insert(hstr, pm); + strings_drawn_this_frame++; + } else { + //cached + strings_cached_this_frame++; } - if (pc) { - pc->last_used = ti; - int h = pc->image.height(); - int w = pc->image.width(); + h = pm.height(); + w = pm.width(); + if (q.angle != 0) { + float xxx = q.x - h - (h / 2); + float yyy = q.y + w / 2; // + buf / 2; -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - h /= dpr; - w /= dpr; -#endif + xxx+=4; + yyy+=4; - if (q.angle != 0) { - float xxx = q.x - h - (h / 2); - float yyy = q.y + w / 2 + buf / 2; - - painter.translate(xxx, yyy); - painter.rotate(-q.angle); - painter.drawImage(QRect(0, h / 2, w, h), pc->image, pc->image.rect()); - painter.rotate(+q.angle); - painter.translate(-xxx, -yyy); - } else { - painter.drawImage(QRect(q.x - buf / 2, q.y - h + buf / 2, w, h), pc->image, pc->image.rect()); - } + painter.translate(xxx, yyy); + painter.rotate(-q.angle); + painter.drawPixmap(QRect(0, h / 2, w, h), pm); + painter.rotate(+q.angle); + painter.translate(-xxx, -yyy); + } else { + painter.drawPixmap(QRect(q.x - buf / 2, q.y - h + buf, w, h), pm); } } else { // Just draw the fonts.. - QBrush b(q.color); - painter.setBrush(b); + painter.setPen(QColor(q.color)); painter.setFont(*q.font); if (q.angle == 0) { - // ********************************************************* - // Holy crap this is slow - // The following line is responsible for 77% of drawing time - // ********************************************************* - painter.drawText(q.x, q.y, q.text); } else { - QBrush b(q.color); - painter.setBrush(b); painter.setFont(*q.font); w = painter.fontMetrics().width(q.text); @@ -690,42 +462,35 @@ void gGraphView::DrawTextQue() painter.rotate(+q.angle); painter.translate(-q.x, -q.y); } + strings_drawn_this_frame++; + } - q.text.clear(); + //q.text.clear(); //q.text.squeeze(); } - if (!usePixmapCache()) { - painter.end(); - } - - //qDebug() << "rendered" << m_textque_items << "text items"; - m_textque_items = 0; + m_textque.clear(); } +#endif -void gGraphView::AddTextQue(QString &text, short x, short y, float angle, QColor color, +void gGraphView::AddTextQue(const QString &text, short x, short y, float angle, QColor color, QFont *font, bool antialias) { #ifdef ENABLED_THREADED_DRAWING text_mutex.lock(); #endif - - if (m_textque_items >= textque_max) { - DrawTextQue(); - } - - TextQue &q = m_textque[m_textque_items++]; + m_textque.append(TextQue(x,y,angle,text,color,font,antialias)); #ifdef ENABLED_THREADED_DRAWING text_mutex.unlock(); #endif - q.text = text; - q.x = x; - q.y = y; - q.angle = angle; - q.color = color; - q.font = font; - q.antialias = antialias; +// q.text = text; +// q.x = x; +// q.y = y; +// q.angle = angle; +// q.color = color; +// q.font = font; +// q.antialias = antialias; } void gGraphView::addGraph(gGraph *g, short group) @@ -748,6 +513,8 @@ void gGraphView::addGraph(gGraph *g, short group) // updateScrollBar(); } } + +// Calculate total height of all graphs including spacers float gGraphView::totalHeight() { float th = 0; @@ -760,6 +527,7 @@ float gGraphView::totalHeight() return ceil(th); } + float gGraphView::findTop(gGraph *graph) { float th = -m_offsetY; @@ -772,9 +540,9 @@ float gGraphView::findTop(gGraph *graph) th += m_graphs[i]->height() * m_scaleY + graphSpacer; } - //th-=m_offsetY; return ceil(th); } + float gGraphView::scaleHeight() { float th = 0; @@ -787,6 +555,7 @@ float gGraphView::scaleHeight() return ceil(th); } + void gGraphView::resizeEvent(QResizeEvent *e) { QGLWidget::resizeEvent(e); // This ques a redraw event.. @@ -797,6 +566,7 @@ void gGraphView::resizeEvent(QResizeEvent *e) m_graphs[i]->resize(e->size().width(), m_graphs[i]->height()*m_scaleY); } } + void gGraphView::scrollbarValueChanged(int val) { //qDebug() << "Scrollbar Changed" << val; @@ -847,6 +617,7 @@ void gGraphView::GetRXBounds(qint64 &st, qint64 &et) st = g->rmin_x; et = g->rmax_x; } + void gGraphView::ResetBounds(bool refresh) //short group) { Q_UNUSED(refresh) @@ -901,6 +672,7 @@ void gGraphView::ResetBounds(bool refresh) //short group) updateScale(); } + void gGraphView::GetXBounds(qint64 &st, qint64 &et) { st = m_minx; @@ -938,6 +710,7 @@ void gGraphView::SetXBounds(qint64 minx, qint64 maxx, short group, bool refresh) if (refresh) { redraw(); } } + void gGraphView::updateScale() { float th = totalHeight(); // height of all graphs @@ -995,9 +768,9 @@ void gGraphView::initializeGL() { setAutoFillBackground(false); setAutoBufferSwap(false); - glDisable(GL_LIGHTING); - glDisable(GL_DEPTH_TEST); - glDisable(GL_TEXTURE_2D); +// glDisable(GL_LIGHTING); +// glDisable(GL_DEPTH_TEST); +// glDisable(GL_TEXTURE_2D); if (cubeimg.size() > 0) { cubetex = bindTexture(*cubeimg[0]); @@ -1008,205 +781,214 @@ void gGraphView::initializeGL() // texid[i]=bindTexture(*images[i]); // } - glBindTexture(GL_TEXTURE_2D, 0); +// glBindTexture(GL_TEXTURE_2D, 0); // glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR); } -void gGraphView::resizeGL(int w, int h) -{ - glViewport(0, 0, w, h); - glMatrixMode(GL_PROJECTION); - glLoadIdentity(); +//void gGraphView::resizeGL(int w, int h) +//{ -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - float dpr = devicePixelRatio(); - glOrtho(0, w / dpr, h / dpr, 0, -1, 1); -#else - glOrtho(0, w, h, 0, -1, 1); -#endif - glMatrixMode(GL_MODELVIEW); - glLoadIdentity(); -} +//#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) +// float dpr = devicePixelRatio(); +//#else +// float dpr = 1; +//#endif -void gGraphView::renderCube(float alpha) +// glViewport(0, 0, w / dpr, h / dpr); +// glMatrixMode(GL_PROJECTION); +// glLoadIdentity(); + +// glOrtho(0, w / dpr, h / dpr, 0, -1, 1); + +// glMatrixMode(GL_MODELVIEW); +// glLoadIdentity(); +//} + +void gGraphView::renderCube(QPainter &painter, float alpha) { + if (cubeimg.size() == 0) { return; } // glPushMatrix(); float w = width(); float h = height(); + #if QT_VERSION >= QT_VERSION_CHECK(5,0,0) float dpr = devicePixelRatio(); w *= dpr; h *= dpr; #endif - glViewport(0, 0, w, h); - glMatrixMode(GL_PROJECTION); - glLoadIdentity(); - gluPerspective(45.0f, (GLfloat)w / (GLfloat)h, 0.1f, 100.0f); - glMatrixMode(GL_MODELVIEW); - glLoadIdentity(); + painter.beginNativePainting(); - /*glShadeModel(GL_SMOOTH); - glClearColor(0.0f, 0.0f, 0.0f, 0.5f); - glClearDepth(1.0f); */ - glEnable(GL_DEPTH_TEST); - glDepthFunc(GL_LEQUAL); - glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST); +// glViewport(0, 0, w, h); +// glMatrixMode(GL_PROJECTION); +// glLoadIdentity(); +// gluPerspective(45.0f, (GLfloat)w / (GLfloat)h, 0.1f, 100.0f); +// glMatrixMode(GL_MODELVIEW); +// glLoadIdentity(); - // This code has been shamelessly pinched of the interwebs.. - // When I'm feeling more energetic, I'll change it to a textured sheep or something. - static float rotqube = 0; +// /*glShadeModel(GL_SMOOTH); +// glClearColor(0.0f, 0.0f, 0.0f, 0.5f); +// glClearDepth(1.0f); */ +// glEnable(GL_DEPTH_TEST); +// glDepthFunc(GL_LEQUAL); +// glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST); - static float xpos = 0, ypos = 7; +// // This code has been shamelessly pinched of the interwebs.. +// // When I'm feeling more energetic, I'll change it to a textured sheep or something. +// static float rotqube = 0; - glLoadIdentity(); +// static float xpos = 0, ypos = 7; - glAlphaFunc(GL_GREATER, 0.1F); - glEnable(GL_ALPHA_TEST); - glEnable(GL_CULL_FACE); - glDisable(GL_COLOR_MATERIAL); +// glLoadIdentity(); - //int imgcount=cubeimg.size(); +// glAlphaFunc(GL_GREATER, 0.1F); +// glEnable(GL_ALPHA_TEST); +// glEnable(GL_CULL_FACE); +// glDisable(GL_COLOR_MATERIAL); - glEnable(GL_BLEND); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); +// //int imgcount=cubeimg.size(); - double xx = 0.0, yy = 0.0; +// glEnable(GL_BLEND); +// glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - // set this to 0 to make the cube stay in the center of the screen - if (1) { - xx = sin(M_PI / 180.0 * xpos) * 2; // ((4.0/width()) * m_mouse.rx())-2.0; - yy = cos(M_PI / 180.0 * ypos) * 2; //2-((4.0/height()) * m_mouse.ry()); - xpos += 1; - ypos += 1.32F; +// double xx = 0.0, yy = 0.0; - if (xpos > 360) { xpos -= 360.0F; } +// // set this to 0 to make the cube stay in the center of the screen +// if (1) { +// xx = sin(M_PI / 180.0 * xpos) * 2; // ((4.0/width()) * m_mouse.rx())-2.0; +// yy = cos(M_PI / 180.0 * ypos) * 2; //2-((4.0/height()) * m_mouse.ry()); +// xpos += 1; +// ypos += 1.32F; - if (ypos > 360) { ypos -= 360.0F; } - } +// if (xpos > 360) { xpos -= 360.0F; } + +// if (ypos > 360) { ypos -= 360.0F; } +// } - //m_mouse.x(); - glTranslatef(xx, 0.0f, -7.0f + yy); - glRotatef(rotqube, 0.0f, 1.0f, 0.0f); - glRotatef(rotqube, 1.0f, 1.0f, 1.0f); +// //m_mouse.x(); +// glTranslatef(xx, 0.0f, -7.0f + yy); +// glRotatef(rotqube, 0.0f, 1.0f, 0.0f); +// glRotatef(rotqube, 1.0f, 1.0f, 1.0f); - int i = 0; - glEnable(GL_TEXTURE_2D); - cubetex = bindTexture(*cubeimg[0]); +// int i = 0; +// glEnable(GL_TEXTURE_2D); +// cubetex = bindTexture(*cubeimg[0]); - //glBindTexture(GL_TEXTURE_2D, cubetex); //texid[i % imgcount]); - i++; - glColor4f(1, 1, 1, alpha); +// //glBindTexture(GL_TEXTURE_2D, cubetex); //texid[i % imgcount]); +// i++; +// glColor4f(1, 1, 1, alpha); - glBegin(GL_QUADS); - glTexCoord2f(0.0f, 0.0f); - glVertex3f(-1.0f, -1.0f, 1.0f); // Bottom Left Of The Texture and Quad - glTexCoord2f(1.0f, 0.0f); - glVertex3f(1.0f, -1.0f, 1.0f); // Bottom Right Of The Texture and Quad - glTexCoord2f(1.0f, 1.0f); - glVertex3f(1.0f, 1.0f, 1.0f); // Top Right Of The Texture and Quad - glTexCoord2f(0.0f, 1.0f); - glVertex3f(-1.0f, 1.0f, 1.0f); // Top Left Of The Texture and Quad - glEnd(); - // Back Face - //bindTexture(*cubeimg[i % imgcount]); - //glBindTexture(GL_TEXTURE_2D, texid[i % imgcount]); - i++; - glBegin(GL_QUADS); - glTexCoord2f(1.0f, 0.0f); - glVertex3f(-1.0f, -1.0f, -1.0f); // Bottom Right Of The Texture and Quad - glTexCoord2f(1.0f, 1.0f); - glVertex3f(-1.0f, 1.0f, -1.0f); // Top Right Of The Texture and Quad - glTexCoord2f(0.0f, 1.0f); - glVertex3f(1.0f, 1.0f, -1.0f); // Top Left Of The Texture and Quad - glTexCoord2f(0.0f, 0.0f); - glVertex3f(1.0f, -1.0f, -1.0f); // Bottom Left Of The Texture and Quad - glEnd(); - // Top Face - //bindTexture(*cubeimg[i % imgcount]); - // glBindTexture(GL_TEXTURE_2D, texid[i % imgcount]); - i++; - glBegin(GL_QUADS); - glTexCoord2f(0.0f, 1.0f); - glVertex3f(-1.0f, 1.0f, -1.0f); // Top Left Of The Texture and Quad - glTexCoord2f(0.0f, 0.0f); - glVertex3f(-1.0f, 1.0f, 1.0f); // Bottom Left Of The Texture and Quad - glTexCoord2f(1.0f, 0.0f); - glVertex3f(1.0f, 1.0f, 1.0f); // Bottom Right Of The Texture and Quad - glTexCoord2f(1.0f, 1.0f); - glVertex3f(1.0f, 1.0f, -1.0f); // Top Right Of The Texture and Quad - glEnd(); - // Bottom Face - //bindTexture(*cubeimg[i % imgcount]); - //glBindTexture(GL_TEXTURE_2D, texid[i % imgcount]); - i++; - glBegin(GL_QUADS); - glTexCoord2f(1.0f, 1.0f); - glVertex3f(-1.0f, -1.0f, -1.0f); // Top Right Of The Texture and Quad - glTexCoord2f(0.0f, 1.0f); - glVertex3f(1.0f, -1.0f, -1.0f); // Top Left Of The Texture and Quad - glTexCoord2f(0.0f, 0.0f); - glVertex3f(1.0f, -1.0f, 1.0f); // Bottom Left Of The Texture and Quad - glTexCoord2f(1.0f, 0.0f); - glVertex3f(-1.0f, -1.0f, 1.0f); // Bottom Right Of The Texture and Quad - glEnd(); - // Right face - //bindTexture(*cubeimg[i % imgcount]); - // glBindTexture(GL_TEXTURE_2D, texid[i % imgcount]); - i++; - glBegin(GL_QUADS); - glTexCoord2f(1.0f, 0.0f); - glVertex3f(1.0f, -1.0f, -1.0f); // Bottom Right Of The Texture and Quad - glTexCoord2f(1.0f, 1.0f); - glVertex3f(1.0f, 1.0f, -1.0f); // Top Right Of The Texture and Quad - glTexCoord2f(0.0f, 1.0f); - glVertex3f(1.0f, 1.0f, 1.0f); // Top Left Of The Texture and Quad - glTexCoord2f(0.0f, 0.0f); - glVertex3f(1.0f, -1.0f, 1.0f); // Bottom Left Of The Texture and Quad - glEnd(); - // Left Face - //GLuint tex=bindTexture(*images["mask"]); - //glBindTexture(GL_TEXTURE_2D, tex); - i++; - glBegin(GL_QUADS); - glTexCoord2f(0.0f, 0.0f); - glVertex3f(-1.0f, -1.0f, -1.0f); // Bottom Left Of The Texture and Quad - glTexCoord2f(1.0f, 0.0f); - glVertex3f(-1.0f, -1.0f, 1.0f); // Bottom Right Of The Texture and Quad - glTexCoord2f(1.0f, 1.0f); - glVertex3f(-1.0f, 1.0f, 1.0f); // Top Right Of The Texture and Quad - glTexCoord2f(0.0f, 1.0f); - glVertex3f(-1.0f, 1.0f, -1.0f); // Top Left Of The Texture and Quad - glEnd(); +// glBegin(GL_QUADS); +// glTexCoord2f(0.0f, 0.0f); +// glVertex3f(-1.0f, -1.0f, 1.0f); // Bottom Left Of The Texture and Quad +// glTexCoord2f(1.0f, 0.0f); +// glVertex3f(1.0f, -1.0f, 1.0f); // Bottom Right Of The Texture and Quad +// glTexCoord2f(1.0f, 1.0f); +// glVertex3f(1.0f, 1.0f, 1.0f); // Top Right Of The Texture and Quad +// glTexCoord2f(0.0f, 1.0f); +// glVertex3f(-1.0f, 1.0f, 1.0f); // Top Left Of The Texture and Quad +// glEnd(); +// // Back Face +// //bindTexture(*cubeimg[i % imgcount]); +// //glBindTexture(GL_TEXTURE_2D, texid[i % imgcount]); +// i++; +// glBegin(GL_QUADS); +// glTexCoord2f(1.0f, 0.0f); +// glVertex3f(-1.0f, -1.0f, -1.0f); // Bottom Right Of The Texture and Quad +// glTexCoord2f(1.0f, 1.0f); +// glVertex3f(-1.0f, 1.0f, -1.0f); // Top Right Of The Texture and Quad +// glTexCoord2f(0.0f, 1.0f); +// glVertex3f(1.0f, 1.0f, -1.0f); // Top Left Of The Texture and Quad +// glTexCoord2f(0.0f, 0.0f); +// glVertex3f(1.0f, -1.0f, -1.0f); // Bottom Left Of The Texture and Quad +// glEnd(); +// // Top Face +// //bindTexture(*cubeimg[i % imgcount]); +// // glBindTexture(GL_TEXTURE_2D, texid[i % imgcount]); +// i++; +// glBegin(GL_QUADS); +// glTexCoord2f(0.0f, 1.0f); +// glVertex3f(-1.0f, 1.0f, -1.0f); // Top Left Of The Texture and Quad +// glTexCoord2f(0.0f, 0.0f); +// glVertex3f(-1.0f, 1.0f, 1.0f); // Bottom Left Of The Texture and Quad +// glTexCoord2f(1.0f, 0.0f); +// glVertex3f(1.0f, 1.0f, 1.0f); // Bottom Right Of The Texture and Quad +// glTexCoord2f(1.0f, 1.0f); +// glVertex3f(1.0f, 1.0f, -1.0f); // Top Right Of The Texture and Quad +// glEnd(); +// // Bottom Face +// //bindTexture(*cubeimg[i % imgcount]); +// //glBindTexture(GL_TEXTURE_2D, texid[i % imgcount]); +// i++; +// glBegin(GL_QUADS); +// glTexCoord2f(1.0f, 1.0f); +// glVertex3f(-1.0f, -1.0f, -1.0f); // Top Right Of The Texture and Quad +// glTexCoord2f(0.0f, 1.0f); +// glVertex3f(1.0f, -1.0f, -1.0f); // Top Left Of The Texture and Quad +// glTexCoord2f(0.0f, 0.0f); +// glVertex3f(1.0f, -1.0f, 1.0f); // Bottom Left Of The Texture and Quad +// glTexCoord2f(1.0f, 0.0f); +// glVertex3f(-1.0f, -1.0f, 1.0f); // Bottom Right Of The Texture and Quad +// glEnd(); +// // Right face +// //bindTexture(*cubeimg[i % imgcount]); +// // glBindTexture(GL_TEXTURE_2D, texid[i % imgcount]); +// i++; +// glBegin(GL_QUADS); +// glTexCoord2f(1.0f, 0.0f); +// glVertex3f(1.0f, -1.0f, -1.0f); // Bottom Right Of The Texture and Quad +// glTexCoord2f(1.0f, 1.0f); +// glVertex3f(1.0f, 1.0f, -1.0f); // Top Right Of The Texture and Quad +// glTexCoord2f(0.0f, 1.0f); +// glVertex3f(1.0f, 1.0f, 1.0f); // Top Left Of The Texture and Quad +// glTexCoord2f(0.0f, 0.0f); +// glVertex3f(1.0f, -1.0f, 1.0f); // Bottom Left Of The Texture and Quad +// glEnd(); +// // Left Face +// //GLuint tex=bindTexture(*images["mask"]); +// //glBindTexture(GL_TEXTURE_2D, tex); +// i++; +// glBegin(GL_QUADS); +// glTexCoord2f(0.0f, 0.0f); +// glVertex3f(-1.0f, -1.0f, -1.0f); // Bottom Left Of The Texture and Quad +// glTexCoord2f(1.0f, 0.0f); +// glVertex3f(-1.0f, -1.0f, 1.0f); // Bottom Right Of The Texture and Quad +// glTexCoord2f(1.0f, 1.0f); +// glVertex3f(-1.0f, 1.0f, 1.0f); // Top Right Of The Texture and Quad +// glTexCoord2f(0.0f, 1.0f); +// glVertex3f(-1.0f, 1.0f, -1.0f); // Top Left Of The Texture and Quad +// glEnd(); - glDisable(GL_BLEND); - glBindTexture(GL_TEXTURE_2D, 0); +// glDisable(GL_BLEND); +// glBindTexture(GL_TEXTURE_2D, 0); - glDisable(GL_ALPHA_TEST); - glDisable(GL_TEXTURE_2D); - glDisable(GL_CULL_FACE); +// glDisable(GL_ALPHA_TEST); +// glDisable(GL_TEXTURE_2D); +// glDisable(GL_CULL_FACE); - glDisable(GL_DEPTH_TEST); +// glDisable(GL_DEPTH_TEST); - rotqube += 0.9f; +// rotqube += 0.9f; - // Restore boring 2D reality.. - glViewport(0, 0, w, h); - glMatrixMode(GL_PROJECTION); - glLoadIdentity(); +// // Restore boring 2D reality.. +// glViewport(0, 0, w, h); +// glMatrixMode(GL_PROJECTION); +// glLoadIdentity(); - glOrtho(0, width(), height(), 0, -1, 1); - glMatrixMode(GL_MODELVIEW); - glLoadIdentity(); +// glOrtho(0, width(), height(), 0, -1, 1); +// glMatrixMode(GL_MODELVIEW); +// glLoadIdentity(); // glPopMatrix(); + painter.endNativePainting(); + } -bool gGraphView::renderGraphs() +bool gGraphView::renderGraphs(QPainter &painter) { float px = m_offsetX; float py = -m_offsetY; @@ -1276,12 +1058,12 @@ bool gGraphView::renderGraphs() if (m_showsplitter) { // draw the splitter handle - QColor ca = QColor(128, 128, 128, 255); - backlines->add(0, py + h, w, py + h, ca.rgba()); - ca = QColor(192, 192, 192, 255); - backlines->add(0, py + h + 1, w, py + h + 1, ca.rgba()); - ca = QColor(90, 90, 90, 255); - backlines->add(0, py + h + 2, w, py + h + 2, ca.rgba()); + painter.setPen(QColor(128,128,128,255)); + painter.drawLine(0, py + h, w, py + h); + painter.setPen(QColor(192, 192, 192, 255)); + painter.drawLine(0, py + h + 1, w, py + h + 1); + painter.setPen(QColor(90, 90, 90, 255)); + painter.drawLine(0, py + h + 2, w, py + h + 2); } } @@ -1295,36 +1077,18 @@ bool gGraphView::renderGraphs() for (int i = 0; i < s; i++) { gGraph *g = m_drawlist.at(0); m_drawlist.pop_front(); - g->paint(g->m_rect.x(), g->m_rect.y(), g->m_rect.width(), g->m_rect.height()); + g->paint(painter, g->m_rect.x(), g->m_rect.y(), g->m_rect.width(), g->m_rect.height()); } - backlines->draw(); - - for (int i = 0; i < m_graphs.size(); i++) { - m_graphs[i]->drawGLBuf(); - } - - quads->draw(); - lines->draw(); - - // can't draw snapshot text using this DrawTextQue function - // TODO: Find a better solution for detecting when in snapshot mode if (m_graphs.size() > 1) { - DrawTextQue(); + DrawTextQue(painter); // Draw a gradient behind pinned graphs - // glEnable(GL_BLEND); - //glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - glBegin(GL_QUADS); - glColor4f(0.85, 0.85, 1.0, 1.0); // Gradient End - glVertex2f(0, pinned_height); - glVertex2f(0, 0); - glColor4f(1.0, 1.0, 1.0, 1.0); // Gradient start - glVertex2f(width(), 0); - glVertex2f(width(), pinned_height); - glEnd(); + QLinearGradient linearGrad(QPointF(100, 100), QPointF(width() / 2, 100)); + linearGrad.setColorAt(0, QColor(216, 216, 255)); + linearGrad.setColorAt(1, Qt::white); - // glDisable(GL_BLEND); + painter.fillRect(0, 0, width(), pinned_height, QBrush(linearGrad)); } py = 0; // start drawing at top... @@ -1352,12 +1116,12 @@ bool gGraphView::renderGraphs() if (m_showsplitter) { // draw the splitter handle - QColor ca = QColor(128, 128, 128, 255); - backlines->add(0, py + h, w, py + h, ca.rgba()); - ca = QColor(192, 192, 192, 255); - backlines->add(0, py + h + 1, w, py + h + 1, ca.rgba()); - ca = QColor(90, 90, 90, 255); - backlines->add(0, py + h + 2, w, py + h + 2, ca.rgba()); + painter.setPen(QColor(128, 128, 128, 255)); + painter.drawLine(0, py + h, w, py + h); + painter.setPen(QColor(192, 192, 192, 255)); + painter.drawLine(0, py + h + 1, w, py + h + 1); + painter.setPen(QColor(90, 90, 90, 255)); + painter.drawLine(0, py + h + 2, w, py + h + 2); } } @@ -1384,7 +1148,7 @@ bool gGraphView::renderGraphs() for (int i = 0; i < s; i++) { gGraph *g = m_drawlist.at(0); m_drawlist.pop_front(); - g->paint(g->m_rect.x(), g->m_rect.y(), g->m_rect.width(), g->m_rect.height()); + g->paint(painter, g->m_rect.x(), g->m_rect.y(), g->m_rect.width(), g->m_rect.height()); } #ifdef ENABLED_THREADED_DRAWING @@ -1393,93 +1157,15 @@ bool gGraphView::renderGraphs() //int elapsed=time.elapsed(); //QColor col=Qt::black; - - backlines->draw(); - - for (int i = 0; i < m_graphs.size(); i++) - { - m_graphs[i]->drawGLBuf(); - } - - quads->draw(); - lines->draw(); - - // lines->setSize(linesize); - // DrawTextQue(); + DrawTextQue(painter); //glDisable(GL_TEXTURE_2D); //glDisable(GL_DEPTH_TEST); return numgraphs > 0; } -void gGraphView::fadeOut() -{ - if (!PROFILE.ExistsAndTrue("AnimationsAndTransitions")) { return; } - //if (m_fadingOut) { - // return; - // } - //if (m_inAnimation) { - // m_inAnimation=false; - // } - //clone graphs to shapshot graphview object, render, and then fade in, before switching back to normal mode - /*gGraphView *sg=mainwin->snapshotGraph(); - sg->trashGraphs(); - sg->setFixedSize(width(),height()); - sg->m_graphs=m_graphs; - sg->showSplitter(); */ - - //bool restart=false; - //if (!m_inAnimation) - // restart=true; - - bool b = m_inAnimation; - m_inAnimation = false; - - previous_day_snapshot = renderPixmap(width(), height(), false); - m_inAnimation = b; - //m_fadingOut=true; - //m_fadingIn=false; - //m_inAnimation=true; - //m_limbo=false; - //m_animationStarted.start(); - // updateGL(); -} -void gGraphView::fadeIn(bool dir) -{ - static bool firstdraw = true; - m_tooltip->cancel(); - - if (firstdraw || !PROFILE.ExistsAndTrue("AnimationsAndTransitions")) { - updateGL(); - firstdraw = false; - return; - } - - if (m_fadingIn) { - m_fadingIn = false; - m_inAnimation = false; - updateGL(); - return; - // previous_day_snapshot=current_day_snapshot; - } - - m_inAnimation = false; - current_day_snapshot = renderPixmap(width(), height(), false); - // qDebug() << current_day_snapshot.depth() << "bit image depth"; - // if (current_day_snapshot.hasAlpha()){ - // qDebug() << "Snapshots are not storing alpha channel needed for texture blending"; - // } - m_inAnimation = true; - - m_animationStarted.start(); - m_fadingIn = true; - m_limbo = false; - m_fadedir = dir; - updateGL(); - -} void gGraphView::paintGL() { @@ -1492,146 +1178,47 @@ void gGraphView::paintGL() redrawtimer->stop(); } - bool something_fun = PROFILE.appearance->animations(); + bool render_cube = false; //PROFILE.appearance->animations(); // do something to + if (width() <= 0) { return; } - if (height() <= 0) { return; } - glClearColor(255, 255, 255, 255); - //glClearDepth(1); - glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + // Create QPainter object, note this is only valid from paintGL events! + QPainter painter(this); - bool numgraphs = true; - const int animTimeout = 200; - float phase = 0; + QRect bgrect(0, 0, width(), height()); + painter.fillRect(bgrect,QBrush(QColor(255,255,255))); - int elapsed = 0; + bool graphs_drawn = true; - if (m_inAnimation || m_fadingIn) { - elapsed = m_animationStarted.elapsed(); + lines_drawn_this_frame = 0; + quads_drawn_this_frame = 0; + strings_drawn_this_frame = 0; + strings_cached_this_frame = 0; - if (elapsed > animTimeout) { - if (m_fadingOut) { - m_fadingOut = false; - m_animationStarted.start(); - elapsed = 0; - m_limbo = true; - } else if (m_fadingIn) { - m_fadingIn = false; - m_inAnimation = false; // end animation - m_limbo = false; - m_fadingOut = false; - } + graphs_drawn = renderGraphs(painter); - // + if (!graphs_drawn) { // No graphs drawn? + int x, y; + GetTextExtent(m_emptytext, x, y, bigfont); + int tp; + + if (render_cube && this->isVisible()) { + renderCube(painter); + + tp = height() - (y / 2); } else { - phase = float(elapsed) / float(animTimeout); //percentage of way through animation timeslot - - if (phase > 1.0) { phase = 1.0; } - - if (phase < 0) { phase = 0; } + tp = height() / 2 + y / 2; } - if (m_inAnimation) { - if (m_fadingOut) { - // bindTexture(previous_day_snapshot); - } else if (m_fadingIn) { - //int offset,offset2; - float aphase; - aphase = 1.0 - phase; - /*if (m_fadedir) { // forwards - //offset2=-width(); - //offset=0; - aphase=phase; - phase=1.0-phase; - } else { // backwards - aphase=phase; - phase=phase - //offset=-width(); - //offset2=0;//-width(); - }*/ - //offset=0; offset2=0; - - glEnable(GL_BLEND); - - glDisable(GL_ALPHA_TEST); - glAlphaFunc(GL_GREATER, 0.0); - glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA); - glColor4f(aphase, aphase, aphase, aphase); - - bindTexture(previous_day_snapshot); - glTexEnvi(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE); - glBegin(GL_QUADS); - glTexCoord2f(0.0f, 1.0f); - glVertex2f(0, 0); - glTexCoord2f(1.0f, 1.0f); - glVertex2f(width(), 0); - glTexCoord2f(1.0f, 0.0f); - glVertex2f(width(), height()); - glTexCoord2f(0.0f, 0.0f); - glVertex2f(0, height()); - glEnd(); - - glColor4f(phase, phase, phase, phase); - // glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - - bindTexture(current_day_snapshot); - glBegin(GL_QUADS); - glTexCoord2f(0.0f, 1.0f); - glVertex2f(0, 0); - glTexCoord2f(1.0f, 1.0f); - glVertex2f(width(), 0); - glTexCoord2f(1.0f, 0.0f); - glVertex2f(width(), height()); - glTexCoord2f(0.0f, 0.0f); - glVertex2f(0, height()); - glEnd(); - - glDisable(GL_ALPHA_TEST); - glDisable(GL_BLEND); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - glBindTexture(GL_TEXTURE_2D, 0); - } - } + // Then display the empty text message + QColor col = Qt::black; + AddTextQue(m_emptytext, (width() / 2) - x / 2, tp, 0.0, col, bigfont); } + DrawTextQue(painter); - // Need a really good condition/excuse to switch this on.. :-} - bool bereallyannoying = false; - - if (!m_inAnimation || (!m_fadingIn)) { - // Not in animation sequence, draw graphs like normal - if (bereallyannoying) { - renderCube(0.7F); - } - - numgraphs = renderGraphs(); - - if (!numgraphs) { // No graphs drawn? - int x, y; - GetTextExtent(m_emptytext, x, y, bigfont); - int tp; - - if (something_fun && this->isVisible()) {// Do something fun instead - if (!bereallyannoying) { - renderCube(); - } - - tp = height() - (y / 2); - } else { - tp = height() / 2 + y / 2; - } - - // Then display the empty text message - QColor col = Qt::black; - AddTextQue(m_emptytext, (width() / 2) - x / 2, tp, 0.0, col, bigfont); - - } - - DrawTextQue(); - } - - m_tooltip->paint(); + m_tooltip->paint(painter); #ifdef DEBUG_EFFICIENCY const int rs = 10; @@ -1652,40 +1239,33 @@ 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(strings_drawn_this_frame, 'f', 0) + " strings " + + QString::number(strings_cached_this_frame, 'f', 0) + " cached "; int w, h; - GetTextExtent(ss, w, - h); // this uses tightBoundingRect, which is different on Mac than it is on Windows & Linux. + // this uses tightBoundingRect, which is different on Mac than it is on Windows & Linux. + GetTextExtent(ss, w, h); QColor col = Qt::white; - quads->add(width() - m_graphs[0]->marginRight(), 0, width() - m_graphs[0]->marginRight(), w, - width(), w, width(), 0, col.rgba()); - quads->draw(); - //renderText(0,0,0,ss,*defaultfont); - // int xx=3; + painter.fillRect(width() - m_graphs[0]->marginRight(), 0, m_graphs[0]->marginRight(), w, QBrush(col)); #ifndef Q_OS_MAC // if (usePixmapCache()) xx+=4; else xx-=3; #endif - AddTextQue(ss, width(), w / 2, 90, col, defaultfont); - DrawTextQue(); + AddTextQue(ss, width(), w / 2, 90, QColor(Qt::black), defaultfont); + DrawTextQue(painter); } #endif swapBuffers(); // Dump to screen. - if (this->isVisible()) { - if (m_limbo || m_inAnimation || (something_fun && (bereallyannoying || !numgraphs))) { - redrawtimer->setInterval(1000.0 / 50); - redrawtimer->setSingleShot(true); - redrawtimer->start(); - } + if (this->isVisible() && !graphs_drawn && render_cube) { // keep the cube spinning + redrawtimer->setInterval(1000.0 / 50); // 50 FPS + redrawtimer->setSingleShot(true); + redrawtimer->start(); } } diff --git a/sleepyhead/Graphs/gGraphView.h b/sleepyhead/Graphs/gGraphView.h index b224b744..33802398 100644 --- a/sleepyhead/Graphs/gGraphView.h +++ b/sleepyhead/Graphs/gGraphView.h @@ -21,6 +21,7 @@ #include #include #include +#include #include #include @@ -36,6 +37,22 @@ const int textque_max = 512; \brief Holds a single item of text for the drawing queue */ struct TextQue { + TextQue() { + } + TextQue(short x, short y, float angle, QString text, QColor color, QFont * font, bool antialias): + x(x), y(y), angle(angle), text(text), color(color), font(font), antialias(antialias) + { + } + TextQue(const TextQue & copy) { + x=copy.x; + y=copy.y; + text=copy.text; + angle=copy.angle; + color=copy.color; + font=copy.font; + antialias=copy.antialias; + } + //! \variable contains the x axis screen position to draw the text short x; //! \variable contains the y axis screen position to draw the text @@ -106,8 +123,8 @@ class gToolTip : public QObject */ virtual void display(QString text, int x, int y, int timeout = 0); - //! \brief Queue the actual OpenGL drawing instructions - virtual void paint(); //actually paints it. + //! \brief Draw the tooltip + virtual void paint(QPainter &paint); //actually paints it. //! \brief Close the tooltip early. void cancel(); @@ -133,16 +150,6 @@ class gToolTip : public QObject void timerDone(); }; -/*! \struct myPixmapCache - \brief My version of Pixmap cache with texture binding support - - */ -struct myPixmapCache { - quint64 last_used; - QImage image; - GLuint textureID; -}; - /*! \class gGraphView \brief Main OpenGL Graph Area, derived from QGLWidget @@ -226,12 +233,6 @@ class gGraphView : public QGLWidget //! \brief Set a redraw timer for ms milliseconds, clearing any previous redraw timer. void timedRedraw(int ms); - //! \brief Start the animation sequence changing/reloading day data. (fade out) - void fadeOut(); - - //! \brief Start the animation sequence showing new Day's data. (fade in) - void fadeIn(bool dir = false); - //! \brief Call UpdateGL unless animation is in progress void redraw(); @@ -243,11 +244,11 @@ class gGraphView : public QGLWidget void selectionTime(); //! \brief Add the Text information to the Text Drawing Queue (called by gGraphs renderText method) - void AddTextQue(QString &text, short x, short y, float angle = 0.0, + void AddTextQue(const QString &text, short x, short y, float angle = 0.0, QColor color = Qt::black, QFont *font = defaultfont, bool antialias = true); - //! \brief Draw all Text in the text drawing queue - void DrawTextQue(); +// //! \brief Draw all Text in the text drawing queue +// void DrawTextQue(); //! \brief Draw all text components using QPainter object painter void DrawTextQue(QPainter &painter); @@ -304,8 +305,6 @@ class gGraphView : public QGLWidget //! \brief Sends day object to be distributed to all Graphs Layers objects void setDay(Day *day); - gVertexBuffer *lines, *backlines, *quads, *frontlines; - //! \brief pops a graph off the list for multithreaded drawing code gGraph *popGraph(); // exposed for multithreaded drawing @@ -318,24 +317,27 @@ class gGraphView : public QGLWidget //! \brief Trash all graph objects listed (without destroying Graph contents) void trashGraphs(); - //! \brief Use a QGLFrameBufferObject to render to a pixmap - QImage fboRenderPixmap(int w, int h); - - //! \brief Use a QGLPixelBuffer to render to a pixmap - QImage pbRenderPixmap(int w, int h); - //! \brief Enable or disable the Text Pixmap Caching system preference overide void setUsePixmapCache(bool b) { use_pixmap_cache = b; } //! \brief Return whether or not the Pixmap Cache for text rendering is being used. bool usePixmapCache(); + //! \brief Graph drawing routines, returns true if there weren't any graphs to draw + bool renderGraphs(QPainter &painter); + + // for profiling purposes, a count of lines drawn in a single frame + int lines_drawn_this_frame; + int quads_drawn_this_frame; + int strings_drawn_this_frame; + int strings_cached_this_frame; + protected: //! \brief Set up the OpenGL basics for the QGLWidget underneath virtual void initializeGL(); - //! \brief Resize the OpenGL ViewPort prior to redrawing - virtual void resizeGL(int width, int height); + // //! \brief Resize the OpenGL ViewPort prior to redrawing + //virtual void resizeGL(int width, int height); //! \brief The heart of the OpenGL drawing code virtual void paintGL(); @@ -346,9 +348,6 @@ class gGraphView : public QGLWidget //! \brief Calculates the sum of all graph heights, taking scaling into consideration float scaleHeight(); - //! \brief Graph drawing routines, returns true if there weren't any graphs to draw - bool renderGraphs(); - //! \brief Update the OpenGL area when the screen is resized virtual void resizeEvent(QResizeEvent *); @@ -375,7 +374,7 @@ class gGraphView : public QGLWidget void queGraph(gGraph *, int originX, int originY, int width, int height); //! \brief Render the annoying spinning graph empty cube - void renderCube(float alpha = 1); + void renderCube(QPainter &painter, float alpha = 1); Day *m_day; @@ -415,9 +414,8 @@ class gGraphView : public QGLWidget int m_graph_index; //! \brief List of all queue text to draw.. not sure why I didn't use a vector here.. Might of been a leak issue - TextQue m_textque[textque_max]; + QVector m_textque; - int m_textque_items; int m_lastxpos, m_lastypos; QString m_emptytext; @@ -439,11 +437,10 @@ class gGraphView : public QGLWidget QTime m_animationStarted; - // turn this into a struct later.. - QHash pixmap_cache; - qint32 pixmap_cache_size; bool use_pixmap_cache; + QPixmapCache pixmapcache; + QTime horizScrollTime, vertScrollTime; public slots: diff --git a/sleepyhead/Graphs/gLineChart.cpp b/sleepyhead/Graphs/gLineChart.cpp index 9c66d81b..3bddf23a 100644 --- a/sleepyhead/Graphs/gLineChart.cpp +++ b/sleepyhead/Graphs/gLineChart.cpp @@ -28,10 +28,7 @@ gLineChart::gLineChart(ChannelID code, QColor col, bool square_plot, bool disabl addPlot(code, col, square_plot); m_line_color = col; m_report_empty = false; - addVertexBuffer(lines = new gVertexBuffer(100000, GL_LINES)); - lines->setColor(col); - lines->setAntiAlias(true); - lines->setBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + lines.reserve(50000); } gLineChart::~gLineChart() { @@ -131,12 +128,7 @@ void gLineChart::SetDay(Day *d) } - //if (m_code==CPAP_Leak) { - // subtract_offset=profile.cpap.[IntentionalLeak].toDouble(); - //} else subtract_offset = 0; - - } EventDataType gLineChart::Miny() { @@ -156,7 +148,7 @@ EventDataType gLineChart::Maxy() } // Time Domain Line Chart -void gLineChart::paint(gGraph &w, int left, int top, int width, int height) +void gLineChart::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { if (!m_visible) { return; @@ -174,7 +166,6 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) top++; - // lines=w.lines(); EventDataType miny, maxy; double minx, maxx; @@ -226,12 +217,12 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) int minz, maxz; // Draw bounding box - gVertexBuffer *outlines = w.lines(); - GLuint blk = QColor(Qt::black).rgba(); - outlines->add(left, top, left, top + height, blk); - outlines->add(left, top + height, left + width, top + height, blk); - outlines->add(left + width, top + height, left + width, top, blk); - outlines->add(left + width, top, left, top, blk); + painter.setPen(QColor(Qt::black)); + painter.drawLine(left, top, left, top + height); + painter.drawLine(left, top + height, left + width, top + height); + painter.drawLine(left + width, top + height, left + width, top); + painter.drawLine(left + width, top, left, top); + width--; height -= 2; @@ -250,16 +241,19 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) int codepoints; - //GLuint color; + painter.setClipRect(left, top, width, height+1); + painter.setClipping(true); + painter.setRenderHint(QPainter::Antialiasing, true); + for (int gi = 0; gi < m_codes.size(); gi++) { ChannelID code = m_codes[gi]; - //m_line_color=m_colors[gi]; - lines->setColor(m_colors[gi]); - //color=m_line_color.rgba(); + + lines.clear(); codepoints = 0; - for (int svi = 0; svi < m_day->size(); svi++) { + int daysize = m_day->size(); + for (int svi = 0; svi < daysize; svi++) { Session *sess = (*m_day)[svi]; if (!sess) { @@ -274,9 +268,10 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) schema::Channel ch = schema::channel[code]; bool fndbetter = false; - for (QList::iterator l = ch.m_links.begin(); l != ch.m_links.end(); l++) { - schema::Channel *c = *l; - ci = (*m_day)[svi]->eventlist.find(c->id()); + QList::iterator mlend=ch.m_links.end(); + for (QList::iterator l = ch.m_links.begin(); l != mlend; l++) { + schema::Channel &c = *(*l); + ci = (*m_day)[svi]->eventlist.find(c.id()); if (ci != (*m_day)[svi]->eventlist.end()) { fndbetter = true; @@ -294,17 +289,19 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) QVector &evec = ci.value(); num_points = 0; + int evecsize=evec.size(); - for (int i = 0; i < evec.size(); i++) { + for (int i = 0; i < evecsize; i++) { num_points += evec[i]->count(); } total_points += num_points; codepoints += num_points; - const int num_averages = - 20; // Max n umber of samples taken from samples per pixel for better min/max values - for (int n = 0; n < evec.size(); n++) { // for each segment + // Max number of samples taken from samples per pixel for better min/max values + const int num_averages = 20; + + for (int n = 0; n < evecsize; ++n) { // for each segment EventList &el = *evec[n]; accel = (el.type() == EVL_Waveform); // Turn on acceleration if this is a waveform. @@ -423,22 +420,9 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) double time; EventDataType data; EventDataType gain = el.gain(); - //EventDataType nmult=ymult*gain; - //EventDataType ymin=EventDataType(miny)/gain; - - //const QVector & dat=el.getData(); - //const QVector & tim=el.getTime(); - //quint32 * tptr; - - - //qint64 stime=el.first(); done = false; - // if (!accel) { - lines->setSize(1.5); - // } else lines->setSize(1); - if (el.type() == EVL_Waveform) { // Waveform Plot if (idx > sam) { idx -= sam; } @@ -451,12 +435,6 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) // Accelerated Waveform Plot ////////////////////////////////////////////////////////////////// - // qint64 tmax=(maxx-time)/rate; - // if ((tmax*sam) < siz) { - // siz=idx+tmax*sam; - // done=true; - // } - for (int i = idx; i < siz; i += sam, ptr += sam) { time += rate; // This is much faster than QVector access. @@ -519,21 +497,12 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) // Cap within VertexBuffer capacity, one vertex per line point - int np = (maxz - minz) * 2; +// int np = (maxz - minz) * 2; - int j = lines->Max() - lines->cnt(); - - if (np < j) { - for (int i = minz; i < maxz; i++, drl++) { - ax1 = drl->x(); - ay1 = drl->y(); - lines->unsafe_add(xst + i, yst - ax1, xst + i, yst - ay1); - - //if (lines->full()) break; - } - } else { - qDebug() << "gLineChart full trying to draw" << schema::channel[code].label(); - done = true; + for (int i = minz; i < maxz; i++, drl++) { + ax1 = drl->x(); + ay1 = drl->y(); + lines.append(QLine(xst + i, yst - ax1, xst + i, yst - ay1)); } } else { // Zoomed in Waveform @@ -541,13 +510,6 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) // Normal Waveform Plot ////////////////////////////////////////////////////////////////// - // Cap within VertexBuffer capacity, one vertex per line point - // int np=((siz-idx)/sam)*2; - // int j=lines->Max()-lines->cnt(); - // if (np > j) { - // siz=j*sam; - // } - // Prime first point data = (*ptr + el.offset()) * gain; lastpx = xst + ((time - minx) * xmult); @@ -563,7 +525,7 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) py = yst - ((data - miny) * ymult); // Same for Y scale, with precomputed gain //py=yst-((data - ymin) * nmult); // Same for Y scale, with precomputed gain - lines->add(lastpx, lastpy, px, py); + lines.append(QLine(lastpx, lastpy, px, py)); lastpx = px; lastpy = py; @@ -572,10 +534,6 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) done = true; break; } - - if (lines->full()) { - break; - } } } @@ -619,20 +577,12 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) siz -= idx; - // Check if would overflow lines gVertexBuffer int gs = siz << 1; - int j = lines->Max() - lines->cnt(); if (square_plot) { gs <<= 1; } - if (gs > j) { - qDebug() << "Would overflow line points.. increase default VertexBuffer size in gLineChart"; - siz = (j >> square_plot) ? 2 : 1; - done = true; // end after this partial draw.. - } - // Unrolling square plot outside of loop to gain a minor speed improvement. EventStoreType *eptr = dptr + siz; @@ -652,12 +602,15 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) // Cap px to right margin if (px > xst + width) { px = xst + width; } - lines->unsafe_add(lastpx, lastpy, px, lastpy, px, lastpy, px, py); + lines.append(QLine(lastpx, lastpy, px, lastpy)); + lines.append(QLine(px, lastpy, px, py)); } else { // Letting the scissor do the dirty work for non horizontal lines // This really should be changed, as it might be cause that weird // display glitch on Linux.. - lines->unsafe_add(lastpx, lastpy, px, lastpy, px, lastpy, px, py); + + lines.append(QLine(lastpx, lastpy, px, lastpy)); + lines.append(QLine(px, lastpy, px, py)); } lastpx = px; @@ -685,12 +638,12 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) // Cap px to right margin if (px > xst + width) { px = xst + width; } - lines->unsafe_add(lastpx, lastpy, px, py); + lines.append(QLine(lastpx, lastpy, px, py)); } else { // Letting the scissor do the dirty work for non horizontal lines // This really should be changed, as it might be cause that weird // display glitch on Linux.. - lines->unsafe_add(lastpx, lastpy, px, py); + lines.append(QLine(lastpx, lastpy, px, py)); } lastpx = px; @@ -707,6 +660,10 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) if (done) { break; } } } + painter.setPen(QPen(m_colors[gi],p_profile->appearance->lineThickness())); + painter.drawLines(lines); + w.graphView()->lines_drawn_this_frame+=lines.count(); + lines.clear(); //////////////////////////////////////////////////////////////////// // Draw Legends on the top line @@ -716,17 +673,19 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) int bw = fm.width('X'); int bh = fm.height() / 1.8; - if ((codepoints > 0)) { //(m_codes.size()>1) && + + if ((codepoints > 0)) { QString text = schema::channel[code].label(); int wid, hi; GetTextExtent(text, wid, hi); legendx -= wid; + painter.setClipping(false); w.renderText(text, legendx, top - 4); + legendx -= bw /2; + painter.fillRect(legendx - bw, top - w.marginTop()-2, bh, w.marginTop()+1, QBrush(m_colors[gi])); + painter.setClipping(true); - int tp = top - 5 - bh / 2; - w.quads()->add(legendx - bw, tp + bh / 2, legendx, tp + bh / 2, legendx, tp - bh / 2, legendx - bw, - tp - bh / 2, m_colors[gi].rgba()); - legendx -= bw * 2; + legendx -= bw*2; } } @@ -738,219 +697,7 @@ void gLineChart::paint(gGraph &w, int left, int top, int width, int height) GetTextExtent(msg, x, y, bigfont); //DrawText(w,msg,left+(width/2.0)-(x/2.0),scry-w.GetBottomMargin()-height/2.0+y/2.0,0,Qt::gray,bigfont); } - } else { -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - float dpr = w.graphView()->devicePixelRatio(); - lines->scissor(left * dpr, w.flipY(top + height + 2)*dpr, (width + 1)*dpr, (height + 1)*dpr); -#else - lines->scissor(left, w.flipY(top + height + 2), width + 1, height + 1); -#endif } -} - - -AHIChart::AHIChart(QColor col) - : Layer(NoChannel), m_color(col) -{ - m_miny = m_maxy = 0; - addVertexBuffer(lines = new gVertexBuffer(100000, GL_LINES)); - lines->setColor(col); - lines->setAntiAlias(true); - lines->setSize(1.5); -} - -AHIChart::~AHIChart() -{ -} - -void AHIChart::paint(gGraph &w, int left, int top, int width, int height) -{ - if (!m_visible) { - return; - } - - if (!m_day) { - return; - } - - // Draw bounding box - gVertexBuffer *outlines = w.lines(); - GLuint blk = QColor(Qt::black).rgba(); - outlines->add(left, top, left, top + height, blk); - outlines->add(left, top + height, left + width, top + height, blk); - outlines->add(left + width, top + height, left + width, top, blk); - outlines->add(left + width, top, left, top, blk); - width--; - height -= 2; - - EventDataType miny, maxy; - double minx, maxx; - - maxx = w.max_x, minx = w.min_x; - - // hmmm.. subtract_offset.. - - if (w.zoomY() == 0 && PROFILE.appearance->allowYAxisScaling()) { - miny = w.physMinY(); - maxy = w.physMaxY(); - } else { - miny = w.min_y, maxy = w.max_y; - } - - w.roundY(miny, maxy); - - double xx = maxx - minx; - double xmult = double(width) / xx; - - EventDataType yy = maxy - miny; - EventDataType ymult = EventDataType(height - 3) / yy; // time to pixel conversion multiplier - - bool first = true; - double px, py; - double lastpx, lastpy; - double top1 = top + height; - bool done = false; - - //GLuint color=m_color.rgba(); - for (int i = 0; i < m_time.size(); i++) { - qint64 ti = m_time[i]; - EventDataType v = m_data[i]; - - if (ti < minx) { continue; } - - if (ti > maxx) { done = true; } - - if (first) { - if (i > 0) { - ti = m_time[i - 1]; - v = m_data[i - 1]; - i--; - } - - px = left + (double(ti - minx) * xmult); - py = top1 - (double(v - miny) * ymult); - first = false; - } else { - px = left + (double(ti - minx) * xmult); - py = top1 - (double(v - miny) * ymult); - lines->add(px, py, lastpx, lastpy); - } - - lastpx = px; - lastpy = py; - - if (done) { break; } - } - -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - float dpr = w.graphView()->devicePixelRatio(); - lines->scissor(left * dpr, w.flipY(top + height + 2)*dpr, (width + 1)*dpr, (height + 1)*dpr); -#else - lines->scissor(left, w.flipY(top + height + 2), width + 1, height + 1); -#endif -} - -void AHIChart::SetDay(Day *d) -{ - m_day = d; - m_data.clear(); - m_time.clear(); - m_maxy = 0; - m_miny = 0; - - if (!d) { return; } - - m_miny = 9999; - QList::iterator s; - qint64 first = d->first(); - qint64 last = d->last(); - qint64 f; - - qint64 winsize = 30000; // 30 second windows - - for (qint64 ti = first; ti < last; ti += winsize) { - f = ti - 3600000L; - //if (fclockDrift()) * 1000L), drift = 0; - - bool fnd = false; - - for (s = d->begin(); s != d->end(); s++) { - if (!(*s)->enabled()) { continue; } - - Session *sess = *s; - - if ((ti < sess->first()) || (f > sess->last())) { continue; } - - drift = (sess->machine()->GetType() == MT_CPAP) ? clockdrift : 0; - - // Drop off suddenly outside of sessions - //if (ti>sess->last()) continue; - - fnd = true; - - if (sess->eventlist.contains(CPAP_Obstructive)) { - el[0] = sess->eventlist[CPAP_Obstructive][0]; - } else { el[0] = nullptr; } - - if (sess->eventlist.contains(CPAP_Apnea)) { - el[1] = sess->eventlist[CPAP_Apnea][0]; - } else { el[1] = nullptr; } - - if (sess->eventlist.contains(CPAP_Hypopnea)) { - el[2] = sess->eventlist[CPAP_Hypopnea][0]; - } else { el[2] = nullptr; } - - if (sess->eventlist.contains(CPAP_ClearAirway)) { - el[3] = sess->eventlist[CPAP_ClearAirway][0]; - } else { el[3] = nullptr; } - - if (sess->eventlist.contains(CPAP_NRI)) { - el[4] = sess->eventlist[CPAP_NRI][0]; - } else { el[4] = nullptr; } - - int znt = 5; - - if (PROFILE.general->calculateRDI()) { - if (sess->eventlist.contains(CPAP_RERA)) {// What about ExP?? - el[5] = sess->eventlist[CPAP_RERA][0]; - znt++; - } else { el[5] = nullptr; } - } - - qint64 t; - - for (int i = 0; i < znt; i++) { - if (!el[i]) { continue; } - - for (quint32 j = 0; j < el[i]->count(); j++) { - t = el[i]->time(j) + drift; - - if ((t >= f) && (t < ti)) { - cnt++; - } - } - } - } - - if (!fnd) { cnt = 0; } - - double g = double(ti - f) / 3600000.0; - - if (g > 0) { ahi = cnt / g; } - - if (ahi < m_miny) { m_miny = ahi; } - - if (ahi > m_maxy) { m_maxy = ahi; } - - m_time.append(ti); - m_data.append(ahi); - } - - m_minx = first; - m_maxx = last; + painter.setClipping(false); + painter.setRenderHint(QPainter::Antialiasing, false); } diff --git a/sleepyhead/Graphs/gLineChart.h b/sleepyhead/Graphs/gLineChart.h index cf34b85c..d952da73 100644 --- a/sleepyhead/Graphs/gLineChart.h +++ b/sleepyhead/Graphs/gLineChart.h @@ -14,50 +14,10 @@ #include -#include "Graphs/gVertexBuffer.h" #include "Graphs/layer.h" #include "SleepLib/event.h" #include "SleepLib/day.h" -/*! \class AHIChart - \brief Another graph calculating the AHI/hour, this one looks at all the sessions for a day. Currently Unused. - */ -class AHIChart: public Layer -{ - public: - //! \brief Constructs an AHIChart object, with QColor col for the line plots. - AHIChart(QColor col = QColor("black")); - ~AHIChart(); - - //! \brief Draws the precalculated data to the Vertex buffers - virtual void paint(gGraph &w, int left, int top, int width, int height); - - //! \brief AHI/hr Calculations are done for this day here. - //! This also uses the sliding window method - virtual void SetDay(Day *d); - - //! \brief Returns the minimum AHI/hr value caculated - virtual EventDataType Miny() { return m_miny; } - - //! \brief Returns the maximum AHI/hr value caculated - virtual EventDataType Maxy() { return m_maxy; } - - //! \brief Returns true if no data was available - virtual bool isEmpty() { return m_data.size() == 0; } - - protected: - //! \brief Contains the plot data (Y-axis) generated for this day - QVector m_data; - - //! \brief Contains the time codes (X-axis) generated for this day - QVector m_time; - - EventDataType m_miny; - EventDataType m_maxy; - QColor m_color; - gVertexBuffer *lines; -}; - /*! \class gLineChart \brief Draws a 2D linechart from all Session data in a day. EVL_Waveforms typed EventLists are accelerated. */ @@ -75,7 +35,7 @@ class gLineChart: public Layer virtual ~gLineChart(); //! \brief The drawing code that fills the vertex buffers - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); //! \brief Set Use Square plots for non EVL_Waveform data void SetSquarePlot(bool b) { m_square_plot = b; } @@ -123,10 +83,6 @@ class gLineChart: public Layer bool m_disable_accel; QColor m_line_color; - gVertexBuffer *lines; - //GLShortBuffer * lines; - //GLShortBuffer * outlines; - //! \brief Used by accelerated waveform plots. Must be >= Screen Resolution (or at least graph width) static const int max_drawlist_size = 10000; @@ -139,6 +95,8 @@ class gLineChart: public Layer QVector m_colors; QVector m_square; QHash m_enabled; + + QVector lines; }; #endif // GLINECHART_H diff --git a/sleepyhead/Graphs/gLineOverlay.cpp b/sleepyhead/Graphs/gLineOverlay.cpp index 59d2b8dc..887d9c48 100644 --- a/sleepyhead/Graphs/gLineOverlay.cpp +++ b/sleepyhead/Graphs/gLineOverlay.cpp @@ -16,30 +16,17 @@ gLineOverlayBar::gLineOverlayBar(ChannelID code, QColor color, QString label, FlagType flt) : Layer(code), m_flag_color(color), m_label(label), m_flt(flt) { - addVertexBuffer(points = new gVertexBuffer(2048, GL_POINTS)); - points->setSize(4); - points->setColor(m_flag_color); - addVertexBuffer(quads = new gVertexBuffer(2048, GL_QUADS)); - //addGLBuf(lines=new GLBuffer(color,1024,GL_LINES)); - points->setAntiAlias(true); - quads->setAntiAlias(true); - quads->setColor(m_flag_color); - //lines->setAntiAlias(true); } gLineOverlayBar::~gLineOverlayBar() { - //delete lines; - //delete quads; - //delete points; } -void gLineOverlayBar::paint(gGraph &w, int left, int topp, int width, int height) +void gLineOverlayBar::paint(QPainter &painter, gGraph &w, int left, int topp, int width, int height) { if (!m_visible) { return; } if (!m_day) { return; } - gVertexBuffer *lines = w.lines(); int start_py = topp; double xx = w.max_x - w.min_x; @@ -56,19 +43,17 @@ void gLineOverlayBar::paint(gGraph &w, int left, int topp, int width, int height double X; double Y; - bool verts_exceeded = false; - m_count = 0; m_sum = 0; m_flag_color = schema::channel[m_code].defaultColor(); - lines->setColor(m_flag_color); - points->setColor(m_flag_color); if (m_flt == FT_Span) { m_flag_color.setAlpha(128); } + painter.setPen(m_flag_color); + EventStoreType raw; quint32 *tptr; @@ -149,16 +134,7 @@ void gLineOverlayBar::paint(gGraph &w, int left, int topp, int width, int height x1 = width + left; } - quads->add(x2, start_py, - x1, start_py, - x1, start_py + height, - x2, start_py + height, - m_flag_color.rgba()); - - if (quads->full()) { - verts_exceeded = true; - break; - } + painter.fillRect(x2, start_py, x1-x2, height, QBrush(m_flag_color)); } } else if (m_flt == FT_Dot) { //////////////////////////////////////////////////////////////////////////// @@ -178,22 +154,15 @@ void gLineOverlayBar::paint(gGraph &w, int left, int topp, int width, int height if ((odt == ODT_Bars) || (xx < 3600000)) { // show the fat dots in the middle - points->add(x1, double(height) / double(yy)*double(-20 - w.min_y) + topp); + painter.setPen(QPen(m_flag_color,4)); + + painter.drawPoint(x1, double(height) / double(yy)*double(-20 - w.min_y) + topp); + painter.setPen(QPen(m_flag_color,1)); - if (points->full()) { - verts_exceeded = true; - break; - } } else { // thin lines down the bottom - lines->add(x1, start_py + 1, x1, start_py + 1 + 12); - - if (lines->full()) { - verts_exceeded = true; - break; - } + painter.drawLine(x1, start_py + 1, x1, start_py + 1 + 12); } - } } else if (m_flt == FT_Bar) { //////////////////////////////////////////////////////////////////////////// @@ -215,20 +184,13 @@ void gLineOverlayBar::paint(gGraph &w, int left, int topp, int width, int height if ((odt == ODT_Bars) || (xx < 3600000)) { z = top; - points->add(x1, top); - lines->add(x1, top, x1, bottom); + painter.setPen(QPen(m_flag_color,4)); + painter.drawPoint(x1, top); + painter.setPen(QPen(m_flag_color,1)); + painter.drawLine(x1, top, x1, bottom); - if (points->full()) { - verts_exceeded = true; - break; - } } else { - lines->add(x1, z, x1, z - 12); - } - - if (lines->full()) { - verts_exceeded = true; - break; + painter.drawLine(x1, z, x1, z - 12); } if (xx < (1800000)) { @@ -238,18 +200,8 @@ void gLineOverlayBar::paint(gGraph &w, int left, int topp, int width, int height } } - if (verts_exceeded) { - break; - } } - if (verts_exceeded) { - break; - } - } - - if (verts_exceeded) { - qWarning() << "exceeded maxverts in gLineOverlay::Plot()"; } } @@ -262,8 +214,10 @@ gLineOverlaySummary::~gLineOverlaySummary() { } -void gLineOverlaySummary::paint(gGraph &w, int left, int top, int width, int height) +void gLineOverlaySummary::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { + Q_UNUSED(painter) + if (!m_visible) { return; } if (!m_day) { return; } @@ -337,7 +291,7 @@ void gLineOverlaySummary::paint(gGraph &w, int left, int top, int width, int hei if (sph > 100) { sph = 100; } } - a += " " + QObject::tr("(\%%1 in events)").arg(sph, 0, 'f', + a += " " + QObject::tr("(% %1 in events)").arg(sph, 0, 'f', 2); // eg: %num of time in a span, like Periodic Breathing } diff --git a/sleepyhead/Graphs/gLineOverlay.h b/sleepyhead/Graphs/gLineOverlay.h index 34476385..da8089ca 100644 --- a/sleepyhead/Graphs/gLineOverlay.h +++ b/sleepyhead/Graphs/gLineOverlay.h @@ -26,7 +26,7 @@ class gLineOverlayBar: public Layer virtual ~gLineOverlayBar(); //! \brief The drawing code that fills the OpenGL vertex GLBuffers - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); virtual EventDataType Miny() { return 0; } virtual EventDataType Maxy() { return 0; } @@ -44,9 +44,6 @@ class gLineOverlayBar: public Layer FlagType m_flt; int m_count; double m_sum; - - gVertexBuffer *quads; - gVertexBuffer *points; }; /*! \class gLineOverlaySummary @@ -58,7 +55,7 @@ class gLineOverlaySummary: public Layer gLineOverlaySummary(QString text, int x, int y); virtual ~gLineOverlaySummary(); - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); virtual EventDataType Miny() { return 0; } virtual EventDataType Maxy() { return 0; } diff --git a/sleepyhead/Graphs/gSegmentChart.cpp b/sleepyhead/Graphs/gSegmentChart.cpp index c4290304..2f24cf2e 100644 --- a/sleepyhead/Graphs/gSegmentChart.cpp +++ b/sleepyhead/Graphs/gSegmentChart.cpp @@ -17,12 +17,6 @@ gSegmentChart::gSegmentChart(GraphSegmentType type, QColor gradient_color, QColo m_outline_color(outline_color) { m_empty = true; - addGLBuf(poly = new GLFloatBuffer(4000, GL_POLYGON)); - addGLBuf(lines = new GLFloatBuffer(4000, GL_LINE_LOOP)); - lines->setSize(1); - poly->forceAntiAlias(false); - lines->forceAntiAlias(true); - lines->setAntiAlias(true); } gSegmentChart::~gSegmentChart() { @@ -69,7 +63,7 @@ bool gSegmentChart::isEmpty() return m_empty; } -void gSegmentChart::paint(gGraph &w, int left, int top, int width, int height) +void gSegmentChart::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { if (!m_visible) { return; } @@ -83,12 +77,6 @@ void gSegmentChart::paint(gGraph &w, int left, int top, int width, int height) diameter -= 8; float radius = diameter / 2.0; - float j = 0.0; - float sum = 0.0; - float step = 1.0 / 720.0; - float px, py; - float q; - float xmult = float(width) / float(m_total); float ymult = float(height) / float(m_total); @@ -113,9 +101,9 @@ void gSegmentChart::paint(gGraph &w, int left, int top, int width, int height) bool line_first = true; int line_last; - gVertexBuffer *quads = w.quads(); - gVertexBuffer *lines2 = w.lines(); + float sum = -90.0; + painter.setFont(*defaultfont); for (unsigned m = 0; m < size; m++) { data = m_values[m]; @@ -126,62 +114,64 @@ void gSegmentChart::paint(gGraph &w, int left, int top, int width, int height) ///////////////////////////////////////////////////////////////////////////////////// if (m_graph_type == GST_Pie) { QColor &col = schema::channel[m_codes[m % m_colors.size()]].defaultColor(); - j = float(data) / float(m_total); // ratio of this pie slice - // Draw Filling - poly->add(start_px + xoffset, start_py + height - yoffset, m_gradient_color); + // length of this segment in degrees + float len = 360.0 / float(m_total) * float(data); - for (q = sum; q < sum + j; q += step) { - px = start_px + xoffset + sin(q * 2 * M_PI) * radius; - py = start_py + height - (yoffset + cos(q * 2 * M_PI) * radius); - poly->add(px, py, col); - } + // Setup the shiny radial gradient - q = sum + j; - px = start_px + xoffset + sin(q * 2 * M_PI) * radius; - py = start_py + height - (yoffset + cos(q * 2 * M_PI) * radius); - poly->add(px, py, col); + painter.setRenderHint(QPainter::Antialiasing); + QRect pierect(start_px+1, start_py+1, width-2, height-2); - if (m_total != data) { - // Draw the center point first - lines->add(start_px + xoffset, start_py + height - yoffset, m_outline_color); - } + painter.setPen(QPen(col, 0)); + QRadialGradient gradient(pierect.center(), pierect.width()/2, pierect.center()); + gradient.setColorAt(0, Qt::white); + gradient.setColorAt(1, col); - for (q = sum; q < sum + j; q += step) { - px = start_px + xoffset + sin(q * 2 * M_PI) * radius; - py = start_py + height - (yoffset + cos(q * 2 * M_PI) * radius); - lines->add(px, py, m_outline_color); - } + // draw filled pie + painter.setBrush(gradient); + painter.setBackgroundMode(Qt::OpaqueMode); + painter.drawPie(pierect, -sum * 16.0, -len * 16.0); - double tpx = start_px + xoffset + sin((sum + (j / 2.0)) * 2 * M_PI) * (radius / 1.7); - double tpy = start_py + height - (yoffset + cos((sum + (j / 2.0)) * 2 * M_PI) * (radius / 1.7)); - q = sum + j; - px = start_px + xoffset + sin(q * 2 * M_PI) * radius; - py = start_py + height - (yoffset + cos(q * 2 * M_PI) * radius); - lines->add(px, py, m_outline_color); + // draw outline + painter.setBackgroundMode(Qt::TransparentMode); + painter.setBrush(QBrush(col,Qt::NoBrush)); + painter.setPen(QPen(QColor(Qt::black),1.5)); + painter.drawPie(pierect, -sum * 16.0, -len * 16.0); - if (j > .09) { - QString a = m_names[m]; //QString::number(floor(100.0/m_total*data),'f',0)+"%"; + + // Draw text labels if they fit + if (len > 20) { + float angle = (sum+90.0) + len / 2.0; + double tpx = (pierect.x() + pierect.width()/2) + (sin((180 - angle) * (M_PI / 180.0)) * (radius / 1.65)); + double tpy = (pierect.y() + pierect.height()/2) + (cos((180 - angle) * (M_PI / 180.0)) * (radius / 1.65)); + + QString txt = m_names[m]; //QString::number(floor(100.0/m_total*data),'f',0)+"%"; int x, y; - GetTextExtent(a, x, y); - w.renderText(a, tpx - (x / 2.0), (tpy + y / 2.0), 0, Qt::black, defaultfont, - false); // antialiasing looks like crap here.. + GetTextExtent(txt, x, y); + // antialiasing looks like crap here.. + painter.setPen(QColor(Qt::black)); + painter.drawText(tpx - (x / 2.0), tpy+3, txt); } + sum += len; - sum = q; - + } else if (m_graph_type == GST_CandleStick) { ///////////////////////////////////////////////////////////////////////////////////// // CandleStick Chart ///////////////////////////////////////////////////////////////////////////////////// - } else if (m_graph_type == GST_CandleStick) { + QColor &col = m_colors[m % m_colors.size()]; + float bw = xmult * float(data); - quads->add(xp, start_py, xp + bw, start_py, m_gradient_color.rgba()); - quads->add(xp + bw, start_py + height, xp, start_py + height, col.rgba()); + QLinearGradient linearGrad(QPointF(0, 0), QPointF(bw, 0)); + linearGrad.setColorAt(0, col); + linearGrad.setColorAt(1, Qt::white); + painter.fillRect(xp, start_py, bw, height, QBrush(linearGrad)); - lines2->add(xp, start_py, xp + bw, start_py, m_outline_color.rgba()); - lines2->add(xp + bw, start_py + height, xp, start_py + height, m_outline_color.rgba()); + painter.setPen(m_outline_color); + painter.drawLine(xp, start_py, xp + bw, start_py); + painter.drawLine(xp + bw, start_py + height, xp, start_py + height); if (!m_names[m].isEmpty()) { int px, py; @@ -198,12 +188,13 @@ void gSegmentChart::paint(gGraph &w, int left, int top, int width, int height) ///////////////////////////////////////////////////////////////////////////////////// } else if (m_graph_type == GST_Line) { QColor col = Qt::black; //m_colors[m % m_colors.size()]; + painter.setPen(col); float h = (top + height) - (float(data) * ymult); if (line_first) { line_first = false; } else { - lines->add(xp, line_last, xp + line_step, h, col); + painter.drawLine(xp, line_last, xp + line_step, h); xp += line_step; } diff --git a/sleepyhead/Graphs/gSegmentChart.h b/sleepyhead/Graphs/gSegmentChart.h index cdcaabd2..321bdc92 100644 --- a/sleepyhead/Graphs/gSegmentChart.h +++ b/sleepyhead/Graphs/gSegmentChart.h @@ -27,7 +27,7 @@ class gSegmentChart : public Layer virtual ~gSegmentChart(); //! \brief The drawing code that fills the Vertex buffers - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); //! \brief Pre-fills a buffer with the data needed to draw virtual void SetDay(Day *d); @@ -57,9 +57,6 @@ class gSegmentChart : public Layer QColor m_gradient_color; QColor m_outline_color; bool m_empty; - - // gah.. can't convert these - GLFloatBuffer *poly, *lines; }; /*! \class gTAPGraph diff --git a/sleepyhead/Graphs/gStatsLine.cpp b/sleepyhead/Graphs/gStatsLine.cpp index b3a68220..06920817 100644 --- a/sleepyhead/Graphs/gStatsLine.cpp +++ b/sleepyhead/Graphs/gStatsLine.cpp @@ -17,8 +17,10 @@ gStatsLine::gStatsLine(ChannelID code, QString label, QColor textcolor) : Layer(code), m_label(label), m_textcolor(textcolor) { } -void gStatsLine::paint(gGraph &w, int left, int top, int width, int height) +void gStatsLine::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { + Q_UNUSED(painter) + if (!m_visible) { return; } //if (m_empty) return; diff --git a/sleepyhead/Graphs/gStatsLine.h b/sleepyhead/Graphs/gStatsLine.h index f79e66e3..02bf490a 100644 --- a/sleepyhead/Graphs/gStatsLine.h +++ b/sleepyhead/Graphs/gStatsLine.h @@ -22,7 +22,7 @@ class gStatsLine : public Layer { public: gStatsLine(ChannelID code, QString label = "", QColor textcolor = Qt::black); - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); void SetDay(Day *d); protected: diff --git a/sleepyhead/Graphs/gSummaryChart.cpp b/sleepyhead/Graphs/gSummaryChart.cpp index 1133e248..3349821f 100644 --- a/sleepyhead/Graphs/gSummaryChart.cpp +++ b/sleepyhead/Graphs/gSummaryChart.cpp @@ -19,18 +19,6 @@ extern QLabel *qstatus2; SummaryChart::SummaryChart(QString label, GraphType type) : Layer(NoChannel), m_label(label), m_graphtype(type) { - //QColor color=Qt::black; - addVertexBuffer(quads = new gVertexBuffer(20000, GL_QUADS)); - addVertexBuffer(lines = new gVertexBuffer(20000, GL_LINES)); - addVertexBuffer(outlines = new gVertexBuffer(20000, GL_LINES)); - addVertexBuffer(points = new gVertexBuffer(20000, GL_POINTS)); - quads->forceAntiAlias(true); - - outlines->setSize(1); - - //lines->setBlendFunc(GL_SRC_COLOR, GL_ZERO); - //lines->forceAntiAlias(false); - m_empty = true; hl_day = -1; m_machinetype = MT_CPAP; @@ -55,8 +43,8 @@ void SummaryChart::SetDay(Day *nullday) m_days.clear(); m_hours.clear(); m_goodcodes.clear(); - m_miny = 999999999; - m_maxy = -999999999; + m_miny = 999999999.0F; + m_maxy = -999999999.0F; m_physmaxy = 0; m_physminy = 0; m_minx = 0; @@ -390,12 +378,10 @@ QColor brighten(QColor color) } -void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) +void SummaryChart::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { if (!m_visible) { return; } - points->setSize(10); - GraphType graphtype = m_graphtype; if (graphtype == GT_LINE || graphtype == GT_POINTS) { @@ -403,21 +389,13 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) graphtype = pts ? GT_POINTS : GT_LINE; } - if (graphtype == GT_POINTS) { - lines->setSize(4); - } else { - lines->setSize(1.5); - } - - rtop = top; - gVertexBuffer *outlines2 = w.lines(); - // outlines2->setColor(Qt::black); - outlines2->add(left, top, left, top + height, left, top + height, left + width, top + height, - QColor("black").rgba()); - outlines2->add(left + width, top + height, left + width, top, left + width, top, left, top, - QColor("black").rgba()); - //if (outlines->full()) qDebug() << "WTF??? Outlines full in SummaryChart::paint()"; + + painter.setPen(QColor(Qt::black)); + painter.drawLine(left, top, left, top+height); + painter.drawLine(left, top+height, left+width, top+height); + painter.drawLine(left+width, top+height, left+width, top); + painter.drawLine( left+width, top, left, top); qint64 minx = w.min_x, maxx = w.max_x; qint64 xx = maxx - minx; @@ -441,8 +419,6 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) barw = (float(width) / float(days)); - float dpr = w.graphView()->devicePixelRatio(); - graph = &w; float px = left; l_left = w.marginLeft() + gYAxis::Margin; @@ -484,7 +460,6 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) QVector goodcodes; goodcodes.resize(m_goodcodes.size()); - points->setSize(5.0 * dpr); lastdaygood = true; for (int i = 0; i < numcodes; i++) { @@ -534,6 +509,10 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) short px2, py2; const qint64 ms_per_day = 86400000L; + + painter.setClipRect(left, top, width, height); + painter.setClipping(true); + for (qint64 Q = minx; Q <= maxx + ms_per_day; Q += ms_per_day) { zd = Q / ms_per_day; d = m_values.find(zd); @@ -555,7 +534,7 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) //x1-=(barw/2.0); int x2 = px + barw; - if (x1 < left) { x1 = left; } + //if (x1 < left) { x1 = left; } if (x2 > left + width) { x2 = left + width; } @@ -568,10 +547,12 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) col.setAlpha(64); if (graphtype != GT_POINTS) { - quads->add(x1 - 1, top, x1 - 1, top + height, x2, top + height, x2, top, col.rgba()); + painter.fillRect(x1-1, top, barw, height, QBrush(col)); +// quads->add(x1 - 1, top, x1 - 1, top + height, x2, top + height, x2, top, col.rgba()); } else { - quads->add((x1 + barw / 2) - 5, top, (x1 + barw / 2) - 5, top + height, (x2 - barw / 2) + 5, - top + height, (x2 - barw / 2) + 5, top, col.rgba()); + painter.fillRect((x1+barw/2)-5, top, barw, height, QBrush(col)); +// quads->add((x1 + barw / 2) - 5, top, (x1 + barw / 2) - 5, top + height, (x2 - barw / 2) + 5, +// top + height, (x2 - barw / 2) + 5, top, col.rgba()); } } @@ -585,8 +566,8 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) col = QColor("gold"); } - GLuint col1 = col.rgba(); - GLuint col2 = brighten(col).rgba(); + QColor col1 = col; + QColor col2 = Qt::white; //outlines->setColor(Qt::black); int np = d.value().size(); @@ -606,11 +587,18 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) //tmp-=miny; h = tmp * ymult; - quads->add(x1, py, x1, py - h, x2, py - h, x2, py, col1, col2); + QLinearGradient gradient(x1, py-h, x1+barw, py-h); + gradient.setColorAt(0,col1); + gradient.setColorAt(1,col2); + painter.fillRect(x1, py-h, barw, h, QBrush(gradient)); +// quads->add(x1, py, x1, py - h, x2, py - h, x2, py, col1, col2); - if (h > 0 && barw > 2) { - outlines->add(x1, py, x1, py - h, x1, py - h, x2, py - h, QColor("black").rgba()); - outlines->add(x1, py, x2, py, x2, py, x2, py - h, QColor("black").rgba()); + if ((h > 0) && (barw > 2)) { + painter.setPen(QColor(Qt::black)); + painter.drawLine(x1, py, x1, py - h); + painter.drawLine(x1, py - h, x2, py - h); + painter.drawLine(x1, py, x2, py); + painter.drawLine(x2, py, x2, py - h); } totalvalues[0] += hours * tmp; @@ -694,17 +682,23 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) h = tmp * ymult; // height in pixels if (graphtype == GT_BAR) { - GLuint col1 = col.rgba(); - GLuint col2 = brighten(col).rgba(); + QColor col1 = col; + QColor col2 = Qt::white; - quads->add(x1, py, x1, py - h, col1); - quads->add(x2, py - h, x2, py, col2); + QLinearGradient gradient(x1, py-h, x1+barw, py-h); + gradient.setColorAt(0,col1); + gradient.setColorAt(1,col2); + painter.fillRect(x1, py-h, barw, h, QBrush(gradient)); + +// quads->add(x1, py, x1, py - h, col1); +// quads->add(x2, py - h, x2, py, col2); if (h > 0 && barw > 2) { - outlines->add(x1, py, x1, py - h, x1, py - h, x2, py - h, QColor("black").rgba()); - outlines->add(x1, py, x2, py, x2, py, x2, py - h, QColor("black").rgba()); - - if (outlines->full()) { qDebug() << "WTF??? Outlines full in SummaryChart::paint()"; } + painter.setPen(QColor(Qt::black)); + painter.drawLine(x1, py, x1, py - h); + painter.drawLine(x1, py - h, x2, py - h); + painter.drawLine(x1, py, x2, py); + painter.drawLine(x2, py, x2, py - h); } // if (bar py -= h; @@ -721,19 +715,22 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) if (lastdaygood) { if (lastY[j] != py2) { // vertical line - lines->add(lastX[j], lastY[j], px, py2, col2); + painter.setPen(col2); + painter.drawLine(lastX[j], lastY[j], px, py2); } - lines->add(px - 1, py2, px2 + 1, py2, col1); + painter.setPen(col1); + painter.drawLine(px - 1, py2, px2 + 1, py2); } else { - lines->add(x1 - 1, py2, x2 + 1, py2, col1); + painter.setPen(col1); + painter.drawLine(x1 - 1, py2, x2 + 1, py2); } lastX[j] = px2; lastY[j] = py2; } else if (graphtype == GT_POINTS) { - GLuint col1 = col.rgba(); - GLuint col2 = m_colors[j].rgba(); + QColor col1 = col; + QColor col2 = m_colors[j]; px2 = px + barw; py2 = (top + height - 2) - h; @@ -743,13 +740,16 @@ void SummaryChart::paint(gGraph &w, int left, int top, int width, int height) } if (zd == hl_day) { - points->add(px2 - barw / 2, py2, col2); + painter.setPen(QPen(brighten(col2),10)); + painter.drawPoint(px2 - barw / 2, py2); } if (lastdaygood) { - lines->add(lastX[j] - barw / 2, lastY[j], px2 - barw / 2, py2, col2); + painter.setPen(QPen(col2,p_profile->appearance->lineThickness())); + painter.drawLine(lastX[j] - barw / 2, lastY[j], px2 - barw / 2, py2); } else { - lines->add(px + barw / 2 - 1, py2, px + barw / 2 + 1, py2, col1); + painter.setPen(QPen(col1,p_profile->appearance->lineThickness())); + painter.drawLine(px + barw / 2 - 1, py2, px + barw / 2 + 1, py2); } lastX[j] = px2; @@ -779,16 +779,17 @@ jumpnext: daynum++; //lastQ=Q; } + painter.setClipping(false); -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - quads->scissor(left * dpr, w.flipY(top + height + 2)*dpr, (width)*dpr, (height + 1)*dpr); - lines->scissor(left * dpr, w.flipY(top + height + 2)*dpr, (width + 1)*dpr, (height + 1)*dpr); - outlines->scissor(left * dpr, w.flipY(top + height + 2)*dpr, (width)*dpr, (height + 1)*dpr); -#else - lines->scissor(left, w.flipY(top + height + 2), width + 1, height + 2); - outlines->scissor(left, w.flipY(top + height + 2), width, height + 2); - quads->scissor(left, w.flipY(top + height + 2), width, height + 2); -#endif +//#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) +// quads->scissor(left * dpr, w.flipY(top + height + 2)*dpr, (width)*dpr, (height + 1)*dpr); +// lines->scissor(left * dpr, w.flipY(top + height + 2)*dpr, (width + 1)*dpr, (height + 1)*dpr); +// outlines->scissor(left * dpr, w.flipY(top + height + 2)*dpr, (width)*dpr, (height + 1)*dpr); +//#else +// lines->scissor(left, w.flipY(top + height + 2), width + 1, height + 2); +// outlines->scissor(left, w.flipY(top + height + 2), width, height + 2); +// quads->scissor(left, w.flipY(top + height + 2), width, height + 2); +//#endif // Draw Ledgend px = left + width - 3; py = top - 5; @@ -910,9 +911,7 @@ jumpnext: w.renderText(a, legendx, top - 4); // legendx-=bw/2; - int tp = top - 5 - bh / 2; - w.quads()->add(legendx - bw, tp + bh / 2, legendx, tp + bh / 2, legendx, tp - bh / 2, legendx - bw, - tp - bh / 2, m_colors[j].rgba()); + painter.fillRect(legendx - bw, top-w.marginTop()-1, bh, w.marginTop(), QBrush(m_colors[j])); legendx -= bw * 2; diff --git a/sleepyhead/Graphs/gSummaryChart.h b/sleepyhead/Graphs/gSummaryChart.h index 657eec8d..3c6b6e8c 100644 --- a/sleepyhead/Graphs/gSummaryChart.h +++ b/sleepyhead/Graphs/gSummaryChart.h @@ -34,7 +34,7 @@ class SummaryChart: public Layer virtual ~SummaryChart(); //! \brief Drawing code that fills the Vertex buffers - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); //! \brief Precalculation code prior to drawing. Day object is not needed here, it's just here for Layer compatability. virtual void SetDay(Day *day = nullptr); @@ -79,10 +79,6 @@ class SummaryChart: public Layer QHash m_hours; QHash m_days; - gVertexBuffer *quads; - gVertexBuffer *lines; - gVertexBuffer *outlines; - gVertexBuffer *points; bool m_empty; int m_fday; QString m_label; diff --git a/sleepyhead/Graphs/gVertexBuffer.cpp b/sleepyhead/Graphs/gVertexBuffer.cpp deleted file mode 100644 index a516e1d9..00000000 --- a/sleepyhead/Graphs/gVertexBuffer.cpp +++ /dev/null @@ -1,304 +0,0 @@ -/* -*- 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 deleted file mode 100644 index 2759739e..00000000 --- a/sleepyhead/Graphs/gVertexBuffer.h +++ /dev/null @@ -1,128 +0,0 @@ -/* -*- 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 71599701..051498e5 100644 --- a/sleepyhead/Graphs/gXAxis.cpp +++ b/sleepyhead/Graphs/gXAxis.cpp @@ -40,7 +40,6 @@ gXAxis::gXAxis(QColor col, bool fadeout) m_show_major_ticks = true; m_utcfix = false; m_fadeout = fadeout; - m_textureID = 0; // QDateTime d=QDateTime::currentDateTime(); // QTime t1=d.time(); // QTime t2=d.toUTC().time(); @@ -55,7 +54,7 @@ gXAxis::gXAxis(QColor col, bool fadeout) gXAxis::~gXAxis() { } -void gXAxis::paint(gGraph &w, int left, int top, int width, int height) +void gXAxis::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { Q_UNUSED(height) QString months[] = { @@ -65,8 +64,9 @@ void gXAxis::paint(gGraph &w, int left, int top, int width, int height) //static QString dow[]={"Sun","Mon","Tue","Wed","Thu","Fri","Sat"}; + QVector ticks; - QPainter painter; // Only need this for pixmap caching + QPainter painter2; // Only need this for pixmap caching // pixmap caching screws font size when printing @@ -75,14 +75,11 @@ void gXAxis::paint(gGraph &w, int left, int top, int width, int height) if (!usepixmap || (usepixmap && w.invalidate_xAxisImage)) { if (usepixmap) { - // Unbind any previous texture - if (m_textureID) { w.graphView()->deleteTexture(m_textureID); } - m_image = QImage(width + 22, height + 4, QImage::Format_ARGB32_Premultiplied); m_image.fill(Qt::transparent); - painter.begin(&m_image); - painter.setPen(Qt::black); - painter.setFont(*defaultfont); + painter2.begin(&m_image); + painter2.setPen(Qt::black); + painter2.setFont(*defaultfont); } double px, py; @@ -187,8 +184,7 @@ void gXAxis::paint(gGraph &w, int left, int top, int width, int height) aligned_start += step; } - gVertexBuffer *lines = w.backlines(); - lines->setColor(Qt::black); + painter.setPen(QColor(Qt::black)); //int utcoff=m_utcfix ? tz_hours : 0; @@ -225,9 +221,9 @@ void gXAxis::paint(gGraph &w, int left, int top, int width, int height) if (py < start_px) { continue; } if (usepixmap) { - painter.drawLine(py - left + 20, 0, py - left + 20, mintop - top); + ticks.append(QLine(py - left + 20, 0, py - left + 20, mintop - top)); } else { - lines->add(py, top, py, mintop); + ticks.append(QLine(py, top+2, py, mintop+2)); } } @@ -239,8 +235,10 @@ void gXAxis::paint(gGraph &w, int left, int top, int width, int height) px += left; if (usepixmap) { - painter.drawLine(px - left + 20, 0, px - left + 20, majtop - top); - } else { lines->add(px, top, px, majtop); } + ticks.append(QLine(px - left + 20, 0, px - left + 20, majtop - top)); + } else { + ticks.append(QLine(px, top+2, px, majtop+2)); + } j = i; @@ -281,7 +279,7 @@ void gXAxis::paint(gGraph &w, int left, int top, int width, int height) if ((tx + x) < (left + width)) { if (!usepixmap) { w.renderText(tmpstr, tx, texttop, 0, Qt::black, defaultfont); } - else { painter.drawText(tx - left + 20, texttop - top, tmpstr); } + else { painter2.drawText(tx - left + 20, texttop - top, tmpstr); } } py = px; @@ -292,34 +290,26 @@ void gXAxis::paint(gGraph &w, int left, int top, int width, int height) if (py >= left + width) { break; } if (usepixmap) { - painter.drawLine(py - left + 20, 0, py - left + 20, mintop - top); - } else { lines->add(py, top, py, mintop); } - } - - if (lines->full()) { - qWarning() << "maxverts exceeded in gXAxis::Plot()"; - break; + ticks.append(QLine(py - left + 20, 0, py - left + 20, mintop - top)); + } else { + ticks.append(QLine(py, top+2, py, mintop+2)); + } } } if (usepixmap) { - painter.end(); - m_image = QGLWidget::convertToGLFormat(m_image); - m_textureID = w.graphView()->bindTexture(m_image, GL_TEXTURE_2D, GL_RGBA, - QGLContext::NoBindOption); - + painter2.drawLines(ticks); + painter2.end(); + } else { + painter.drawLines(ticks); } + w.graphView()->lines_drawn_this_frame += ticks.size(); w.invalidate_xAxisImage = false; } if (usepixmap && !m_image.isNull()) { - glEnable(GL_BLEND); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - glEnable(GL_TEXTURE_2D); - w.graphView()->drawTexture(QPoint(left - 20, (top + height) - m_image.height() + 4), m_textureID); - glDisable(GL_TEXTURE_2D); - glDisable(GL_BLEND); + painter.drawImage(QPoint(left - 20, top + height - m_image.height() + 4), m_image); } } diff --git a/sleepyhead/Graphs/gXAxis.h b/sleepyhead/Graphs/gXAxis.h index 9c629d2a..1d1a09ff 100644 --- a/sleepyhead/Graphs/gXAxis.h +++ b/sleepyhead/Graphs/gXAxis.h @@ -12,6 +12,8 @@ #ifndef GXAXIS_H #define GXAXIS_H +#include +#include #include "Graphs/layer.h" /*! \class gXAxis @@ -21,7 +23,7 @@ class gXAxis: public Layer public: gXAxis(QColor col = Qt::black, bool fadeout = true); virtual ~gXAxis(); - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); static const int Margin = 20; // How much room does this take up. (Bottom margin) void SetShowMinorLines(bool b) { m_show_minor_lines = b; } void SetShowMajorLines(bool b) { m_show_major_lines = b; } @@ -51,6 +53,5 @@ class gXAxis: public Layer float tz_hours; QImage m_image; - GLuint m_textureID; }; #endif // GXAXIS_H diff --git a/sleepyhead/Graphs/gYAxis.cpp b/sleepyhead/Graphs/gYAxis.cpp index af276a2f..4e0ee7bd 100644 --- a/sleepyhead/Graphs/gYAxis.cpp +++ b/sleepyhead/Graphs/gYAxis.cpp @@ -34,12 +34,10 @@ gXGrid::gXGrid(QColor col) gXGrid::~gXGrid() { } -void gXGrid::paint(gGraph &w, int left, int top, int width, int height) +void gXGrid::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { int x, y; - gVertexBuffer *stippled, * lines; - EventDataType miny, maxy; if (w.zoomY() == 0 && PROFILE.appearance->allowYAxisScaling()) { @@ -108,17 +106,15 @@ void gXGrid::paint(gGraph &w, int left, int top, int width, int height) if (min_ytick >= 1000000) { min_ytick = 100; } - - - stippled = w.backlines(); - lines = w.backlines(); + QVector majorlines; + QVector minorlines; for (double i = miny; i <= maxy + min_ytick - 0.00001; i += min_ytick) { ty = (i - miny) * ymult; h = top + height - ty; if (m_show_major_lines && (i > miny)) { - stippled->add(left, h, left + width, h, m_major_color.rgba()); + majorlines.append(QLine(left, h, left + width, h)); } double z = (min_ytick / 4) * ymult; @@ -134,20 +130,15 @@ void gXGrid::paint(gGraph &w, int left, int top, int width, int height) // break; // } if (m_show_minor_lines) {// && (i > miny)) { - stippled->add(left, g, left + width, g, m_minor_color.rgba()); + minorlines.append(QLine(left, g, left + width, g)); } - - if (stippled->full()) { - break; - } - } - - if (lines->full() || stippled->full()) { - qWarning() << "vertarray bounds exceeded in gYAxis for " << w.title() << "graph" << "MinY =" << - miny << "MaxY =" << maxy << "min_ytick=" << min_ytick; - break; } } + painter.setPen(QPen(m_major_color,1)); + painter.drawLines(majorlines); + painter.setPen(QPen(m_minor_color,1)); + painter.drawLines(minorlines); + w.graphView()->lines_drawn_this_frame += majorlines.size() + minorlines.size(); } @@ -157,14 +148,12 @@ gYAxis::gYAxis(QColor col) { m_line_color = col; m_text_color = col; - m_textureID = 0; - m_yaxis_scale = 1; } gYAxis::~gYAxis() { } -void gYAxis::paint(gGraph &w, int left, int top, int width, int height) +void gYAxis::paint(QPainter &painter, gGraph &w, int left, int top, int width, int height) { int x, y; //,yh=0; @@ -371,9 +360,7 @@ void gYAxis::paint(gGraph &w, int left, int top, int width, int height) min_ytick = 100; } - lines = w.backlines(); - - GLuint line_color = m_line_color.rgba(); + QVector ticks; for (double i = miny; i <= maxy + min_ytick - 0.00001; i += min_ytick) { ty = (i - miny) * ymult; @@ -394,7 +381,7 @@ void gYAxis::paint(gGraph &w, int left, int top, int width, int height) w.renderText(fd, left + width - 8 - x, (h + (y / 2.0)), 0, m_text_color, defaultfont); - lines->add(left + width - 4, h, left + width, h, line_color); + ticks.append(QLine(left + width - 4, h, left + width, h)); double z = (min_ytick / 4) * ymult; double g = h; @@ -404,21 +391,13 @@ void gYAxis::paint(gGraph &w, int left, int top, int width, int height) if (g > top + height) { break; } - lines->add(left + width - 3, g, left + width, g, line_color); - - if (lines->full()) { - qWarning() << "vertarray bounds exceeded in gYAxis for " << w.title() << "graph" << "MinY =" << - miny << "MaxY =" << maxy << "min_ytick=" << min_ytick; - break; - } - } - - if (lines->full()) { - qWarning() << "vertarray bounds exceeded in gYAxis for " << w.title() << "graph" << "MinY =" << - miny << "MaxY =" << maxy << "min_ytick=" << min_ytick; - break; + ticks.append(QLine(left + width - 3, g, left + width, g)); } } + painter.setPen(m_line_color); + painter.drawLines(ticks); + w.graphView()->lines_drawn_this_frame += ticks.size(); + } } const QString gYAxis::Format(EventDataType v, int dp) diff --git a/sleepyhead/Graphs/gYAxis.h b/sleepyhead/Graphs/gYAxis.h index 8a23be9e..1db0fc95 100644 --- a/sleepyhead/Graphs/gYAxis.h +++ b/sleepyhead/Graphs/gYAxis.h @@ -12,6 +12,7 @@ #ifndef GYAXIS_H #define GYAXIS_H +#include #include "Graphs/layer.h" /*! \class gXGrid @@ -25,7 +26,7 @@ class gXGrid: public Layer virtual ~gXGrid(); //! \brief Draw the horizontal lines by adding the to the Vertex GLbuffers - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); //! \brief set the visibility status of Major lines void setShowMinorLines(bool b) { m_show_minor_lines = b; } @@ -56,7 +57,7 @@ class gYAxis: public Layer virtual ~gYAxis(); //! \brief Draw the horizontal tickers display - virtual void paint(gGraph &w, int left, int top, int width, int height); + virtual void paint(QPainter &painter, gGraph &w, int left, int top, int width, int height); // void SetShowMinorLines(bool b) { m_show_minor_lines=b; } // void SetShowMajorLines(bool b) { m_show_major_lines=b; } @@ -96,13 +97,10 @@ class gYAxis: public Layer QColor m_line_color; QColor m_text_color; - gVertexBuffer *lines; virtual bool mouseMoveEvent(QMouseEvent *event, gGraph *graph); virtual bool mouseDoubleClickEvent(QMouseEvent *event, gGraph *graph); QImage m_image; - GLuint m_textureID; - }; /*! \class gYAxisTime diff --git a/sleepyhead/Graphs/glcommon.cpp b/sleepyhead/Graphs/glcommon.cpp index fd0f3b53..c97c81c3 100644 --- a/sleepyhead/Graphs/glcommon.cpp +++ b/sleepyhead/Graphs/glcommon.cpp @@ -13,85 +13,12 @@ #include "glcommon.h" #ifdef BUILD_WITH_MSVC + +#if (_MSC_VER < 1800) double round(double number) { return number < 0.0 ? ceil(number - 0.5) : floor(number + 0.5); } #endif - - -void RoundedRectangle(int x, int y, int w, int h, int radius, const QColor color) -{ - - glEnable(GL_BLEND); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - glColor4ub(color.red(), color.green(), color.blue(), color.alpha()); - - glBegin(GL_POLYGON); - glVertex2i(x + radius, y); - glVertex2i(x + w - radius, y); - - for (float i = (float)M_PI * 1.5f; i < M_PI * 2; i += 0.1f) { - glVertex2f(x + w - radius + cos(i)*radius, y + radius + sin(i)*radius); - } - - glVertex2i(x + w, y + radius); - glVertex2i(x + w, y + h - radius); - - for (float i = 0; i < (float)M_PI * 0.5f; i += 0.1f) { - glVertex2f(x + w - radius + cos(i)*radius, y + h - radius + sin(i)*radius); - } - - glVertex2i(x + w - radius, y + h); - glVertex2i(x + radius, y + h); - - for (float i = (float)M_PI * 0.5f; i < M_PI; i += 0.1f) { - glVertex2f(x + radius + cos(i)*radius, y + h - radius + sin(i)*radius); - } - - glVertex2i(x, y + h - radius); - glVertex2i(x, y + radius); - - for (float i = (float)M_PI; i < M_PI * 1.5f; i += 0.1f) { - glVertex2f(x + radius + cos(i)*radius, y + radius + sin(i)*radius); - } - - glEnd(); - - glDisable(GL_BLEND); -} - -void LinedRoundedRectangle(int x, int y, int w, int h, int radius, int lw, QColor color) -{ - //glDisable(GL_TEXTURE_2D); - glShadeModel(GL_SMOOTH); - glEnable(GL_BLEND); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - glColor4ub(color.red(), color.green(), color.blue(), color.alpha()); - glLineWidth((GLfloat)lw); - - glBegin(GL_LINE_STRIP); - - for (float i = (float)M_PI; i <= 1.5f * M_PI; i += 0.1f) { - glVertex2f(radius * cos(i) + x + radius, radius * sin(i) + y + radius); - } - - for (float i = 1.5f * (float)M_PI; i <= 2 * M_PI; i += 0.1f) { - glVertex2f(radius * cos(i) + x + w - radius, radius * sin(i) + y + radius); - } - - for (float i = 0; i <= 0.5f * M_PI; i += 0.1f) { - glVertex2f(radius * cos(i) + x + w - radius, radius * sin(i) + y + h - radius); - } - - for (float i = 0.5f * (float)M_PI; i <= M_PI; i += 0.1f) { - glVertex2f(radius * cos(i) + x + radius, radius * sin(i) + y + h - radius); - } - - glVertex2i(x, y + radius); - glEnd(); - - //glEnable(GL_TEXTURE_2D); - glDisable(GL_BLEND); -} +#endif diff --git a/sleepyhead/Graphs/glcommon.h b/sleepyhead/Graphs/glcommon.h index d50efac7..49522075 100644 --- a/sleepyhead/Graphs/glcommon.h +++ b/sleepyhead/Graphs/glcommon.h @@ -12,20 +12,12 @@ #ifndef GLCOMMON_H #define GLCOMMON_H -#include #include #ifndef nullptr #define nullptr NULL #endif -#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)); @@ -64,27 +56,15 @@ const QColor COLOR_ALT_BG2 = COLOR_White; // Alternating Background Color 2 (Event Flags) -/*! \brief Draw an outline of a rounded rectangle - \param radius Radius of corner rounding - \param lw Line Width - \param color Color of drawn lines - */ -void LinedRoundedRectangle(int x, int y, int w, int h, int radius, int lw, QColor color); - -/*! \brief Draws a filled rounded rectangle - \param radius Radius of corner rounding - \param color Color of entire rectangle - */ -void RoundedRectangle(int x, int y, int w, int h, int radius, const QColor color); - #ifndef M_PI const double M_PI = 3.141592653589793; #endif #ifdef BUILD_WITH_MSVC -// Visual C++ doesn't have round in it's maths header.. - +// Visual C++ earlier than 2013 doesn't have round in it's maths header.. +#if (_MSC_VER < 1800) double round(double number); #endif +#endif #endif // GLCOMMON_H diff --git a/sleepyhead/Graphs/gspacer.h b/sleepyhead/Graphs/gspacer.h index ec300ad9..a77d862a 100644 --- a/sleepyhead/Graphs/gspacer.h +++ b/sleepyhead/Graphs/gspacer.h @@ -22,8 +22,9 @@ class gSpacer: public Layer { public: gSpacer(int space = 20); // orientation? - virtual void paint(gGraph &g, int left, int top, int width, int height) { + virtual void paint(QPainter &painter, gGraph &g, int left, int top, int width, int height) { Q_UNUSED(g) + Q_UNUSED(painter) Q_UNUSED(left) Q_UNUSED(top) Q_UNUSED(width) diff --git a/sleepyhead/Graphs/layer.cpp b/sleepyhead/Graphs/layer.cpp index bf52431b..b7b9d88c 100644 --- a/sleepyhead/Graphs/layer.cpp +++ b/sleepyhead/Graphs/layer.cpp @@ -11,52 +11,53 @@ Layer::~Layer() { - for (int i = 0; i < mgl_buffers.size(); i++) { - delete mgl_buffers[i]; - } +// 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]; - } +// 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; } +//void Layer::drawGLBuf(float linesize) +//{ +// int type; +// float size; - GLBuffer *buf; - gVertexBuffer *vb; +// if (!m_visible) { return; } - for (int i = 0; i < mv_buffers.size(); i++) { - vb = mv_buffers[i]; - size = vb->size(); - type = vb->type(); +// GLBuffer *buf; +// gVertexBuffer *vb; - if ((linesize > size) && ((type == GL_LINES) || (type == GL_LINE_LOOP))) { - vb->setSize(linesize); - } +// for (int i = 0; i < mv_buffers.size(); i++) { +// vb = mv_buffers[i]; +// size = vb->size(); +// type = vb->type(); - vb->draw(); - vb->setSize(size); - } +// if ((linesize > size) && ((type == GL_LINES) || (type == GL_LINE_LOOP))) { +// vb->setSize(linesize); +// } - for (int i = 0; i < mgl_buffers.size(); i++) { - buf = mgl_buffers[i]; - size = buf->size(); - type = buf->type(); +// vb->draw(); +// vb->setSize(size); +// } - if ((linesize > size) && ((type == GL_LINES) || (type == GL_LINE_LOOP))) { - buf->setSize(linesize); - } +// for (int i = 0; i < mgl_buffers.size(); i++) { +// buf = mgl_buffers[i]; +// size = buf->size(); +// type = buf->type(); - buf->draw(); - //if ((linesize>size) && ((type==GL_LINES) || (type==GL_LINE_LOOP))) { - buf->setSize(size); - //} - } -} +// 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) { @@ -110,14 +111,14 @@ bool LayerGroup::isEmpty() return empty; } -void LayerGroup::drawGLBuf(float linesize) -{ - Layer::drawGLBuf(linesize); +//void LayerGroup::drawGLBuf(float linesize) +//{ +// Layer::drawGLBuf(linesize); - for (int i = 0; i < layers.size(); i++) { - layers[i]->drawGLBuf(linesize); - } -} +// for (int i = 0; i < layers.size(); i++) { +// layers[i]->drawGLBuf(linesize); +// } +//} void LayerGroup::SetDay(Day *d) { diff --git a/sleepyhead/Graphs/layer.h b/sleepyhead/Graphs/layer.h index 1008664d..6d0515fa 100644 --- a/sleepyhead/Graphs/layer.h +++ b/sleepyhead/Graphs/layer.h @@ -16,8 +16,6 @@ #include #include -#include "Graphs/GLBuffer.h" -#include "Graphs/gVertexBuffer.h" #include "SleepLib/common.h" #include "SleepLib/day.h" #include "SleepLib/machine_common.h" @@ -119,7 +117,7 @@ class Layer \param int width \param int height */ - virtual void paint(gGraph &gv, int left, int top, int width, int height) = 0; + virtual void paint(QPainter &painter, 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); @@ -134,8 +132,8 @@ class Layer //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 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; @@ -146,9 +144,9 @@ class Layer } 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); } +// //! \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; @@ -166,9 +164,9 @@ class Layer LayerPosition m_position; QRect m_rect; - //! \brief A vector containing all this layers custom drawing buffers - QVector mgl_buffers; - QVector mv_buffers; +// //! \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) { @@ -241,8 +239,8 @@ class LayerGroup : public Layer //! \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 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; } diff --git a/sleepyhead/SleepLib/calcs.cpp b/sleepyhead/SleepLib/calcs.cpp index 20f2a180..5d678190 100644 --- a/sleepyhead/SleepLib/calcs.cpp +++ b/sleepyhead/SleepLib/calcs.cpp @@ -367,7 +367,7 @@ void FlowParser::calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool if (calcResp) { RR = m_session->AddEventList(CPAP_RespRate, EVL_Event); minrr = RR->Min(), maxrr = RR->Max(); - RR->setGain(0.2); + RR->setGain(0.2F); RR->setFirst(time + minute); RR->getData().resize(nm); RR->getTime().resize(nm); @@ -391,12 +391,12 @@ void FlowParser::calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool if (calcTi) { Ti = m_session->AddEventList(CPAP_Ti, EVL_Event); - Ti->setGain(0.02); + Ti->setGain(0.02F); } if (calcTe) { Te = m_session->AddEventList(CPAP_Te, EVL_Event); - Te->setGain(0.02); + Te->setGain(0.02F); } @@ -641,13 +641,13 @@ void FlowParser::flagEvents() //EventList * uf2=m_session->AddEventList(CPAP_UserFlag2,EVL_Event); //EventList * uf3=m_session->AddEventList(CPAP_UserFlag3,EVL_Event); - const EventDataType perc = 0.6; + const EventDataType perc = 0.6F; int idx = float(br.size()) * perc; nth_element(br.begin(), br.begin() + idx, br.end() - 1); EventDataType peak = br[idx]; //*(br.begin()+idx); - EventDataType cutoffval = peak * (PROFILE.cpap->userFlowRestriction() / 100.0); + EventDataType cutoffval = peak * (PROFILE.cpap->userFlowRestriction() / 100.0F); int bs, bm, be, bs1, bm1, be1; @@ -906,14 +906,14 @@ int calcAHIGraph(Session *session) f; EventList *AHI = new EventList(EVL_Event); - AHI->setGain(0.02); + AHI->setGain(0.02F); session->eventlist[CPAP_AHI].push_back(AHI); EventList *RDI = nullptr; if (calcrdi) { RDI = new EventList(EVL_Event); - RDI->setGain(0.02); + RDI->setGain(0.02F); session->eventlist[CPAP_RDI].push_back(RDI); } @@ -926,7 +926,7 @@ int calcAHIGraph(Session *session) int cnt = 0; double events; - double hours = (window_size / 60.0); + double hours = (window_size / 60.0F); if (zeroreset) { // I personally don't see the point of resetting each hour. diff --git a/sleepyhead/SleepLib/common.h b/sleepyhead/SleepLib/common.h index 394fc87b..1f933232 100644 --- a/sleepyhead/SleepLib/common.h +++ b/sleepyhead/SleepLib/common.h @@ -83,6 +83,7 @@ const QString STR_GEN_On = QObject::tr("On"); const QString STR_GEN_Off = QObject::tr("Off"); const QString STR_PREF_AllowEarlyUpdates = "AllowEarlyUpdates"; +const QString STR_PREF_ReimportBackup = "ReimportBackup"; const QString STR_PROP_Brand = "Brand"; const QString STR_PROP_Model = "Model"; diff --git a/sleepyhead/SleepLib/day.cpp b/sleepyhead/SleepLib/day.cpp index c42100b4..78805aad 100644 --- a/sleepyhead/SleepLib/day.cpp +++ b/sleepyhead/SleepLib/day.cpp @@ -305,7 +305,7 @@ EventDataType Day::percentile(ChannelID code, EventDataType percentile) EventDataType Day::p90(ChannelID code) { - return percentile(code, 0.90); + return percentile(code, 0.90F); } EventDataType Day::avg(ChannelID code) diff --git a/sleepyhead/SleepLib/event.cpp b/sleepyhead/SleepLib/event.cpp index 5bf0b0f2..9266de8a 100644 --- a/sleepyhead/SleepLib/event.cpp +++ b/sleepyhead/SleepLib/event.cpp @@ -22,8 +22,8 @@ EventList::EventList(EventListType et, EventDataType gain, EventDataType offset, if (min == max) { // Update Min & Max unless forceably set here.. m_update_minmax = true; - m_min2 = m_min = 999999999; - m_max2 = m_max = -999999999; + m_min2 = m_min = 999999999.0F; + m_max2 = m_max = -999999999.0F; } else { m_update_minmax = false; } @@ -35,8 +35,8 @@ EventList::EventList(EventListType et, EventDataType gain, EventDataType offset, void EventList::clear() { - m_min2 = m_min = 999999999; - m_max2 = m_max = -999999999; + m_min2 = m_min = 999999999.0F; + m_max2 = m_max = -999999999.0F; m_update_minmax = true; m_first = m_last = 0; m_count = 0; @@ -310,7 +310,7 @@ void EventList::AddWaveform(qint64 start, char *data, int recs, qint64 duration) if (m_update_minmax) { for (sp = data; sp < ep; sp++) { raw = *sp; - val = EventDataType(val) * m_gain + m_offset; + val = EventDataType(raw) * m_gain + m_offset; if (m_min > val) { m_min = val; } @@ -321,7 +321,7 @@ void EventList::AddWaveform(qint64 start, char *data, int recs, qint64 duration) } else { for (sp = data; sp < ep; sp++) { raw = *sp; - val = EventDataType(val) * m_gain + m_offset; + val = EventDataType(raw) * m_gain + m_offset; *dp++ = raw; } } diff --git a/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp b/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp index d8b79d23..1141a72c 100644 --- a/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/icon_loader.cpp @@ -121,6 +121,7 @@ int FPIconLoader::Open(QString &path, Profile *profile) try { if (m) { OpenMachine(m, npath, profile); } } catch (OneTypePerDay e) { + Q_UNUSED(e) profile->DelMachine(m); MachList.erase(MachList.find(sn)); QMessageBox::warning(nullptr, "Import Error", @@ -477,9 +478,9 @@ bool FPIconLoader::OpenFLW(Machine *mach, QString filename, Profile *profile) const double rate = 1000.0 / double(samples_per_block); // F&P Overwrites this file, not appends to it. - flow = new EventList(EVL_Waveform, 1.0, 0, 0, 0, rate); + flow = new EventList(EVL_Waveform, 1.0F, 0, 0, 0, rate); //leak=new EventList(EVL_Event,1.0,0,0,0,rate*double(samples_per_block)); // 1 per second - pressure = new EventList(EVL_Event, 0.01, 0, 0, 0, + pressure = new EventList(EVL_Event, 0.01F, 0, 0, 0, rate * double(samples_per_block)); // 1 per second flow->setFirst(ti); @@ -832,7 +833,7 @@ bool FPIconLoader::OpenDetail(Machine *mach, QString filename, Profile *profile) ti = qint64(sessid) * 1000L; sess->really_set_first(ti); EventList *LK = sess->AddEventList(CPAP_LeakTotal, EVL_Event, 1); - EventList *PR = sess->AddEventList(CPAP_Pressure, EVL_Event, 0.1); + EventList *PR = sess->AddEventList(CPAP_Pressure, EVL_Event, 0.1F); EventList *FLG = sess->AddEventList(CPAP_FLG, EVL_Event); EventList *OA = sess->AddEventList(CPAP_Obstructive, EVL_Event); EventList *H = sess->AddEventList(CPAP_Hypopnea, EVL_Event); diff --git a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp index ababf3fa..7c86a0d0 100644 --- a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp @@ -288,6 +288,7 @@ int PRS1Loader::Open(QString &path, Profile *profile) OpenMachine(m, newpath + "/" + (*sn), profile); } } catch (OneTypePerDay e) { + Q_UNUSED(e) profile->DelMachine(m); PRS1List.erase(PRS1List.find(s)); QMessageBox::warning(nullptr, QObject::tr("Import Error"), @@ -726,8 +727,6 @@ bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, // Summary files vary too much between Families/versions - unsigned duration; - if (size < 59) { // duration = data[offset + 0x12] | (data[offset + 0x13] << 8); // duration *= 2; @@ -849,16 +848,16 @@ bool PRS1Loader::Parse002v5(qint32 sequence, quint32 timestamp, unsigned char *b EventList *CSR = session->AddEventList(CPAP_CSR, EVL_Event); EventList *LEAK = session->AddEventList(CPAP_LeakTotal, EVL_Event); EventList *SNORE = session->AddEventList(CPAP_Snore, EVL_Event); - EventList *IPAP = session->AddEventList(CPAP_IPAP, EVL_Event, 0.1); - EventList *EPAP = session->AddEventList(CPAP_EPAP, EVL_Event, 0.1); - EventList *PS = session->AddEventList(CPAP_PS, EVL_Event, 0.1); - EventList *IPAPLo = session->AddEventList(CPAP_IPAPLo, EVL_Event, 0.1); - EventList *IPAPHi = session->AddEventList(CPAP_IPAPHi, EVL_Event, 0.1); + EventList *IPAP = session->AddEventList(CPAP_IPAP, EVL_Event, 0.1F); + EventList *EPAP = session->AddEventList(CPAP_EPAP, EVL_Event, 0.1F); + EventList *PS = session->AddEventList(CPAP_PS, EVL_Event, 0.1F); + EventList *IPAPLo = session->AddEventList(CPAP_IPAPLo, EVL_Event, 0.1F); + EventList *IPAPHi = session->AddEventList(CPAP_IPAPHi, EVL_Event, 0.1F); EventList *RR = session->AddEventList(CPAP_RespRate, EVL_Event); EventList *PTB = session->AddEventList(CPAP_PTB, EVL_Event); EventList *MV = session->AddEventList(CPAP_MinuteVent, EVL_Event); - EventList *TV = session->AddEventList(CPAP_TidalVolume, EVL_Event, 10); + EventList *TV = session->AddEventList(CPAP_TidalVolume, EVL_Event, 10.0F); EventList *CA = nullptr; //session->AddEventList(CPAP_ClearAirway, EVL_Event); EventList *VS = nullptr, * FL = nullptr; //,* RE=nullptr,* VS2=nullptr; @@ -934,7 +933,7 @@ bool PRS1Loader::Parse002v5(qint32 sequence, quint32 timestamp, unsigned char *b case 0x01: // Unknown if (!Code[1]) { - if (!(Code[1] = session->AddEventList(cpapcode, EVL_Event, 0.1))) { return false; } + if (!(Code[1] = session->AddEventList(cpapcode, EVL_Event, 0.1F))) { return false; } } Code[1]->AddEvent(t, 0); @@ -1176,7 +1175,7 @@ bool PRS1Loader::Parse002(qint32 sequence, quint32 timestamp, unsigned char *buf return false; } - unsigned char code; + unsigned char code=0; EventList *Code[0x20] = {0}; EventDataType data[10]; int cnt = 0; @@ -1263,7 +1262,7 @@ bool PRS1Loader::Parse002(qint32 sequence, quint32 timestamp, unsigned char *buf if (family == 0 && familyVersion >= 4) { if (!PRESSURE) { - PRESSURE = session->AddEventList(CPAP_Pressure, EVL_Event, 0.1); + PRESSURE = session->AddEventList(CPAP_Pressure, EVL_Event, 0.1F); if (!PRESSURE) { return false; } } @@ -1276,11 +1275,11 @@ bool PRS1Loader::Parse002(qint32 sequence, quint32 timestamp, unsigned char *buf case 0x02: // Pressure if (family == 0 && familyVersion >= 4) { // BiPAP Pressure if (!EPAP) { - if (!(EPAP = session->AddEventList(CPAP_EPAP, EVL_Event, 0.1))) { return false; } + if (!(EPAP = session->AddEventList(CPAP_EPAP, EVL_Event, 0.1F))) { return false; } - if (!(IPAP = session->AddEventList(CPAP_IPAP, EVL_Event, 0.1))) { return false; } + if (!(IPAP = session->AddEventList(CPAP_IPAP, EVL_Event, 0.1F))) { return false; } - if (!(PS = session->AddEventList(CPAP_PS, EVL_Event, 0.1))) { return false; } + if (!(PS = session->AddEventList(CPAP_PS, EVL_Event, 0.1F))) { return false; } } EPAP->AddEvent(t, data[0] = buffer[pos++]); @@ -1288,7 +1287,7 @@ bool PRS1Loader::Parse002(qint32 sequence, quint32 timestamp, unsigned char *buf PS->AddEvent(t, data[1] - data[0]); } else { if (!PRESSURE) { - PRESSURE = session->AddEventList(CPAP_Pressure, EVL_Event, 0.1); + PRESSURE = session->AddEventList(CPAP_Pressure, EVL_Event, 0.1F); if (!PRESSURE) { return false; } } @@ -1300,11 +1299,11 @@ bool PRS1Loader::Parse002(qint32 sequence, quint32 timestamp, unsigned char *buf case 0x03: // BIPAP Pressure if (!EPAP) { - if (!(EPAP = session->AddEventList(CPAP_EPAP, EVL_Event, 0.1))) { return false; } + if (!(EPAP = session->AddEventList(CPAP_EPAP, EVL_Event, 0.1F))) { return false; } - if (!(IPAP = session->AddEventList(CPAP_IPAP, EVL_Event, 0.1))) { return false; } + if (!(IPAP = session->AddEventList(CPAP_IPAP, EVL_Event, 0.1F))) { return false; } - if (!(PS = session->AddEventList(CPAP_PS, EVL_Event, 0.1))) { return false; } + if (!(PS = session->AddEventList(CPAP_PS, EVL_Event, 0.1F))) { return false; } } EPAP->AddEvent(t, data[0] = buffer[pos++]); @@ -1663,11 +1662,6 @@ bool PRS1Loader::OpenFile(Machine *mach, QString filename) bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename) { - - if (sid == 1532) { - int i=5; - } - Session *session = new_sessions[sid]; //int sequence,seconds,br,htype,version,numsignals; QFile file(filename); diff --git a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp index 86889986..36a7d673 100644 --- a/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/resmed_loader.cpp @@ -149,7 +149,7 @@ qint16 EDFParser::Read16() QString EDFParser::Read(unsigned n) { - if ((pos + n) > filesize) { + if ((pos + long(n)) > filesize) { return ""; } @@ -2262,12 +2262,14 @@ bool ResmedLoader::LoadSAD(Session *sess, EDFParser &edf) } if (!hasdata) continue; - if (matchSignal(code = OXI_Pulse, es.label)) { + if (matchSignal(OXI_Pulse, es.label)) { + code = OXI_Pulse; ToTimeDelta(sess, edf, es, code, recs, duration); sess->setPhysMax(code, 180); sess->setPhysMin(code, 18); - } else if (matchSignal(code = OXI_SPO2, es.label)) { + } else if (matchSignal(OXI_SPO2, es.label)) { + code = OXI_SPO2; es.physical_minimum = 60; ToTimeDelta(sess, edf, es, code, recs, duration); sess->setPhysMax(code, 100); @@ -2307,30 +2309,37 @@ bool ResmedLoader::LoadPLD(Session *sess, EDFParser &edf) rate = double(duration) / double(recs); //qDebug() << "EVE:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum << es.gain; - if (matchSignal(code = CPAP_Snore, es.label)) { + if (matchSignal(CPAP_Snore, es.label)) { + code = CPAP_Snore; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (matchSignal(code = CPAP_Pressure, es.label)) { + } else if (matchSignal(CPAP_Pressure, es.label)) { + code = CPAP_Pressure; es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (matchSignal(code = CPAP_IPAP, es.label)) { + } else if (matchSignal(CPAP_IPAP, es.label)) { + code = CPAP_IPAP; sess->settings[CPAP_Mode] = MODE_BIPAP; es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (matchSignal(code = CPAP_MinuteVent,es.label)) { + } else if (matchSignal(CPAP_MinuteVent,es.label)) { + code = CPAP_MinuteVent; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (matchSignal(code = CPAP_RespRate, es.label)) { + } else if (matchSignal(CPAP_RespRate, es.label)) { + code = CPAP_RespRate; a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); - } else if (matchSignal(code = CPAP_TidalVolume, es.label)) { + } else if (matchSignal(CPAP_TidalVolume, es.label)) { + code = CPAP_TidalVolume; es.gain *= 1000.0; es.physical_maximum *= 1000.0; es.physical_minimum *= 1000.0; // es.digital_maximum*=1000.0; // es.digital_minimum*=1000.0; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (matchSignal(code = CPAP_Leak, es.label)) { + } else if (matchSignal(CPAP_Leak, es.label)) { + code = CPAP_Leak; es.gain *= 60.0; es.physical_maximum *= 60.0; es.physical_minimum *= 60.0; @@ -2340,31 +2349,38 @@ bool ResmedLoader::LoadPLD(Session *sess, EDFParser &edf) ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0, true); sess->setPhysMax(code, 120.0); sess->setPhysMin(code, 0); - } else if (matchSignal(code = CPAP_FLG, es.label)) { + } else if (matchSignal(CPAP_FLG, es.label)) { + code = CPAP_FLG; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (matchSignal(code = CPAP_MaskPressure, es.label)) { + } else if (matchSignal(CPAP_MaskPressure, es.label)) { + code = CPAP_MaskPressure; es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (matchSignal(code = CPAP_EPAP, es.label)) { // Expiratory Pressure + } else if (matchSignal(CPAP_EPAP, es.label)) { // Expiratory Pressure + code = CPAP_EPAP; es.physical_maximum = 25; es.physical_minimum = 4; ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0); - } else if (matchSignal(code = CPAP_IE, es.label)) { //I:E ratio + } else if (matchSignal(CPAP_IE, es.label)) { //I:E ratio + code = CPAP_IE; a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); //a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0); - } else if (matchSignal(code = CPAP_Ti, es.label)) { + } else if (matchSignal(CPAP_Ti, es.label)) { + code = CPAP_Ti; a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); //a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0); - } else if (matchSignal(code = CPAP_Te, es.label)) { + } else if (matchSignal(CPAP_Te, es.label)) { + code = CPAP_Te; a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); //a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0); - } else if (matchSignal(code = CPAP_TgMV, es.label)) { + } else if (matchSignal(CPAP_TgMV, es.label)) { + code = CPAP_TgMV; a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate); a->AddWaveform(edf.startdate, es.data, recs, duration); //a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0); diff --git a/sleepyhead/SleepLib/machine.cpp b/sleepyhead/SleepLib/machine.cpp index 87c6f1c6..b1ed4d22 100644 --- a/sleepyhead/SleepLib/machine.cpp +++ b/sleepyhead/SleepLib/machine.cpp @@ -221,6 +221,7 @@ bool Machine::Purge(int secret) // Boring api key to stop this function getting called by accident :) if (secret != 3478216) { return false; } + // FIXME: should really clear out the majority of the machine properties here // It would be joyous if this function screwed up.. @@ -266,6 +267,9 @@ bool Machine::Purge(int secret) // qWarning() << "Could not purge path\n" << path << "\n\n" << could_not_kill << " file(s) remain.. Suggest manually deleting this path\n"; // return false; } + // PROFILE.machlist.erase(PROFILE.machlist.find(m->id())); + + PROFILE.p_preferences[STR_PREF_ReimportBackup] = true; return true; } diff --git a/sleepyhead/SleepLib/profiles.cpp b/sleepyhead/SleepLib/profiles.cpp index ca59f17c..05f9797e 100644 --- a/sleepyhead/SleepLib/profiles.cpp +++ b/sleepyhead/SleepLib/profiles.cpp @@ -119,17 +119,23 @@ void Profile::DataFormatError(Machine *m) msg = msg + QObject::tr("Would you like me to purge this data this for you so you can run the new version?"); - if (QMessageBox::warning(nullptr, QObject::tr("Machine Database Changes"), msg, + if (QMessageBox::warning(nullptr, + QObject::tr("Machine Database Changes"), + msg, QMessageBox::Yes | QMessageBox::Cancel, QMessageBox::Yes) == QMessageBox::Yes) { - if (!m->Purge( - 3478216)) { // Do not copy this line without thinking.. You will be eaten by a Grue if you do + + if (!m->Purge(3478216)) { + // Do not copy this line without thinking.. You will be eaten by a Grue if you do QMessageBox::critical(nullptr, QObject::tr("Purge Failed"), QObject::tr("Sorry, I could not purge this data, which means this version of SleepyHead can't start.. SleepyHead's Data folder needs to be removed manually\n\nThis folder currently resides at the following location:\n") + PREF[STR_GEN_DataFolder].toString(), QMessageBox::Ok); QApplication::exit(-1); } + // Note: I deliberately haven't added a Profile help for this + PROFILE.p_preferences[STR_PREF_ReimportBackup] = true; + PROFILE.Save(); } else { QApplication::exit(-1); } @@ -167,6 +173,7 @@ void Profile::LoadMachineData() try { m->Load(); } catch (OldDBVersion e) { + Q_UNUSED(e) DataFormatError(m); } } diff --git a/sleepyhead/SleepLib/profiles.h b/sleepyhead/SleepLib/profiles.h index cad18a71..b01126a7 100644 --- a/sleepyhead/SleepLib/profiles.h +++ b/sleepyhead/SleepLib/profiles.h @@ -244,6 +244,7 @@ const QString STR_AS_OverviewLinechartMode = "OverviewLinechartMode"; const QString STR_AS_UsePixmapCaching = "UsePixmapCaching"; const QString STR_AS_AllowYAxisScaling = "AllowYAxisScaling"; const QString STR_AS_GraphTooltips = "GraphTooltips"; +const QString STR_AS_LineThickness = "LineThickness"; // UserSettings Strings const QString STR_US_UnitSystem = "UnitSystem"; @@ -568,6 +569,7 @@ class AppearanceSettings : public ProfileSettings initPref(STR_AS_UsePixmapCaching, true); initPref(STR_AS_OverlayType, ODT_Bars); initPref(STR_AS_OverviewLinechartMode, OLC_Bartop); + initPref(STR_AS_LineThickness, 1.0); } //! \brief Returns the normal (unscaled) height of a graph @@ -586,6 +588,10 @@ class AppearanceSettings : public ProfileSettings bool allowYAxisScaling() const { return getPref(STR_AS_AllowYAxisScaling).toBool(); } //! \brief Whether to show graph tooltips bool graphTooltips() const { return getPref(STR_AS_GraphTooltips).toBool(); } + //! \brief Pen width of line plots + float lineThickness() const { return getPref(STR_AS_LineThickness).toFloat(); } + + //! \brief Returns the type of overlay flags (which are displayed over the Flow Waveform) OverlayDisplayType overlayType() const { return (OverlayDisplayType)getPref(STR_AS_OverlayType).toInt(); @@ -617,6 +623,8 @@ class AppearanceSettings : public ProfileSettings void setOverviewLinechartMode(OverviewLinechartModes od) { setPref(STR_AS_OverviewLinechartMode, (int)od); } + //! \brief Set the pen width of line plots. + void setLineThickness(float size) { setPref(STR_AS_LineThickness, size); } }; /*! \class UserSettings diff --git a/sleepyhead/SleepLib/session.cpp b/sleepyhead/SleepLib/session.cpp index 4efb5856..c294d6ae 100644 --- a/sleepyhead/SleepLib/session.cpp +++ b/sleepyhead/SleepLib/session.cpp @@ -1057,7 +1057,7 @@ EventDataType Session::physMin(ChannelID id) return 0; } - EventDataType min = round(Min(id)); + EventDataType min = floor(Min(id)); m_physmin[id] = min; return min; } @@ -1077,7 +1077,7 @@ EventDataType Session::physMax(ChannelID id) return 0; } - EventDataType max = round(Max(id) + 0.5); + EventDataType max = ceil(Max(id) + 0.5); m_physmax[id] = max; return max; } diff --git a/sleepyhead/common_gui.cpp b/sleepyhead/common_gui.cpp index 7b87a2fb..8bca9837 100644 --- a/sleepyhead/common_gui.cpp +++ b/sleepyhead/common_gui.cpp @@ -74,7 +74,7 @@ QColor COLOR_RERA = COLOR_Gold; QColor COLOR_VibratorySnore = QColor("#ff4040"); QColor COLOR_FlowLimit = QColor("#404040"); QColor COLOR_LeakFlag = QColor("#40c0c0"); // Qt::darkBlue; -QColor COLOR_NRI = Qt::darkMagenta; +QColor COLOR_NRI = COLOR_ClearAirway; QColor COLOR_ExP = Qt::darkCyan; QColor COLOR_PressurePulse = Qt::red; QColor COLOR_PulseChange = COLOR_LightGray; diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index 7024e9c8..c8f68d85 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -46,6 +46,12 @@ //extern QProgressBar *qprogress; extern MainWindow * mainwin; + +// This was Sean Stangl's idea.. but I couldn't apply that patch. +inline QString channelInfo(ChannelID code) { + return schema::channel[code].fullname()+"\n"+schema::channel[code].description()+"\n("+schema::channel[code].units()+")"; +} + Daily::Daily(QWidget *parent,gGraphView * shared) :QWidget(parent), ui(new Ui::Daily) { @@ -121,42 +127,50 @@ Daily::Daily(QWidget *parent,gGraphView * shared) layout->addWidget(scrollbar,0); int default_height=PROFILE.appearance->graphHeight(); - SF=new gGraph(GraphView,STR_TR_EventFlags,STR_TR_EventFlags,default_height); + gGraph *GAHI = nullptr, +// *TAP = nullptr, + *SF = nullptr, + *AHI = nullptr; + + graphlist["SF"] = SF = new gGraph(GraphView,STR_TR_EventFlags,STR_TR_EventFlags,default_height); SF->setPinned(true); - FRW=new gGraph(GraphView,STR_TR_FlowRate, schema::channel[CPAP_FlowRate].fullname()+"\n"+schema::channel[CPAP_FlowRate].description()+"\n("+schema::channel[CPAP_FlowRate].units()+")",default_height); - //FRW->setPinned(true); + ChannelID cpapcodes[] = { + CPAP_FlowRate, CPAP_MaskPressure, CPAP_Pressure, CPAP_Leak, CPAP_Snore, CPAP_RespRate, + CPAP_TidalVolume, CPAP_MinuteVent, CPAP_FLG, CPAP_PTB, CPAP_RespEvent, CPAP_Ti, CPAP_Te, + CPAP_IE, ZEO_SleepStage, POS_Inclination, POS_Orientation + }; + + int cpapsize = sizeof(cpapcodes) / sizeof(ChannelID); + + ChannelID oxicodes[] = { + OXI_Pulse, OXI_SPO2, OXI_Plethy + }; + int oxisize = sizeof(oxicodes) / sizeof(ChannelID); + + + for (int i=0; i < cpapsize; ++i) { + ChannelID code = cpapcodes[i]; + graphlist[schema::channel[code].label()] = new gGraph(GraphView, schema::channel[code].label(), channelInfo(code), default_height); + } + + int oxigrp=PROFILE.ExistsAndTrue("SyncOximetry") ? 0 : 1; // Contemplating killing this setting... + for (int i=0; i < oxisize; ++i) { + ChannelID code = oxicodes[i]; + graphlist[schema::channel[code].label()] = new gGraph(GraphView, schema::channel[code].label(), channelInfo(code), default_height, oxigrp); + } if (PROFILE.general->calculateRDI()) { - AHI=new gGraph(GraphView,STR_TR_RDI, schema::channel[CPAP_RDI].fullname()+"\n"+schema::channel[CPAP_RDI].description()+"\n("+schema::channel[CPAP_RDI].units()+")",default_height); - } else AHI=new gGraph(GraphView,STR_TR_AHI,schema::channel[CPAP_AHI].fullname()+"\n"+schema::channel[CPAP_AHI].description()+"\n("+schema::channel[CPAP_AHI].units()+")",default_height); + AHI=new gGraph(GraphView,STR_TR_RDI, channelInfo(CPAP_RDI), default_height); + } else { + AHI=new gGraph(GraphView,STR_TR_AHI, channelInfo(CPAP_AHI), default_height); + } - MP=new gGraph(GraphView,schema::channel[CPAP_MaskPressure].label(), schema::channel[CPAP_MaskPressure].fullname()+"\n"+schema::channel[CPAP_MaskPressure].description()+"\n("+schema::channel[CPAP_MaskPressure].units()+")",default_height); - PRD=new gGraph(GraphView,schema::channel[CPAP_Pressure].label(), schema::channel[CPAP_Pressure].fullname()+"\n"+schema::channel[CPAP_Pressure].description()+"\n("+schema::channel[CPAP_Pressure].units()+")",default_height); - LEAK=new gGraph(GraphView,STR_TR_Leak, schema::channel[CPAP_Leak].fullname()+"\n"+schema::channel[CPAP_Leak].description()+"\n("+schema::channel[CPAP_Leak].units()+")",default_height); - SNORE=new gGraph(GraphView,STR_TR_Snore, schema::channel[CPAP_Snore].fullname()+"\n"+schema::channel[CPAP_Snore].description()+"\n("+schema::channel[CPAP_Snore].units()+")",default_height); - RR=new gGraph(GraphView,STR_TR_RespRate, schema::channel[CPAP_RespRate].fullname()+"\n"+schema::channel[CPAP_RespRate].description()+"\n("+schema::channel[CPAP_RespRate].units()+")",default_height); - TV=new gGraph(GraphView,STR_TR_TidalVolume, schema::channel[CPAP_TidalVolume].fullname()+"\n"+schema::channel[CPAP_TidalVolume].description()+"\n("+schema::channel[CPAP_TidalVolume].units()+")",default_height); - MV=new gGraph(GraphView,STR_TR_MinuteVent, schema::channel[CPAP_MinuteVent].fullname()+"\n"+schema::channel[CPAP_MinuteVent].description()+"\n("+schema::channel[CPAP_MinuteVent].units()+")",default_height); - //TgMV=new gGraph(GraphView,STR_TR_TgtMinVent, schema::channel[CPAP_TgMV].fullname()+"\n"+schema::channel[CPAP_TgMV].description()+"\n("+schema::channel[CPAP_TgMV].units()+")",default_height); - FLG=new gGraph(GraphView,STR_TR_FlowLimit, schema::channel[CPAP_FLG].fullname()+"\n"+schema::channel[CPAP_FLG].description()+"\n("+schema::channel[CPAP_FLG].units()+")",default_height); - PTB=new gGraph(GraphView,STR_TR_PatTrigBreath, schema::channel[CPAP_PTB].fullname()+"\n"+schema::channel[CPAP_PTB].description()+"\n("+schema::channel[CPAP_PTB].units()+")",default_height); - RE=new gGraph(GraphView,STR_TR_RespEvent, schema::channel[CPAP_RespEvent].fullname()+"\n"+schema::channel[CPAP_RespEvent].description()+"\n("+schema::channel[CPAP_RespEvent].units()+")",default_height); - TI=new gGraph(GraphView,STR_TR_InspTime, schema::channel[CPAP_Ti].fullname()+"\n"+schema::channel[CPAP_Ti].description()+"\n("+schema::channel[CPAP_Ti].units()+")",default_height); - TE=new gGraph(GraphView,STR_TR_ExpTime, schema::channel[CPAP_Te].fullname()+"\n"+schema::channel[CPAP_Te].description()+"\n("+schema::channel[CPAP_Te].units()+")",default_height); - IE=new gGraph(GraphView,schema::channel[CPAP_IE].label(), schema::channel[CPAP_IE].fullname()+"\n"+schema::channel[CPAP_IE].description()+"\n("+schema::channel[CPAP_IE].units()+")",default_height); + graphlist["AHI"] = AHI; - STAGE=new gGraph(GraphView,STR_TR_SleepStage, schema::channel[ZEO_SleepStage].fullname()+"\n"+schema::channel[ZEO_SleepStage].description()+"\n("+schema::channel[ZEO_SleepStage].units()+")",default_height); - int oxigrp=PROFILE.ExistsAndTrue("SyncOximetry") ? 0 : 1; - PULSE=new gGraph(GraphView,STR_TR_PulseRate, schema::channel[OXI_Pulse].fullname()+"\n"+schema::channel[OXI_Pulse].description()+"\n("+schema::channel[OXI_Pulse].units()+")",default_height,oxigrp); - SPO2=new gGraph(GraphView,STR_TR_SpO2, schema::channel[OXI_SPO2].fullname()+"\n"+schema::channel[OXI_SPO2].description()+"\n("+schema::channel[OXI_SPO2].units()+")",default_height,oxigrp); - INTPULSE=new gGraph(GraphView,tr("Int. Pulse"), schema::channel[OXI_Pulse].fullname()+"\n"+schema::channel[OXI_Pulse].description()+"\n("+schema::channel[OXI_Pulse].units()+")",default_height,oxigrp); - INTSPO2=new gGraph(GraphView,tr("Int. SpO2"), schema::channel[OXI_SPO2].fullname()+"\n"+schema::channel[OXI_SPO2].description()+"\n("+schema::channel[OXI_SPO2].units()+")",default_height,oxigrp); - PLETHY=new gGraph(GraphView,STR_TR_Plethy, schema::channel[OXI_Plethy].fullname()+"\n"+schema::channel[OXI_Plethy].description()+"\n("+schema::channel[OXI_Plethy].units()+")",default_height,oxigrp); - - INC=new gGraph(GraphView,STR_TR_Inclination, schema::channel[POS_Inclination].fullname()+"\n"+schema::channel[POS_Inclination].description()+"\n("+schema::channel[POS_Inclination].units()+")",default_height); - ORI=new gGraph(GraphView,STR_TR_Orientation, schema::channel[POS_Orientation].fullname()+"\n"+schema::channel[POS_Orientation].description()+"\n("+schema::channel[POS_Orientation].units()+")",default_height); + graphlist["INTPULSE"] = new gGraph(GraphView,tr("Int. Pulse"), channelInfo(OXI_Pulse), default_height, oxigrp); + graphlist["INTSPO2"] = new gGraph(GraphView,tr("Int. SpO2"), channelInfo(OXI_SPO2), default_height, oxigrp); // Event Pie Chart (for snapshot purposes) // TODO: Convert snapGV to generic for snapshotting multiple graphs (like reports does) @@ -166,8 +180,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared) // TAP->AddLayer(AddCPAP(tap)); //TAP->setMargins(0,0,0,0); - - GAHI=new gGraph(snapGV,tr("Breakdown"),tr("events"),172); + graphlist["EventBreakdown"] = GAHI = new gGraph(snapGV,tr("Breakdown"),tr("events"),172); gSegmentChart * evseg=new gSegmentChart(GST_Pie); evseg->AddSlice(CPAP_Hypopnea,QColor(0x40,0x40,0xff,0xff),STR_TR_H); evseg->AddSlice(CPAP_Apnea,QColor(0x20,0x80,0x20,0xff),STR_TR_UA); @@ -180,13 +193,13 @@ Daily::Daily(QWidget *parent,gGraphView * shared) GAHI->AddLayer(AddCPAP(evseg)); GAHI->setMargins(0,0,0,0); - //SF->AddLayer(AddCPAP(evseg),LayerRight,100); gFlagsGroup *fg=new gFlagsGroup(); SF->AddLayer(AddCPAP(fg)); + // Spans fg->AddLayer((new gFlagsLine(CPAP_CSR, COLOR_CSR, STR_TR_PB, false, FT_Span))); fg->AddLayer((new gFlagsLine(PRS1_10, COLOR_LargeLeak, STR_TR_LL, false, FT_Span))); - + // Flags fg->AddLayer((new gFlagsLine(CPAP_ClearAirway, COLOR_ClearAirway, STR_TR_CA,false))); fg->AddLayer((new gFlagsLine(CPAP_Obstructive, COLOR_Obstructive, STR_TR_OA,true))); fg->AddLayer((new gFlagsLine(CPAP_Apnea, COLOR_Apnea, STR_TR_UA))); @@ -212,25 +225,45 @@ Daily::Daily(QWidget *parent,gGraphView * shared) SF->AddLayer(new gXAxis(COLOR_Text,false),LayerBottom,0,20); //gXAxis::Margin); + // The following list contains graphs that don't have standard xgrid/yaxis labels + QStringList skipgraph; + skipgraph.push_back("EventBreakdown"); + skipgraph.push_back("SF"); + + QHash::iterator it; + + for (it = graphlist.begin(); it != graphlist.end(); ++it) { + if (skipgraph.contains(it.key())) continue; + it.value()->AddLayer(new gXGrid()); + } + + gLineChart *l; l=new gLineChart(CPAP_FlowRate,COLOR_Black,false,false); gLineOverlaySummary *los=new gLineOverlaySummary(tr("Selection AHI"),5,-4); AddCPAP(l); - FRW->AddLayer(new gXGrid()); + + gGraph *FRW = graphlist[schema::channel[CPAP_FlowRate].label()]; + + // Draw layer is important... spans first.. FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_CSR, COLOR_CSR, STR_TR_CSR, FT_Span))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_10, COLOR_LargeLeak, STR_TR_LL, FT_Span))); + + // Then the graph itself FRW->AddLayer(l); - FRW->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); - FRW->AddLayer(new gXAxis(),LayerBottom,0,20); + + // Then the LineOverlaySummaries FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Hypopnea,COLOR_Hypopnea,STR_TR_H)))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_PressurePulse,COLOR_PressurePulse,STR_TR_PP,FT_Dot))); //FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_Pressure, COLOR_White,STR_TR_P,FT_Dot))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0B,COLOR_Blue,"0B",FT_Dot))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(PRS1_0E,COLOR_DarkRed,"0E",FT_Dot))); + + gLineOverlayBar * rera = new gLineOverlayBar(CPAP_RERA, COLOR_RERA, STR_TR_RE); if (PROFILE.general->calculateRDI()) { - FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_RERA, COLOR_RERA, STR_TR_RE)))); + FRW->AddLayer(AddCPAP(los->add(rera))); } else { - FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_RERA, COLOR_RERA, STR_TR_RE))); + FRW->AddLayer(AddCPAP(rera)); } FRW->AddLayer(AddCPAP(los->add(new gLineOverlayBar(CPAP_Apnea, COLOR_Apnea, STR_TR_UA)))); FRW->AddLayer(AddCPAP(new gLineOverlayBar(CPAP_VSnore, COLOR_VibratorySnore, STR_TR_VS))); @@ -249,31 +282,9 @@ Daily::Daily(QWidget *parent,gGraphView * shared) FRW->AddLayer(AddCPAP(los)); - - - gGraph *graphs[]={ PRD, LEAK, AHI, SNORE, PTB, MP, RR, MV, TV, FLG, IE, TI, TE, SPO2, PLETHY, PULSE, STAGE, INTSPO2, INTPULSE, ORI, INC }; - int ng=sizeof(graphs)/sizeof(gGraph*); - for (int i=0;iAddLayer(new gXGrid()); - } - /*PRD->AddLayer(AddCPAP(new gStatsLine(CPAP_Pressure,"Pressure")),LayerBottom,0,20,1); - PRD->AddLayer(AddCPAP(new gStatsLine(CPAP_EPAP,"EPAP")),LayerBottom,0,20,1); - PRD->AddLayer(AddCPAP(new gStatsLine(CPAP_IPAP,"IPAP")),LayerBottom,0,20,1); - LEAK->AddLayer(AddCPAP(new gStatsLine(CPAP_Leak)),LayerBottom,0,20,1); - SNORE->AddLayer(AddCPAP(new gStatsLine(CPAP_Snore)),LayerBottom,0,20,1); - PTB->AddLayer(AddCPAP(new gStatsLine(CPAP_PatientTriggeredBreaths)),LayerBottom,0,20,1); - RR->AddLayer(AddCPAP(new gStatsLine(CPAP_RespiratoryRate)),LayerBottom,0,20,1); - MV->AddLayer(AddCPAP(new gStatsLine(CPAP_MinuteVentilation)),LayerBottom,0,20,1); - TV->AddLayer(AddCPAP(new gStatsLine(CPAP_TidalVolume)),LayerBottom,0,20,1); - FLG->AddLayer(AddCPAP(new gStatsLine(CPAP_FlowLimitGraph)),LayerBottom,0,20,1); - IE->AddLayer(AddCPAP(new gStatsLine(CPAP_IE)),LayerBottom,0,20,1); - TE->AddLayer(AddCPAP(new gStatsLine(CPAP_Te)),LayerBottom,0,20,1); - TI->AddLayer(AddCPAP(new gStatsLine(CPAP_Ti)),LayerBottom,0,20,1); */ - - bool square=PROFILE.appearance->squareWavePlots(); gLineChart *pc=new gLineChart(CPAP_Pressure, COLOR_Pressure, square); - PRD->AddLayer(AddCPAP(pc)); + graphlist[schema::channel[CPAP_Pressure].label()]->AddLayer(AddCPAP(pc)); pc->addPlot(CPAP_EPAP, COLOR_EPAP, square); pc->addPlot(CPAP_IPAPLo, COLOR_IPAPLo, square); pc->addPlot(CPAP_IPAP, COLOR_IPAP, square); @@ -289,68 +300,64 @@ Daily::Daily(QWidget *parent,gGraphView * shared) gLineChart *lc=new gLineChart(CPAP_LeakTotal, COLOR_LeakTotal, square); lc->addPlot(CPAP_Leak, COLOR_Leak, square); lc->addPlot(CPAP_MaxLeak, COLOR_MaxLeak, square); - LEAK->AddLayer(AddCPAP(lc)); + graphlist[schema::channel[CPAP_Leak].label()]->AddLayer(AddCPAP(lc)); //LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_Leak, COLOR_Leak,square))); //LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_MaxLeak, COLOR_MaxLeak,square))); - SNORE->AddLayer(AddCPAP(new gLineChart(CPAP_Snore, COLOR_Snore, true))); + graphlist[schema::channel[CPAP_Snore].label()]->AddLayer(AddCPAP(new gLineChart(CPAP_Snore, COLOR_Snore, true))); - PTB->AddLayer(AddCPAP(new gLineChart(CPAP_PTB, COLOR_PTB, square))); - MP->AddLayer(AddCPAP(new gLineChart(CPAP_MaskPressure, COLOR_MaskPressure, false))); - RR->AddLayer(AddCPAP(lc=new gLineChart(CPAP_RespRate, COLOR_RespRate, square))); + graphlist[schema::channel[CPAP_PTB].label()]->AddLayer(AddCPAP(new gLineChart(CPAP_PTB, COLOR_PTB, square))); + graphlist[schema::channel[CPAP_MaskPressure].label()]->AddLayer(AddCPAP(new gLineChart(CPAP_MaskPressure, COLOR_MaskPressure, false))); + graphlist[schema::channel[CPAP_RespRate].label()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_RespRate, COLOR_RespRate, square))); - INC->AddLayer(AddPOS(new gLineChart(POS_Inclination))); - ORI->AddLayer(AddPOS(new gLineChart(POS_Orientation))); + graphlist[schema::channel[POS_Inclination].label()]->AddLayer(AddPOS(new gLineChart(POS_Inclination))); + graphlist[schema::channel[POS_Orientation].label()]->AddLayer(AddPOS(new gLineChart(POS_Orientation))); - // Delete me!! -// lc->addPlot(CPAP_Test1, COLOR_DarkRed,square); - - MV->AddLayer(AddCPAP(lc=new gLineChart(CPAP_MinuteVent, COLOR_MinuteVent, square))); + graphlist[schema::channel[CPAP_MinuteVent].label()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_MinuteVent, COLOR_MinuteVent, square))); lc->addPlot(CPAP_TgMV,COLOR_TgMV,square); - TV->AddLayer(AddCPAP(lc=new gLineChart(CPAP_TidalVolume,COLOR_TidalVolume,square))); + graphlist[schema::channel[CPAP_TidalVolume].label()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_TidalVolume,COLOR_TidalVolume,square))); //lc->addPlot(CPAP_Test2,COLOR_DarkYellow,square); - - - //TV->AddLayer(AddCPAP(new gLineChart("TidalVolume2",COLOR_Magenta,square))); - FLG->AddLayer(AddCPAP(new gLineChart(CPAP_FLG, COLOR_FLG, true))); - //RE->AddLayer(AddCPAP(new gLineChart(CPAP_RespiratoryEvent,COLOR_Magenta,true))); - IE->AddLayer(AddCPAP(lc=new gLineChart(CPAP_IE, COLOR_IE, square))); - TE->AddLayer(AddCPAP(lc=new gLineChart(CPAP_Te, COLOR_Te, square))); - TI->AddLayer(AddCPAP(lc=new gLineChart(CPAP_Ti, COLOR_Ti, square))); + //graphlist[schema::channel[CPAP_TidalVolume].label()]->AddLayer(AddCPAP(new gLineChart("TidalVolume2",COLOR_Magenta,square))); + graphlist[schema::channel[CPAP_FLG].label()]->AddLayer(AddCPAP(new gLineChart(CPAP_FLG, COLOR_FLG, true))); + //graphlist[schema::channel[CPAP_RespiratoryEvent].label()]->AddLayer(AddCPAP(new gLineChart(CPAP_RespiratoryEvent,COLOR_Magenta,true))); + graphlist[schema::channel[CPAP_IE].label()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_IE, COLOR_IE, square))); + graphlist[schema::channel[CPAP_Te].label()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_Te, COLOR_Te, square))); + graphlist[schema::channel[CPAP_Ti].label()]->AddLayer(AddCPAP(lc=new gLineChart(CPAP_Ti, COLOR_Ti, square))); //lc->addPlot(CPAP_Test2,COLOR:DarkYellow,square); - STAGE->AddLayer(AddSTAGE(new gLineChart(ZEO_SleepStage, COLOR_SleepStage, true))); + graphlist[schema::channel[ZEO_SleepStage].label()]->AddLayer(AddSTAGE(new gLineChart(ZEO_SleepStage, COLOR_SleepStage, true))); gLineOverlaySummary *los1=new gLineOverlaySummary(tr("Events/hour"),5,-4); gLineOverlaySummary *los2=new gLineOverlaySummary(tr("Events/hour"),5,-4); - PULSE->AddLayer(AddOXI(los1->add(new gLineOverlayBar(OXI_PulseChange, COLOR_PulseChange, STR_TR_PC,FT_Span)))); - PULSE->AddLayer(AddOXI(los1)); - SPO2->AddLayer(AddOXI(los2->add(new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2,FT_Span)))); - SPO2->AddLayer(AddOXI(los2)); + graphlist[schema::channel[OXI_Pulse].label()]->AddLayer(AddOXI(los1->add(new gLineOverlayBar(OXI_PulseChange, COLOR_PulseChange, STR_TR_PC,FT_Span)))); + graphlist[schema::channel[OXI_Pulse].label()]->AddLayer(AddOXI(los1)); + graphlist[schema::channel[OXI_SPO2].label()]->AddLayer(AddOXI(los2->add(new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2,FT_Span)))); + graphlist[schema::channel[OXI_SPO2].label()]->AddLayer(AddOXI(los2)); - PULSE->AddLayer(AddOXI(new gLineChart(OXI_Pulse, COLOR_Pulse, square))); - SPO2->AddLayer(AddOXI(new gLineChart(OXI_SPO2, COLOR_SPO2, true))); - PLETHY->AddLayer(AddOXI(new gLineChart(OXI_Plethy, COLOR_Plethy,false))); + graphlist[schema::channel[OXI_Pulse].label()]->AddLayer(AddOXI(new gLineChart(OXI_Pulse, COLOR_Pulse, square))); + graphlist[schema::channel[OXI_SPO2].label()]->AddLayer(AddOXI(new gLineChart(OXI_SPO2, COLOR_SPO2, true))); + graphlist[schema::channel[OXI_Plethy].label()]->AddLayer(AddOXI(new gLineChart(OXI_Plethy, COLOR_Plethy,false))); + + // Fix me gLineOverlaySummary *los3=new gLineOverlaySummary(tr("Events/hour"),5,-4); + graphlist["INTPULSE"]->AddLayer(AddCPAP(los3->add(new gLineOverlayBar(OXI_PulseChange, COLOR_PulseChange, STR_TR_PC,FT_Span)))); + graphlist["INTPULSE"]->AddLayer(AddCPAP(los3)); + graphlist["INTPULSE"]->AddLayer(AddCPAP(new gLineChart(OXI_Pulse, COLOR_Pulse, square))); gLineOverlaySummary *los4=new gLineOverlaySummary(tr("Events/hour"),5,-4); + graphlist["INTSPO2"]->AddLayer(AddCPAP(los4->add(new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2,FT_Span)))); + graphlist["INTSPO2"]->AddLayer(AddCPAP(los4)); + graphlist["INTSPO2"]->AddLayer(AddCPAP(new gLineChart(OXI_SPO2, COLOR_SPO2, true))); - INTPULSE->AddLayer(AddCPAP(los3->add(new gLineOverlayBar(OXI_PulseChange, COLOR_PulseChange, STR_TR_PC,FT_Span)))); - INTPULSE->AddLayer(AddCPAP(los3)); - INTSPO2->AddLayer(AddCPAP(los4->add(new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2,FT_Span)))); - INTSPO2->AddLayer(AddCPAP(los4)); - INTPULSE->AddLayer(AddCPAP(new gLineChart(OXI_Pulse, COLOR_Pulse, square))); - INTSPO2->AddLayer(AddCPAP(new gLineChart(OXI_SPO2, COLOR_SPO2, true))); + graphlist[schema::channel[CPAP_PTB].label()]->setForceMaxY(100); + graphlist[schema::channel[OXI_SPO2].label()]->setForceMaxY(100); + for (it = graphlist.begin(); it != graphlist.end(); ++it) { + if (skipgraph.contains(it.key())) continue; - PTB->setForceMaxY(100); - SPO2->setForceMaxY(100); - INTSPO2->setForceMaxY(100); - - for (int i=0;iAddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); - graphs[i]->AddLayer(new gXAxis(),LayerBottom,0,20); + it.value()->AddLayer(new gYAxis(),LayerLeft,gYAxis::Margin); + it.value()->AddLayer(new gXAxis(),LayerBottom,0,20); } layout->layout(); @@ -396,15 +403,12 @@ Daily::Daily(QWidget *parent,gGraphView * shared) previous_date=QDate(); } + Daily::~Daily() { - GraphView->SaveSettings("Daily"); // disconnect(sessbar, SIGNAL(toggledSession(Session*)), this, SLOT(doToggleSession(Session*))); - disconnect(webView,SIGNAL(linkClicked(QUrl)),this,SLOT(Link_clicked(QUrl))); // Save any last minute changes.. - if (previous_date.isValid()) - Unload(previous_date); // delete splitter; delete ui; @@ -412,6 +416,17 @@ Daily::~Daily() delete icon_off; } +void Daily::closeEvent(QCloseEvent *event) +{ + disconnect(webView,SIGNAL(linkClicked(QUrl)),this,SLOT(Link_clicked(QUrl))); + + if (previous_date.isValid()) + Unload(previous_date); + GraphView->SaveSettings("Daily"); + QWidget::closeEvent(event); +} + + void Daily::doToggleSession(Session * sess) { Q_UNUSED(sess) @@ -900,9 +915,9 @@ QString Daily::getOximeterInformation(Day * oxi) html+=" "; html+=""+oxi->machine->properties[STR_PROP_Brand]+" "+oxi->machine->properties[STR_PROP_Model]+"\n"; html+=" "; - html+=QString("%1: %2 (%3)\%").arg(tr("SpO2 Desaturations")).arg(oxi->count(OXI_SPO2Drop)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_SPO2Drop)/3600.0),0,'f',2); - html+=QString("%1: %2 (%3)\%").arg(tr("Pulse Change events")).arg(oxi->count(OXI_PulseChange)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_PulseChange)/3600.0),0,'f',2); - html+=QString("%1: %2\%").arg(tr("SpO2 Baseline Used")).arg(oxi->settings_wavg(OXI_SPO2Drop),0,'f',2); // CHECKME: Should this value be wavg OXI_SPO2 isntead? + html+=QString("%1: %2 (%3)%%").arg(tr("SpO2 Desaturations")).arg(oxi->count(OXI_SPO2Drop)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_SPO2Drop)/3600.0),0,'f',2); + html+=QString("%1: %2 (%3)%%").arg(tr("Pulse Change events")).arg(oxi->count(OXI_PulseChange)).arg((100.0/oxi->hours()) * (oxi->sum(OXI_PulseChange)/3600.0),0,'f',2); + html+=QString("%1: %2%%").arg(tr("SpO2 Baseline Used")).arg(oxi->settings_wavg(OXI_SPO2Drop),0,'f',2); // CHECKME: Should this value be wavg OXI_SPO2 isntead? html+="\n"; html+="
\n"; } @@ -1315,11 +1330,11 @@ void Daily::Load(QDate date) if ((oai+hi+cai+uai+rei+fli)>0) { html+=" "; html+=QString("%1").arg(tr("Event Breakdown")); - GAHI->setShowTitle(false); + eventBreakdownPie()->setShowTitle(false); int w=155; int h=155; - QPixmap pixmap=GAHI->renderPixmap(w,h,false); + QPixmap pixmap=eventBreakdownPie()->renderPixmap(w,h,false); if (!pixmap.isNull()) { QByteArray byteArray; QBuffer buffer(&byteArray); // use buffer to store pixmap into byteArray diff --git a/sleepyhead/daily.h b/sleepyhead/daily.h index 7207a687..676eb0d8 100644 --- a/sleepyhead/daily.h +++ b/sleepyhead/daily.h @@ -79,6 +79,8 @@ public: */ explicit Daily(QWidget *parent, gGraphView *shared); ~Daily(); + void closeEvent(QCloseEvent *); + /*! \fn ReloadGraphs() \brief Reload all graph information from disk and updates the view. */ @@ -127,7 +129,7 @@ public: \brief Returns a pointer to the Event Breakdown Piechart for the Report Printing module \returns gGraph * object containing this chart */ - gGraph * eventBreakdownPie() { return GAHI; } + gGraph * eventBreakdownPie() { return graphlist["EventBreakdown"]; } QWidget * sessionBar() { return sessbar; } void clearLastDay(); @@ -303,9 +305,7 @@ private: QString getEventBreakdown(Day *cpap); QString getSleepTime(Day *cpap, Day *oxi); - gGraph *PRD,*FRW,*GAHI,*TAP,*LEAK,*SF,*TAP_EAP,*TAP_IAP,*PULSE,*SPO2, - *SNORE,*RR,*MP,*MV,*TV,*FLG,*PTB,*OF, *THPR, - *PLETHY,*TI,*TE, *RE, *IE, *AHI, *RDI, *STAGE, *INTPULSE, *INTSPO2, *INC, *ORI; + QHash graphlist; QList OXIData; QList CPAPData; @@ -326,7 +326,6 @@ private: void UpdateSTAGEGraphs(Day *day); void UpdatePOSGraphs(Day *day); - Ui::Daily *ui; QDate previous_date; QMenu *show_graph_menu; diff --git a/sleepyhead/main.cpp b/sleepyhead/main.cpp index 09ecfc96..a6f9075e 100644 --- a/sleepyhead/main.cpp +++ b/sleepyhead/main.cpp @@ -53,6 +53,7 @@ void MyOutputHandler(QtMsgType type, const char *msgtxt) #else void MyOutputHandler(QtMsgType type, const QMessageLogContext &context, const QString &msgtxt) { + Q_UNUSED(context) #endif if (!mainwin) { diff --git a/sleepyhead/mainwindow.cpp b/sleepyhead/mainwindow.cpp index 349eef5d..4d39fa70 100644 --- a/sleepyhead/mainwindow.cpp +++ b/sleepyhead/mainwindow.cpp @@ -31,6 +31,7 @@ #include #include #include +#include #include // Custom loaders that don't autoscan.. @@ -241,17 +242,20 @@ void MainWindow::closeEvent(QCloseEvent * event) { if (daily) { daily->close(); - delete daily; + daily->deleteLater(); +// delete daily; } if (overview) { overview->close(); - delete overview; + overview->deleteLater(); +// delete overview; } if (oximetry) { oximetry->close(); - delete oximetry; + oximetry->deleteLater(); +// delete oximetry; } // Shutdown and Save the current User profile @@ -266,9 +270,9 @@ void MainWindow::closeEvent(QCloseEvent * event) extern MainWindow *mainwin; MainWindow::~MainWindow() { - if (systraymenu) { delete systraymenu; } +// if (systraymenu) { delete systraymenu; } - if (systray) { delete systray; } +// if (systray) { delete systray; } // Trash anything allocated by the Graph objects DestroyGraphGlobals(); @@ -285,11 +289,14 @@ void MainWindow::Notify(QString s, QString title, int ms) // As a workaround, add an extra line to bump the message back // into the visible area. QString msg = s; + +#ifdef Q_OS_UNIX char *desktop = getenv("DESKTOP_SESSION"); if (desktop && !strncmp(desktop, "gnome", 5)) { msg += "\n"; } +#endif systray->showMessage(title, msg, QSystemTrayIcon::Information, ms); } else { @@ -342,6 +349,75 @@ void MainWindow::Startup() qprogress->hide(); qstatus->setText(""); + if (PROFILE.p_preferences[STR_PREF_ReimportBackup].toBool()) { + importCPAPBackups(); + PROFILE.p_preferences[STR_PREF_ReimportBackup]=false; + } + +} + +int MainWindow::importCPAP(const QString &path, const QString &message) +{ + QDialog popup(this, Qt::SplashScreen); + QLabel waitmsg(message); + QVBoxLayout waitlayout(&popup); + waitlayout.addWidget(&waitmsg,1,Qt::AlignCenter); + waitlayout.addWidget(qprogress,1); + qprogress->setVisible(true); + popup.show(); + int c=PROFILE.Import(path); + popup.hide(); + ui->statusbar->insertWidget(2,qprogress,1); + qprogress->setVisible(false); + + return c; +} + +void MainWindow::finishCPAPImport() +{ + PROFILE.Save(); + GenerateStatistics(); + + if (overview) { overview->ReloadGraphs(); } + if (daily) { daily->ReloadGraphs(); } +} + +void MainWindow::importCPAPBackups() +{ + + // Get BackupPaths for all CPAP machines + QList machlist = PROFILE.GetMachines(MT_CPAP); + QStringList paths; + Q_FOREACH(Machine *m, machlist) { + if (m->properties.contains(STR_PROP_BackupPath)) { + paths.push_back(PROFILE.Get(m->properties[STR_PROP_BackupPath])); + } + } + + if (paths.size() > 0) { + if (QMessageBox::question( + this, + tr("Question"), + tr("CPAP data was recently purged and needs to be re-imported.")+"\n\n"+ + tr("Would you like this done automatically from the Backup Folder?")+"\n\n"+ + QDir::toNativeSeparators(paths.join("\n")), + QMessageBox::Yes | QMessageBox::No, + QMessageBox::Yes) == QMessageBox::Yes) + { + int c=0; + Q_FOREACH(QString path, paths) { + c+=importCPAP(path,tr("Please wait, importing from backup folder(s)...")); + } + if (c>0) { + QString str=tr("Data successfully imported from the following locations\n\n")+ + QDir::toNativeSeparators(paths.join("\n")); + mainwin->Notify(str); + finishCPAPImport(); + } else { + mainwin->Notify(tr("Import Problem\n\nCouldn't find any new Machine Data at the locations given")); + } + } + } } #ifdef Q_OS_UNIX @@ -406,8 +482,8 @@ QStringList getDriveList() for (int i = 0; i < list.size(); ++i) { QFileInfo fileInfo = list.at(i); - QString name = fileInfo.fileName(); - if (name[0].toUpper() > 'C') { // Only bother looking after the C: drive + QString name = fileInfo.filePath(); + if (name.at(0).toUpper() != QChar('C')) { // Ignore the C drive drivelist.push_back(name); } } @@ -417,6 +493,60 @@ QStringList getDriveList() return drivelist; } +QStringList MainWindow::detectCPAPCards() +{ + const int timeout = 20000; + + QStringList datapaths; + + QListloaders = GetLoaders(); + QTime time; + time.start(); + + // Create dialog + QDialog popup(this, Qt::SplashScreen); + QLabel waitmsg(tr("Please wait, scanning for CPAP data cards...")); + QProgressBar progress; + QVBoxLayout waitlayout(&popup); + QPushButton skipbtn("Click here to choose a folder"); + waitlayout.addWidget(&waitmsg,1,Qt::AlignCenter); + waitlayout.addWidget(&progress,1); + waitlayout.addWidget(&skipbtn); + popup.connect(&skipbtn, SIGNAL(clicked()), &popup, SLOT(hide())); + progress.setValue(0); + progress.setMaximum(timeout); + progress.setVisible(true); + popup.show(); + QApplication::processEvents(); + + do { + // Rescan in case card inserted + QStringList AutoScannerPaths = getDriveList(); + + Q_FOREACH(const QString &path, AutoScannerPaths) { + qDebug() << "Scanning" << path; + // Scan through available machine loaders and test if this folder contains valid folder structure + Q_FOREACH(MachineLoader * loader, loaders) { + if (loader->Detect(path)) { + datapaths.push_back(path); + + qDebug() << "Found" << loader->ClassName() << "datacard at" << path; + } + } + } + int el=time.elapsed(); + progress.setValue(el); + QApplication::processEvents(); + if (el > timeout) break; + if (!popup.isVisible()) break; + // needs a slight delay here + + } while (datapaths.size() == 0); + popup.hide(); + popup.disconnect(&skipbtn, SIGNAL(clicked()), &popup, SLOT(hide())); + + return datapaths; +} void MainWindow::on_action_Import_Data_triggered() { @@ -425,56 +555,53 @@ void MainWindow::on_action_Import_Data_triggered() return; } - QHash datacard; - - QString datacard_path = QString(); - MachineLoader * datacard_loader = nullptr; + QStringList datapaths = detectCPAPCards(); QListloaders = GetLoaders(); - QStringList AutoScannerPaths = getDriveList(); - Q_FOREACH(const QString &path, AutoScannerPaths) { - qDebug() << "Scanning" << path; - // Scan through available machine loaders and test if this folder contains valid folder structure - Q_FOREACH(MachineLoader * loader, loaders) { - if (loader->Detect(path)) { - datacard[loader->ClassName()] = path; + QTime time; + time.start(); + QDialog popup(this, Qt::FramelessWindowHint); + popup.setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Minimum); + QLabel waitmsg(tr("Please wait, scanning for CPAP data cards...")); + QVBoxLayout waitlayout(&popup); + waitlayout.addWidget(&waitmsg,1,Qt::AlignCenter); + waitlayout.addWidget(qprogress,1); - qDebug() << "Found" << loader->ClassName() << "datacard at" << path; - if (datacard_path.isEmpty()) { - datacard_loader = loader; - datacard_path = path; - } - } - } - } - - QStringList importFrom; bool asknew = false; + qprogress->setVisible(false); - if (datacard.size() > 0) { - if (datacard.size() > 1) { - qWarning() << "User has more than detected datacard folder structure in scan path, only using the first one found."; - } - - int res = QMessageBox::question(this, tr("Datacard Located"), - QString(tr("A %1 datacard structure was detected at\n%2\n\nWould you like to import from this location?")). - arg(datacard_loader->ClassName()).arg(datacard_path), tr("Yes"), - tr("Select another folder"), tr("Cancel"), 0, 2); + if (datapaths.size() > 0) { + int res = QMessageBox::question(this, + tr("CPAP Data Located"), + QString((tr("CPAP Datacard structures were detected at the following locations:")+ + "\n\n%1\n\n"+ + tr("Would you like to import from the path(s) shown above?"))). + arg(QDir::toNativeSeparators(datapaths.join("\n"))), + tr("Yes"), + tr("Select another folder"), + tr("Cancel"), + 0, 2); if (res == 1) { + waitmsg.setText(tr("Please wait, launching file dialog...")); + datapaths.clear(); asknew = true; - } else { - importFrom.push_back(datacard_path); } - if (res == 2) { return; } + if (res == 2) { + // Give the communal progress bar back + ui->statusbar->insertWidget(2,qprogress,1); + return; + } } else { + waitmsg.setText(tr("No CPAP data card detected, launching file dialog...")); asknew = true; } if (asknew) { - mainwin->Notify("Please remember to point the importer at the root folder or drive letter of your data-card, and not a subfolder.","Import Reminder",8000); + popup.show(); + mainwin->Notify(tr("Please remember to point the importer at the root folder or drive letter of your data-card, and not a subfolder."),tr("Import Reminder"),8000); QFileDialog w; #if QT_VERSION < QT_VERSION_CHECK(5,0,0) @@ -487,20 +614,18 @@ void MainWindow::on_action_Import_Data_triggered() w.setFileMode(QFileDialog::Directory); w.setOption(QFileDialog::ShowDirsOnly, true); -#if defined(Q_OS_MAC) && (QT_VERSION_CHECK(4,8,0) > QT_VERSION) +#if defined(Q_OS_MAC) && (QT_VERSION < QT_VERSION_CHECK(4,8,0)) // Fix for tetragon, 10.6 barfs up Qt's custom dialog w.setOption(QFileDialog::DontUseNativeDialog, true); #else w.setOption(QFileDialog::DontUseNativeDialog, false); QListView *l = w.findChild("listView"); - if (l) { l->setSelectionMode(QAbstractItemView::MultiSelection); } QTreeView *t = w.findChild(); - if (t) { t->setSelectionMode(QAbstractItemView::MultiSelection); } @@ -508,16 +633,15 @@ void MainWindow::on_action_Import_Data_triggered() #endif if (w.exec() != QDialog::Accepted) { + popup.hide(); + ui->statusbar->insertWidget(2,qprogress,1); + return; } + popup.hide(); for (int i = 0; i < w.selectedFiles().size(); i++) { - QString newdir = w.selectedFiles().at(i); - - if (!importFrom.contains(newdir)) { - importFrom.append(newdir); - //addnew=true; - } + datapaths.append(w.selectedFiles().at(i)); } } @@ -525,15 +649,12 @@ void MainWindow::on_action_Import_Data_triggered() QStringList goodlocations; - QDialog dlg(this,Qt::SplashScreen); - QVBoxLayout layout; - dlg.setLayout(&layout); - QLabel label(tr("Please wait, SleepyHead is importing data...")); - layout.addWidget(&label,1); - layout.addWidget(qprogress,1); - dlg.show(); - for (int i = 0; i < importFrom.size(); i++) { - QString dir = importFrom[i]; + waitmsg.setText(tr("Please wait, SleepyHead is importing data...")); + qprogress->setVisible(true); + + popup.show(); + for (int i = 0; i < datapaths.size(); i++) { + QString dir = datapaths[i]; if (!dir.isEmpty()) { qprogress->setValue(0); @@ -552,17 +673,12 @@ void MainWindow::on_action_Import_Data_triggered() qprogress->hide(); } } - dlg.hide(); + popup.hide(); ui->statusbar->insertWidget(2,qprogress,1); if (successful) { - PROFILE.Save(); - - GenerateStatistics(); - - if (overview) { overview->ReloadGraphs(); } - if (daily) { daily->ReloadGraphs(); } + finishCPAPImport(); QString str=tr("Data successfully imported from the following locations\n\n"); for (int i=0; iproperties[STR_PROP_Brand] + " " + m->properties[STR_PROP_Model] + " " + m->properties[STR_PROP_ModelNumber] + " (" + m->properties[STR_PROP_Serial] + ")", - QMessageBox::Yes, QMessageBox::No) == QMessageBox::Yes) { - m->Purge(3478216); - PROFILE.machlist.erase(PROFILE.machlist.find(m->id())); + QMessageBox::Yes | QMessageBox::No, + QMessageBox::No) == QMessageBox::Yes) { + + + if (m->Purge(3478216)) { + // Turn on automatic re-import + // Note: I deliberately haven't added a Profile help for this + PROFILE.Save(); + } // delete or not to delete.. this needs to delete later.. :/ //delete m; RestartApplication(); diff --git a/sleepyhead/mainwindow.h b/sleepyhead/mainwindow.h index f3c2700e..3cb66e53 100644 --- a/sleepyhead/mainwindow.h +++ b/sleepyhead/mainwindow.h @@ -312,6 +312,11 @@ class MainWindow : public QMainWindow void on_reportModeStandard_clicked(); private: + int importCPAP(const QString &path, const QString &message); + void importCPAPBackups(); + void finishCPAPImport(); + QStringList detectCPAPCards(); + QString getWelcomeHTML(); void FreeSessions(); diff --git a/sleepyhead/overview.cpp b/sleepyhead/overview.cpp index 2d9dd5f3..089f2e32 100644 --- a/sleepyhead/overview.cpp +++ b/sleepyhead/overview.cpp @@ -299,13 +299,19 @@ Overview::Overview(QWidget *parent, gGraphView *shared) : } Overview::~Overview() { - GraphView->SaveSettings("Overview");//no trans - disconnect(this, SLOT(dateStart_currentPageChanged(int, int))); - disconnect(this, SLOT(dateEnd_currentPageChanged(int, int))); delete ui; delete icon_on; delete icon_off; } + +void Overview::closeEvent(QCloseEvent *event) +{ + GraphView->SaveSettings("Overview");//no trans + disconnect(this, SLOT(dateStart_currentPageChanged(int, int))); + disconnect(this, SLOT(dateEnd_currentPageChanged(int, int))); + QWidget::closeEvent(event); +} + gGraph *Overview::createGraph(QString name, QString units, YTickerType yttype) { int default_height = PROFILE.appearance->graphHeight(); diff --git a/sleepyhead/overview.h b/sleepyhead/overview.h index 8ccbb33b..fe4a146a 100644 --- a/sleepyhead/overview.h +++ b/sleepyhead/overview.h @@ -39,6 +39,7 @@ class Overview : public QWidget public: explicit Overview(QWidget *parent, gGraphView *shared = nullptr); ~Overview(); + void closeEvent(QCloseEvent *); //! \brief Returns Overview gGraphView object containing it's graphs gGraphView *graphView() { return GraphView; } diff --git a/sleepyhead/oximetry.cpp b/sleepyhead/oximetry.cpp index 8dfd098f..d6feb002 100644 --- a/sleepyhead/oximetry.cpp +++ b/sleepyhead/oximetry.cpp @@ -1248,10 +1248,16 @@ Oximetry::Oximetry(QWidget *parent, gGraphView *shared) : } Oximetry::~Oximetry() +{ + delete ui; +} + +void Oximetry::closeEvent(QCloseEvent *event) { delete oximeter; GraphView->SaveSettings("Oximetry"); - delete ui; + + QWidget::closeEvent(event); } void Oximetry::on_RefreshPortsButton_clicked() diff --git a/sleepyhead/oximetry.h b/sleepyhead/oximetry.h index 0bfac21d..762611a6 100644 --- a/sleepyhead/oximetry.h +++ b/sleepyhead/oximetry.h @@ -306,6 +306,7 @@ class Oximetry : public QWidget public: explicit Oximetry(QWidget *parent, gGraphView *shared = nullptr); ~Oximetry(); + void closeEvent(QCloseEvent *); //! \brief Calls updateGL to redraw the graphs void RedrawGraphs(); diff --git a/sleepyhead/preferencesdialog.cpp b/sleepyhead/preferencesdialog.cpp index 4dc54088..1923d0c7 100644 --- a/sleepyhead/preferencesdialog.cpp +++ b/sleepyhead/preferencesdialog.cpp @@ -156,6 +156,8 @@ PreferencesDialog::PreferencesDialog(QWidget *parent, Profile *_profile) : ui->bigFontBold->setChecked(bigfont->weight() == QFont::Bold); ui->bigFontItalic->setChecked(bigfont->italic()); + ui->lineThicknessSlider->setValue(profile->appearance->lineThickness()*2); + ui->startedUsingMask->setDate(profile->cpap->maskStartDate()); ui->leakModeCombo->setCurrentIndex(profile->cpap->leakMode()); @@ -388,6 +390,8 @@ bool PreferencesDialog::Save() profile->appearance->setUsePixmapCaching(ui->usePixmapCaching->isChecked()); profile->appearance->setSquareWavePlots(ui->useSquareWavePlots->isChecked()); profile->appearance->setGraphSnapshots(ui->enableGraphSnapshots->isChecked()); + profile->appearance->setLineThickness(float(ui->lineThicknessSlider->value()) / 2.0); + profile->general->setSkipEmptyDays(ui->skipEmptyDays->isChecked()); profile->general->setTooltipTimeout(ui->tooltipTimeoutSlider->value() * 50); diff --git a/sleepyhead/preferencesdialog.ui b/sleepyhead/preferencesdialog.ui index 12576249..d237c7ac 100644 --- a/sleepyhead/preferencesdialog.ui +++ b/sleepyhead/preferencesdialog.ui @@ -10,7 +10,7 @@ 0 0 721 - 572 + 610 @@ -51,7 +51,7 @@ - 5 + 6 @@ -1882,7 +1882,7 @@ p, li { white-space: pre-wrap; } - + Qt::Vertical @@ -1895,7 +1895,7 @@ p, li { white-space: pre-wrap; } - + @@ -1943,7 +1943,7 @@ p, li { white-space: pre-wrap; } - + @@ -1991,7 +1991,7 @@ p, li { white-space: pre-wrap; } - + Scroll Dampening @@ -2001,7 +2001,7 @@ p, li { white-space: pre-wrap; } - + Tooltip Timeout @@ -2093,6 +2093,42 @@ p, li { white-space: pre-wrap; } + + + + Line Thickness + + + Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter + + + + + + + + + The pixel thickness of line plots + + + 2 + + + 8 + + + 1 + + + Qt::Horizontal + + + QSlider::TicksBelow + + + + + @@ -2175,8 +2211,7 @@ this application to be unstable with this feature enabled. - Turn on/off the spinning "context" cube. -It really doesn't use that much resources.. :) + <html><head/><body><p>These features have recently been pruned. They will come back later. </p></body></html> Animations && Fancy Stuff diff --git a/sleepyhead/reports.cpp b/sleepyhead/reports.cpp index 3282e1d5..a7fccc06 100644 --- a/sleepyhead/reports.cpp +++ b/sleepyhead/reports.cpp @@ -291,7 +291,7 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date) 2); if (cpap->machine->GetClass() == STR_MACH_PRS1) { - stats += QObject::tr("REI=%1 VSI=%2 FLI=%3 PB/CSR=%4\%") + stats += QObject::tr("REI=%1 VSI=%2 FLI=%3 PB/CSR=%4%%") .arg(rei, 0, 'f', 2).arg(vsi, 0, 'f', 2) .arg(fli, 0, 'f', 2).arg(csr, 0, 'f', 2); } else if (cpap->machine->GetClass() == STR_MACH_ResMed) { @@ -532,9 +532,6 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date) int page = 1; int gcnt = 0; -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - float dpr = gv->devicePixelRatio(); -#endif for (int i = 0; i < graphs.size(); i++) { diff --git a/sleepyhead/sleepyhead.pro b/sleepyhead/sleepyhead.pro index cb341f03..787abbaf 100644 --- a/sleepyhead/sleepyhead.pro +++ b/sleepyhead/sleepyhead.pro @@ -63,15 +63,21 @@ macx { } win32 { - DEFINES += WINVER=0x0501 # needed for mingw to pull in appropriate dbt business...probably a better way to do this - RC_FILE += win_icon.rc - LIBS += -lsetupapi -lz -} -if (win32-msvc2008|win32-msvc2010|win32-msvc2012):!equals(TEMPLATE_PREFIX, "vc") { - LIBS += -ladvapi32 - DEFINES += BUILD_WITH_MSVC=1 -} + DEFINES += WINVER=0x0501 # needed for mingw to pull in appropriate dbt business...probably a better way to do this + RC_FILE += win_icon.rc + LIBS += -lsetupapi + INCLUDEPATH += $$PWD + INCLUDEPATH += $$[QT_INSTALL_PREFIX]/../src/qtbase/src/3rdparty/zlib + + if (*-msvc*):!equals(TEMPLATE_PREFIX, "vc") { + LIBS += -ladvapi32 + DEFINES += BUILD_WITH_MSVC=1 + } else { + # MingW needs this + LIBS += -lz + } +} #include(..3rdparty/qextserialport/src/qextserialport.pri) #include(3rdparty/quazip-0.5.1/quazip/quazip.pri) @@ -97,7 +103,6 @@ SOURCES += \ Graphs/gFooBar.cpp \ Graphs/gGraph.cpp \ Graphs/gGraphView.cpp \ - Graphs/GLBuffer.cpp \ Graphs/glcommon.cpp \ Graphs/gLineChart.cpp \ Graphs/gLineOverlay.cpp \ @@ -105,7 +110,6 @@ SOURCES += \ Graphs/gspacer.cpp \ Graphs/gStatsLine.cpp \ Graphs/gSummaryChart.cpp \ - Graphs/gVertexBuffer.cpp \ Graphs/gXAxis.cpp \ Graphs/gYAxis.cpp \ Graphs/layer.cpp \ @@ -149,7 +153,6 @@ HEADERS += \ Graphs/gFooBar.h \ Graphs/gGraph.h \ Graphs/gGraphView.h \ - Graphs/GLBuffer.h \ Graphs/glcommon.h \ Graphs/gLineChart.h \ Graphs/gLineOverlay.h \ @@ -157,7 +160,6 @@ HEADERS += \ Graphs/gspacer.h \ Graphs/gStatsLine.h \ Graphs/gSummaryChart.h \ - Graphs/gVertexBuffer.h \ Graphs/gXAxis.h \ Graphs/gYAxis.h \ Graphs/layer.h \ @@ -235,6 +237,7 @@ win32 { for(FILE,TRANS_FILES_WIN){ system(xcopy /y $$quote($$FILE) $$quote($$DDIR)) } + } mac { diff --git a/sleepyhead/statistics.cpp b/sleepyhead/statistics.cpp index e8939edd..9d0a06e3 100644 --- a/sleepyhead/statistics.cpp +++ b/sleepyhead/statistics.cpp @@ -1215,10 +1215,10 @@ QString StatisticsRow::value(QDate start, QDate end) value = QString("%1").arg(p_profile->calcWavg(code, type, start, end), 0, 'f', decimals); break; case SC_MEDIAN: - value = QString("%1").arg(p_profile->calcPercentile(code, 0.5, type, start, end), 0, 'f', decimals); + value = QString("%1").arg(p_profile->calcPercentile(code, 0.5F, type, start, end), 0, 'f', decimals); break; case SC_90P: - value = QString("%1").arg(p_profile->calcPercentile(code, 0.9, type, start, end), 0, 'f', decimals); + value = QString("%1").arg(p_profile->calcPercentile(code, 0.9F, type, start, end), 0, 'f', decimals); break; case SC_MIN: value = QString("%1").arg(p_profile->calcMin(code, type, start, end), 0, 'f', decimals); diff --git a/sleepyhead/win_icon.rc b/sleepyhead/win_icon.rc index 6f53ae06..b700a656 100644 --- a/sleepyhead/win_icon.rc +++ b/sleepyhead/win_icon.rc @@ -1 +1 @@ -IDI_ICON1 ICON DISCARDABLE "/icons/bob-v3.0.ico" +IDI_ICON1 ICON DISCARDABLE "./icons/bob-v3.0.ico"