Fix QOpenGLWidget on Cocoa when used as viewport

Having a QOpenGLWidget as a graphics view viewport was not functioning
on OS X: it was showing incomplete content due to accessing the texture
attached to the framebuffer object before the rendering is complete.

On the normal path, when rendering is done via paintGL(), the flush was
there. When used as a viewport however, this path is not used. The missing
flush is now added for the other case too. For performance reasons, we will
not flush on every paint engine end(). Instead, the flush is deferred until
composition starts.

QGLWidget also featured a weird on-by-default autoFillBackground concept. To
maintain compatibility with apps that used QGLWidget as the viewport for
QGraphicsView, we will now do the same for QOpenGLWidget, but only when it
is used as a viewport. For regular QOpenGLWidgets autoFillBackground defaults
to false, like for any other widget. The docs are extended with a small section
about differences between QGLWidget and QOpenGLWidget.

Task-number: QTBUG-41046
Change-Id: I42c2033fdd2ef5815783fd640fe11373761061e0
Reviewed-by: Jørgen Lind <jorgen.lind@digia.com>
This commit is contained in:
Laszlo Agocs 2014-10-14 12:07:41 +02:00
parent 4bf0660ae4
commit ebc835c2aa
6 changed files with 103 additions and 5 deletions

View File

@ -350,15 +350,40 @@ bool QOpenGLPaintDevice::paintFlipped() const
return d_ptr->flipped; return d_ptr->flipped;
} }
/*!
This virtual method is called when starting to paint.
The default implementation does nothing.
\sa endPaint()
*/
void QOpenGLPaintDevice::beginPaint()
{
}
/*! /*!
This virtual method is provided as a callback to allow re-binding a target This virtual method is provided as a callback to allow re-binding a target
frame buffer object or context when different QOpenGLPaintDevice instances frame buffer object or context when different QOpenGLPaintDevice instances
are issuing draw calls alternately. are issuing draw calls alternately.
QPainter::beginNativePainting will also trigger this method. \l{QPainter::beginNativePainting()}{beginNativePainting()} will also trigger
this method.
The default implementation does nothing.
*/ */
void QOpenGLPaintDevice::ensureActiveTarget() void QOpenGLPaintDevice::ensureActiveTarget()
{ {
} }
/*!
This virtual method is called when the painting has finished.
The default implementation does nothing.
\sa beginPaint()
*/
void QOpenGLPaintDevice::endPaint()
{
}
QT_END_NAMESPACE QT_END_NAMESPACE

View File

@ -73,7 +73,9 @@ public:
void setPaintFlipped(bool flipped); void setPaintFlipped(bool flipped);
bool paintFlipped() const; bool paintFlipped() const;
virtual void beginPaint();
virtual void ensureActiveTarget(); virtual void ensureActiveTarget();
virtual void endPaint();
protected: protected:
int metric(QPaintDevice::PaintDeviceMetric metric) const; int metric(QPaintDevice::PaintDeviceMetric metric) const;

View File

