Graphics snapshot system broken, added FrameBufferObject & PixelBuffer methods

This commit is contained in:
Mark Watkins 2013-01-15 15:55:39 +10:00
parent 377ee48768
commit 365b57ace4
5 changed files with 128 additions and 34 deletions

View File

@ -11,6 +11,8 @@
#include <QTimer> #include <QTimer>
#include <QLabel> #include <QLabel>
#include <QDir> #include <QDir>
#include <QGLPixelBuffer>
#include <QGLFramebufferObject>
#include "mainwindow.h" #include "mainwindow.h"
#include "Graphs/gYAxis.h" #include "Graphs/gYAxis.h"
@ -1770,11 +1772,79 @@ Layer * gGraph::getLineChart()
return NULL; return NULL;
} }
// QTBUG-24710 pixmaps might not be freeing properly..
QPixmap gGraphView::pbRenderPixmap(int w,int h)
{
QPixmap pm=QPixmap();
QGLFormat pbufferFormat = format();
QGLPixelBuffer pbuffer(w,h,pbufferFormat,this);
if (pbuffer.isValid()) {
pbuffer.makeCurrent();
resizeGL(w,h);
initializeGL();
paintGL();
QImage image=pbuffer.toImage();
pm=QPixmap::fromImage(image);
pbuffer.doneCurrent();
}
return pm;
}
//MW: ick globals, but I want a system wide framebuffer of decent proprotions..
bool fbo_unsupported=false;
QGLFramebufferObject *fbo=NULL;
const int max_fbo_width=2048;
const int max_fbo_height=2048;
QPixmap gGraphView::fboRenderPixmap(int w,int h)
{
QPixmap pm=QPixmap();
if (fbo_unsupported)
return pm;
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::NoAttachment);
}
if (fbo && fbo->isValid()) {
makeCurrent();
if (fbo->bind()) {
initializeGL();
resizeGL(w,h);
paintGL();
glFlush();
//QImage img=grabFrameBuffer(true);
fbo->release();
// Copy just the section of the image (remember openGL draws from the bottom up)
pm=QPixmap::fromImage(fbo->toImage()).copy(0,max_fbo_height-h,w,h);
doneCurrent();
}
} else {
delete fbo;
fbo=NULL;
fbo_unsupported=true;
}
return pm;
}
QPixmap gGraph::renderPixmap(int w, int h, bool printing) QPixmap gGraph::renderPixmap(int w, int h, bool printing)
{ {
QPixmap pm=QPixmap();
gGraphView *sg=mainwin->snapshotGraph(); gGraphView *sg=mainwin->snapshotGraph();
if (!sg) return QPixmap(); if (!sg)
return QPixmap();
QFont * _defaultfont=defaultfont; QFont * _defaultfont=defaultfont;
QFont * _mediumfont=mediumfont; QFont * _mediumfont=mediumfont;
@ -1815,8 +1885,18 @@ QPixmap gGraph::renderPixmap(int w, int h, bool printing)
sg->setScaleY(1.0); sg->setScaleY(1.0);
QPixmap pm=sg->renderPixmap(w,h,false); //sg->makeCurrent();
pm=sg->fboRenderPixmap(w,h);
if (pm.isNull()) {
// this one gives nags
pm=sg->renderPixmap(w,h,false);
} else if (pm.isNull()) { // not sure if this will work with printing
qDebug() << "Had to use PixelBuffer for snapshots\n";
pm=sg->pbRenderPixmap(w,h);
}
//sg->doneCurrent();
sg->trashGraphs(); sg->trashGraphs();
m_graphview=tgv; m_graphview=tgv;

View File

@ -955,7 +955,24 @@ public:
//! \brief Trash all graph objects listed (without destroying Graph contents) //! \brief Trash all graph objects listed (without destroying Graph contents)
void trashGraphs(); void trashGraphs();
//! \brief Use a QGLFrameBufferObject to render to a pixmap
QPixmap fboRenderPixmap(int w,int h);
//! \brief Use a QGLPixelBuffer to render to a pixmap
QPixmap pbRenderPixmap(int w,int h);
protected: 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 The heart of the OpenGL drawing code
virtual void paintGL();
Day * m_day; Day * m_day;
//! \brief Calculates the sum of all graph heights //! \brief Calculates the sum of all graph heights
@ -964,17 +981,11 @@ protected:
//! \brief Calculates the sum of all graph heights, taking scaling into consideration //! \brief Calculates the sum of all graph heights, taking scaling into consideration
float scaleHeight(); float scaleHeight();
//! \brief Set up the OpenGL basics for the QGLWidget underneath
virtual void initializeGL();
//! \brief The heart of the OpenGL drawing code
virtual void paintGL();
//! \brief Graph drawing routines, returns true if there weren't any graphs to draw //! \brief Graph drawing routines, returns true if there weren't any graphs to draw
bool renderGraphs(); bool renderGraphs();
//! \brief Resize the OpenGL ViewPort prior to redrawing
virtual void resizeGL(int width, int height);
//! \brief Update the OpenGL area when the screen is resized //! \brief Update the OpenGL area when the screen is resized
virtual void resizeEvent(QResizeEvent *); virtual void resizeEvent(QResizeEvent *);

View File

@ -917,11 +917,15 @@ void Daily::Load(QDate date)
GAHI->setShowTitle(false); GAHI->setShowTitle(false);
QPixmap pixmap=GAHI->renderPixmap(150,150,false); QPixmap pixmap=GAHI->renderPixmap(150,150,false);
QByteArray byteArray; if (!pixmap.isNull()) {
QBuffer buffer(&byteArray); // use buffer to store pixmap into byteArray QByteArray byteArray;
buffer.open(QIODevice::WriteOnly); QBuffer buffer(&byteArray); // use buffer to store pixmap into byteArray
pixmap.save(&buffer, "PNG"); buffer.open(QIODevice::WriteOnly);
html += "<tr><td colspan=4 align=center><img src=\"data:image/png;base64," + byteArray.toBase64() + "\"></td></tr>\n"; pixmap.save(&buffer, "PNG");
html += "<tr><td colspan=4 align=center><img src=\"data:image/png;base64," + byteArray.toBase64() + "\"></td></tr>\n";
} else {
html += "<tr><td colspan=4 align=center>Unable to display Pie Chart on this system</td></tr>\n";
}
} else { } else {
html += "<tr><td colspan=4 align=center><img src=\"qrc:/docs/0.0.gif\"></td></tr>\n"; html += "<tr><td colspan=4 align=center><img src=\"qrc:/docs/0.0.gif\"></td></tr>\n";
} }

View File

@ -1,25 +1,14 @@
<html> <html>
<body> <body>
<h1><image src='qrc:/docs/sheep.png' width=64 height=64>SleepyHead v0.9.2 <b>BETA</b></h1> <h1><image src='qrc:/docs/sheep.png' width=64 height=64>SleepyHead v0.9.3 <b>BETA</b></h1>
<p><h2><b>Release Notes</b></h2></p> <p><h2><b>Release Notes</b></h2></p>
<p>Hi There!</p> <p>Better late than never!</p>
<p>This is just a minor update, but it added some new functionality to the auto-updater.</p> <p>It's Bug fixing time. A huge shout out of appreciation to Rich Freeman for his help with this one.</p>
<b>New features & bugs fixes in this Update:</b></br> <b>New features & bugs fixes in this Update:</b></br>
<list> <list>
<li>Auto-Updater test for Windows & Mac Platforms</li> <li>Philips Respironics '60 series' support</li>
<li>Adds Total Leaks Overview chart for PRS1 Users.</li>
<li>Preliminary ZEO CSV Support, and simple SleepStage line graph</li>
<li>Fixes Overview AHI chart showing "No Data" on 0.00 days.</li>
<li>Fixes crash after using Preferences before importing first data.</li>
<li>Fixes first minute of Resp. Rate & Minute Vent calcs not showing data. (You will have to Data->Advanced->Purge CPAP Data and then reimport if you want this, as recalc won't do it)</li>
<li>Cursor up/down zoom did not take into account hidden graphs</li>
</list> </list>
<p><b>Would you like to help test breaky stuff?</b></p>
<p>Advanced users who are willing, can now help test future updates, so we can hopefully avoid unleashing unintentonally buggy versions on everyone else.</p>
<p>There is an option in General Preferences that tells the updater to look at potentially breaky experimental & test builds.</p>
<p>If your experiencing a specific bug or problem I might occasionally ask you to switch this setting on so I can deliver a custom fix for you to test.</p>
<p>Please always read the release notes carefully before accepting an update with experimental/test builds switched on.</p>
<p><b>Sleep Well, and have fun!</b></p> <p><b>Sleep Well, and have fun!</b></p>
<p>Mark Watkins (JediMark)</p> <p>Mark Watkins (JediMark)</p>

View File

@ -241,7 +241,9 @@ void MainWindow::Startup()
// profile is a global variable set in main after login // profile is a global variable set in main after login
PROFILE.LoadMachineData(); PROFILE.LoadMachineData();
SnapshotGraph=new gGraphView(this); //daily->graphView()); SnapshotGraph=new gGraphView(this,daily->graphView());
SnapshotGraph->setFormat(daily->graphView()->format());
//SnapshotGraph->setMaximumSize(1024,512); //SnapshotGraph->setMaximumSize(1024,512);
//SnapshotGraph->setMinimumSize(1024,512); //SnapshotGraph->setMinimumSize(1024,512);
SnapshotGraph->hide(); SnapshotGraph->hide();
@ -1894,7 +1896,9 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date)
} else { } else {
ebp=QPixmap(":/icons/smileyface.png"); ebp=QPixmap(":/icons/smileyface.png");
} }
painter.drawPixmap(virt_width-piesize,bounds.height(),piesize,piesize,ebp); if (!ebp.isNull()) {
painter.drawPixmap(virt_width-piesize,bounds.height(),piesize,piesize,ebp);
}
getDaily()->eventBreakdownPie()->setShowTitle(true); getDaily()->eventBreakdownPie()->setShowTitle(true);
cpapinfo+="\n\n"; cpapinfo+="\n\n";
@ -2117,6 +2121,7 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date)
first=true; first=true;
if (page > pages) if (page > pages)
break; break;
if (!printer->newPage()) { if (!printer->newPage()) {
qWarning("failed in flushing page to disk, disk full?"); qWarning("failed in flushing page to disk, disk full?");
break; break;
@ -2155,14 +2160,19 @@ void MainWindow::PrintReport(gGraphView *gv,QString name, QDate date)
int tmb=g->m_marginbottom; int tmb=g->m_marginbottom;
g->m_marginbottom=0; g->m_marginbottom=0;
//g->showTitle(false);
QPixmap pm=g->renderPixmap(virt_width,full_graph_height-normal_height,1);//fscale);
//g->showTitle(true);
//painter.beginNativePainting();
//g->showTitle(false);
int hhh=full_graph_height-normal_height;
QPixmap pm(virt_width,hhh); //full_graph_height=g->renderPixmap(virt_width,full_graph_height-normal_height,1);//fscale);
//g->showTitle(true);
//painter.endNativePainting();
g->m_marginbottom=tmb; g->m_marginbottom=tmb;
PROFILE.appearance->setAntiAliasing(aa_setting); PROFILE.appearance->setAntiAliasing(aa_setting);
painter.drawPixmap(0,top,virt_width,full_graph_height-normal_height,pm); if (!pm.isNull()) {
painter.drawPixmap(0,top,virt_width,full_graph_height-normal_height,pm);
}
top+=full_graph_height; top+=full_graph_height;
gcnt++; gcnt++;