@ -1994,6 +1994,8 @@ bool QOpenGL2PaintEngineEx::begin(QPaintDevice *pdev)
d->ctx = QOpenGLContext::currentContext(); d->ctx = QOpenGLContext::currentContext();
d->ctx->d_func()->active_engine = this; d->ctx->d_func()->active_engine = this;
d->device->beginPaint();
d->funcs.initializeOpenGLFunctions(); d->funcs.initializeOpenGLFunctions();
for (int i = 0; i < QT_GL_VERTEX_ARRAY_TRACKED_COUNT; ++i) for (int i = 0; i < QT_GL_VERTEX_ARRAY_TRACKED_COUNT; ++i)
@ -2044,6 +2046,8 @@ bool QOpenGL2PaintEngineEx::end()
{ {
Q_D(QOpenGL2PaintEngineEx); Q_D(QOpenGL2PaintEngineEx);
d->device->endPaint();
QOpenGLContext *ctx = d->ctx; QOpenGLContext *ctx = d->ctx;
d->funcs.glUseProgram(0); d->funcs.glUseProgram(0);
d->transferMode(BrushDrawingMode); d->transferMode(BrushDrawingMode);

View File

@ -239,6 +239,28 @@ QT_BEGIN_NAMESPACE
\note Avoid calling winId() on a QOpenGLWidget. This function triggers the creation of \note Avoid calling winId() on a QOpenGLWidget. This function triggers the creation of
a native window, resulting in reduced performance and possibly rendering glitches. a native window, resulting in reduced performance and possibly rendering glitches.
\section1 Differences to QGLWidget
Besides the main conceptual difference of being backed by a framebuffer object, there
are a number of smaller, internal differences between QOpenGLWidget and the older
QGLWidget:
\list
\li OpenGL state when invoking paintGL(). QOpenGLWidget sets up the viewport via
glViewport(). It does not perform any clearing.
\li Clearing when starting to paint via QPainter. Unlike regular widgets, QGLWidget
defaulted to a value of \c true for
\l{QWidget::autoFillBackground()}{autoFillBackground}. It then performed clearing to the
palette's background color every time QPainter::begin() was used. QOpenGLWidget does not
follow this: \l{QWidget::autoFillBackground()}{autoFillBackground} defaults to false,
like for any other widget. The only exception is when being used as a viewport for other
widgets like QGraphicsView. In such a case autoFillBackground will be automatically set
to true to ensure compatibility with QGLWidget-based viewports.
\endlist
\section1 Multisampling \section1 Multisampling
To enable multisampling, set the number of requested samples on the To enable multisampling, set the number of requested samples on the
@ -436,6 +458,7 @@ class QOpenGLWidgetPaintDevice : public QOpenGLPaintDevice
{ {
public: public:
QOpenGLWidgetPaintDevice(QOpenGLWidget *widget) : w(widget) { } QOpenGLWidgetPaintDevice(QOpenGLWidget *widget) : w(widget) { }
void beginPaint() Q_DECL_OVERRIDE;
void ensureActiveTarget() Q_DECL_OVERRIDE; void ensureActiveTarget() Q_DECL_OVERRIDE;
private: private:
@ -454,7 +477,8 @@ public:
initialized(false), initialized(false),
fakeHidden(false), fakeHidden(false),
paintDevice(0), paintDevice(0),
inBackingStorePaint(false) inBackingStorePaint(false),
flushPending(false)
{ {
requestedFormat = QSurfaceFormat::defaultFormat(); requestedFormat = QSurfaceFormat::defaultFormat();
} }
@ -478,6 +502,7 @@ public:
void endBackingStorePainting() Q_DECL_OVERRIDE { inBackingStorePaint = false; } void endBackingStorePainting() Q_DECL_OVERRIDE { inBackingStorePaint = false; }
void beginCompose() Q_DECL_OVERRIDE; void beginCompose() Q_DECL_OVERRIDE;
void endCompose() Q_DECL_OVERRIDE; void endCompose() Q_DECL_OVERRIDE;
void initializeViewportFramebuffer() Q_DECL_OVERRIDE;
void resizeViewportFramebuffer() Q_DECL_OVERRIDE; void resizeViewportFramebuffer() Q_DECL_OVERRIDE;
void resolveSamples() Q_DECL_OVERRIDE; void resolveSamples() Q_DECL_OVERRIDE;
@ -490,8 +515,28 @@ public:
QOpenGLPaintDevice *paintDevice; QOpenGLPaintDevice *paintDevice;
bool inBackingStorePaint; bool inBackingStorePaint;
QSurfaceFormat requestedFormat; QSurfaceFormat requestedFormat;
bool flushPending;
}; };
void QOpenGLWidgetPaintDevice::beginPaint()
{
// NB! autoFillBackground is and must be false by default. Otherwise we would clear on
// every QPainter begin() which is not desirable. This is only for legacy use cases,
// like using QOpenGLWidget as the viewport of a graphics view, that expect clearing
// with the palette's background color.
if (w->autoFillBackground()) {
QOpenGLFunctions *f = QOpenGLContext::currentContext()->functions();
if (w->testAttribute(Qt::WA_TranslucentBackground)) {
f->glClearColor(0, 0, 0, 0);
} else {
QColor c = w->palette().brush(w->backgroundRole()).color();
float alpha = c.alphaF();
f->glClearColor(c.redF() * alpha, c.greenF() * alpha, c.blueF() * alpha, alpha);
}
f->glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
}
}
void QOpenGLWidgetPaintDevice::ensureActiveTarget() void QOpenGLWidgetPaintDevice::ensureActiveTarget()
{ {
QOpenGLWidgetPrivate *d = static_cast<QOpenGLWidgetPrivate *>(QWidgetPrivate::get(w)); QOpenGLWidgetPrivate *d = static_cast<QOpenGLWidgetPrivate *>(QWidgetPrivate::get(w));
@ -502,6 +547,11 @@ void QOpenGLWidgetPaintDevice::ensureActiveTarget()
w->makeCurrent(); w->makeCurrent();
else else
d->fbo->bind(); d->fbo->bind();
// When used as a viewport, drawing is done via opening a QPainter on the widget
// without going through paintEvent(). We will have to make sure a glFlush() is done
// before the texture is accessed also in this case.
d->flushPending = true;
} }
GLuint QOpenGLWidgetPrivate::textureId() const GLuint QOpenGLWidgetPrivate::textureId() const
@ -572,6 +622,11 @@ void QOpenGLWidgetPrivate::recreateFbo()
void QOpenGLWidgetPrivate::beginCompose() void QOpenGLWidgetPrivate::beginCompose()
{ {
Q_Q(QOpenGLWidget); Q_Q(QOpenGLWidget);
if (flushPending) {
flushPending = false;
q->makeCurrent();
context->functions()->glFlush();
}
emit q->aboutToCompose(); emit q->aboutToCompose();
} }
@ -653,9 +708,10 @@ void QOpenGLWidgetPrivate::invokeUserPaint()
{ {
Q_Q(QOpenGLWidget); Q_Q(QOpenGLWidget);
QOpenGLFunctions *f = QOpenGLContext::currentContext()->functions(); QOpenGLFunctions *f = QOpenGLContext::currentContext()->functions();
f->glViewport(0, 0, q->width() * q->devicePixelRatio(), q->height() * q->devicePixelRatio());
f->glViewport(0, 0, q->width() * q->devicePixelRatio(), q->height() * q->devicePixelRatio());
q->paintGL(); q->paintGL();
f->glFlush();
} }
void QOpenGLWidgetPrivate::render() void QOpenGLWidgetPrivate::render()
@ -667,7 +723,6 @@ void QOpenGLWidgetPrivate::render()
q->makeCurrent(); q->makeCurrent();
invokeUserPaint(); invokeUserPaint();
context->functions()->glFlush();
} }
extern Q_GUI_EXPORT QImage qt_gl_read_framebuffer(const QSize &size, bool alpha_format, bool include_alpha); extern Q_GUI_EXPORT QImage qt_gl_read_framebuffer(const QSize &size, bool alpha_format, bool include_alpha);
@ -686,6 +741,14 @@ QImage QOpenGLWidgetPrivate::grabFramebuffer()
return res; return res;
} }
void QOpenGLWidgetPrivate::initializeViewportFramebuffer()
{
Q_Q(QOpenGLWidget);
// Legacy behavior for compatibility with QGLWidget when used as a graphics view
// viewport: enable clearing on each painter begin.
q->setAutoFillBackground(true);
}
void QOpenGLWidgetPrivate::resizeViewportFramebuffer() void QOpenGLWidgetPrivate::resizeViewportFramebuffer()
{ {
Q_Q(QOpenGLWidget); Q_Q(QOpenGLWidget);
@ -929,7 +992,6 @@ void QOpenGLWidget::resizeEvent(QResizeEvent *e)
d->recreateFbo(); d->recreateFbo();
resizeGL(width(), height()); resizeGL(width(), height());
d->invokeUserPaint(); d->invokeUserPaint();
d->context->functions()->glFlush();
d->resolveSamples(); d->resolveSamples();
} }

View File

@ -641,6 +641,8 @@ public:
} }
} }
static void sendComposeStatus(QWidget *w, bool end); static void sendComposeStatus(QWidget *w, bool end);
// Called on setViewport().
virtual void initializeViewportFramebuffer() { }
// When using a QOpenGLWidget as viewport with QAbstractScrollArea, resize events are // When using a QOpenGLWidget as viewport with QAbstractScrollArea, resize events are
// filtered away from the widget. This is fine for QGLWidget but bad for QOpenGLWidget // filtered away from the widget. This is fine for QGLWidget but bad for QOpenGLWidget
// since the fbo must be resized. We need an alternative way to notify. // since the fbo must be resized. We need an alternative way to notify.

View File

@ -609,6 +609,9 @@ void QAbstractScrollArea::setViewport(QWidget *widget)
#endif #endif
#endif #endif
d->layoutChildren(); d->layoutChildren();
#ifndef QT_NO_OPENGL
QWidgetPrivate::get(d->viewport)->initializeViewportFramebuffer();
#endif
if (isVisible()) if (isVisible())
d->viewport->show(); d->viewport->show();
setupViewport(widget); setupViewport(widget);