2013-12-18 13:48:22 +00:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
2015-01-28 08:44:43 +00:00
|
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
|
|
** Contact: http://www.qt.io/licensing/
|
2013-12-18 13:48:22 +00:00
|
|
|
**
|
|
|
|
** This file is part of the test suite of the Qt Toolkit.
|
|
|
|
**
|
2014-08-21 13:51:22 +00:00
|
|
|
** $QT_BEGIN_LICENSE:LGPL21$
|
2013-12-18 13:48:22 +00:00
|
|
|
** Commercial License Usage
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2015-01-28 08:44:43 +00:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
** and conditions see http://www.qt.io/terms-conditions. For further
|
|
|
|
** information use the contact form at http://www.qt.io/contact-us.
|
2013-12-18 13:48:22 +00:00
|
|
|
**
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
2014-08-21 13:51:22 +00:00
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2013-12-18 13:48:22 +00:00
|
|
|
**
|
2015-01-28 08:44:43 +00:00
|
|
|
** As a special exception, The Qt Company gives you certain additional
|
|
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
2013-12-18 13:48:22 +00:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
**
|
|
|
|
** $QT_END_LICENSE$
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#define GL_GLEXT_PROTOTYPES
|
|
|
|
|
|
|
|
#include "openglwidget.h"
|
|
|
|
#include <QtWidgets/private/qwidget_p.h>
|
|
|
|
#include <QOpenGLFramebufferObject>
|
|
|
|
#include <QWindow>
|
|
|
|
#include <qpa/qplatformwindow.h>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QTimer>
|
|
|
|
|
|
|
|
#include <QtGui/QOpenGLFunctions>
|
|
|
|
#include <QtGui/QGuiApplication>
|
|
|
|
#include <QtGui/QMatrix4x4>
|
|
|
|
#include <QtGui/QOpenGLShaderProgram>
|
|
|
|
#include <QtGui/QScreen>
|
|
|
|
|
|
|
|
#include <QtCore/qmath.h>
|
|
|
|
#include <qopengl.h>
|
|
|
|
|
2014-06-20 09:58:34 +00:00
|
|
|
class OpenGLWidgetPrivate : protected QOpenGLFunctions
|
2013-12-18 13:48:22 +00:00
|
|
|
{
|
|
|
|
public:
|
2014-03-18 12:36:10 +00:00
|
|
|
OpenGLWidgetPrivate(QWidget *q)
|
|
|
|
: m_program(0), m_frame(0), q(q)
|
2013-12-18 13:48:22 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void initialize();
|
|
|
|
void render();
|
|
|
|
|
|
|
|
|
|
|
|
int width() {return w;}
|
|
|
|
int height() {return h;}
|
|
|
|
|
|
|
|
GLuint m_posAttr;
|
|
|
|
GLuint m_colAttr;
|
|
|
|
GLuint m_matrixUniform;
|
|
|
|
|
|
|
|
QOpenGLShaderProgram *m_program;
|
|
|
|
int m_frame;
|
|
|
|
|
|
|
|
int w,h;
|
2014-03-18 12:36:10 +00:00
|
|
|
QWidget *q;
|
Support mixing native child widgets with texture-based ones
Currently QOpenGLWidget and QQuickWidget do not support having native
child widgets inside the same top-level window. In some cases this is
inevitable, f.ex. multimedia may require native windows when used from
widget apps. winId() calls made for various (valid or invalid) reasons
are also problematic.
There are no blockers for supporting this setup, however. By storing
multiple texture lists (one for each subtree where the root is a
native widget), adding the missing markDirtyOnScreen calls, letting
each native widget access the correct texture list (i.e. the one
corresponding to its children) when they are (separately) flushed, and
fixing composeAndFlush() to take the update region and the (native
child) offset into account, it can all be made functional.
The change also fixes the issue of keeping GL-based compositing
enabled even after all render-to-texture widgets in the window become
hidden. Due to the changes of how such widgets are gathered,
composeAndFlush() is not invoked anymore when no such widgets are
discovered for a given native parent. This is great since having
compositing enabled infinitely is an issue for applications like Qt
Creator that implement certain views with QQuickWidgets but cannot
afford the cost of texture uploads in other places (e.g. for the text
editor) on slower machines.
The openglwidget manual test is greatly enhanced to test various
situations (MDI, scroll areas, tab widgets, QOpenGLWidget as native
child, QOpenGLWidget with non-tlw native parent, etc.)
Task-number: QTBUG-48130
Task-number: QTBUG-49172
Change-Id: Iad098359c8bcf749f01c050da0853415e1550eda
Reviewed-by: Paul Olav Tvete <paul.tvete@theqtcompany.com>
Reviewed-by: Morten Johan Sørvig <morten.sorvig@theqtcompany.com>
2015-11-03 10:41:01 +00:00
|
|
|
|
|
|
|
int m_interval;
|
|
|
|
QVector3D m_rotAxis;
|
2016-01-27 17:12:44 +00:00
|
|
|
|
|
|
|
float clearColor[3];
|
2013-12-18 13:48:22 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
Support mixing native child widgets with texture-based ones
Currently QOpenGLWidget and QQuickWidget do not support having native
child widgets inside the same top-level window. In some cases this is
inevitable, f.ex. multimedia may require native windows when used from
widget apps. winId() calls made for various (valid or invalid) reasons
are also problematic.
There are no blockers for supporting this setup, however. By storing
multiple texture lists (one for each subtree where the root is a
native widget), adding the missing markDirtyOnScreen calls, letting
each native widget access the correct texture list (i.e. the one
corresponding to its children) when they are (separately) flushed, and
fixing composeAndFlush() to take the update region and the (native
child) offset into account, it can all be made functional.
The change also fixes the issue of keeping GL-based compositing
enabled even after all render-to-texture widgets in the window become
hidden. Due to the changes of how such widgets are gathered,
composeAndFlush() is not invoked anymore when no such widgets are
discovered for a given native parent. This is great since having
compositing enabled infinitely is an issue for applications like Qt
Creator that implement certain views with QQuickWidgets but cannot
afford the cost of texture uploads in other places (e.g. for the text
editor) on slower machines.
The openglwidget manual test is greatly enhanced to test various
situations (MDI, scroll areas, tab widgets, QOpenGLWidget as native
child, QOpenGLWidget with non-tlw native parent, etc.)
Task-number: QTBUG-48130
Task-number: QTBUG-49172
Change-Id: Iad098359c8bcf749f01c050da0853415e1550eda
Reviewed-by: Paul Olav Tvete <paul.tvete@theqtcompany.com>
Reviewed-by: Morten Johan Sørvig <morten.sorvig@theqtcompany.com>
2015-11-03 10:41:01 +00:00
|
|
|
OpenGLWidget::OpenGLWidget(int interval, const QVector3D &rotAxis, QWidget *parent)
|
2013-12-18 13:48:22 +00:00
|
|
|
: QOpenGLWidget(parent)
|
|
|
|
{
|
Support mixing native child widgets with texture-based ones
Currently QOpenGLWidget and QQuickWidget do not support having native
child widgets inside the same top-level window. In some cases this is
inevitable, f.ex. multimedia may require native windows when used from
widget apps. winId() calls made for various (valid or invalid) reasons
are also problematic.
There are no blockers for supporting this setup, however. By storing
multiple texture lists (one for each subtree where the root is a
native widget), adding the missing markDirtyOnScreen calls, letting
each native widget access the correct texture list (i.e. the one
corresponding to its children) when they are (separately) flushed, and
fixing composeAndFlush() to take the update region and the (native
child) offset into account, it can all be made functional.
The change also fixes the issue of keeping GL-based compositing
enabled even after all render-to-texture widgets in the window become
hidden. Due to the changes of how such widgets are gathered,
composeAndFlush() is not invoked anymore when no such widgets are
discovered for a given native parent. This is great since having
compositing enabled infinitely is an issue for applications like Qt
Creator that implement certain views with QQuickWidgets but cannot
afford the cost of texture uploads in other places (e.g. for the text
editor) on slower machines.
The openglwidget manual test is greatly enhanced to test various
situations (MDI, scroll areas, tab widgets, QOpenGLWidget as native
child, QOpenGLWidget with non-tlw native parent, etc.)
Task-number: QTBUG-48130
Task-number: QTBUG-49172
Change-Id: Iad098359c8bcf749f01c050da0853415e1550eda
Reviewed-by: Paul Olav Tvete <paul.tvete@theqtcompany.com>
Reviewed-by: Morten Johan Sørvig <morten.sorvig@theqtcompany.com>
2015-11-03 10:41:01 +00:00
|
|
|
d.reset(new OpenGLWidgetPrivate(this));
|
2016-01-27 17:12:44 +00:00
|
|
|
d->clearColor[0] = d->clearColor[1] = d->clearColor[2] = 0.0f;
|
Support mixing native child widgets with texture-based ones
Currently QOpenGLWidget and QQuickWidget do not support having native
child widgets inside the same top-level window. In some cases this is
inevitable, f.ex. multimedia may require native windows when used from
widget apps. winId() calls made for various (valid or invalid) reasons
are also problematic.
There are no blockers for supporting this setup, however. By storing
multiple texture lists (one for each subtree where the root is a
native widget), adding the missing markDirtyOnScreen calls, letting
each native widget access the correct texture list (i.e. the one
corresponding to its children) when they are (separately) flushed, and
fixing composeAndFlush() to take the update region and the (native
child) offset into account, it can all be made functional.
The change also fixes the issue of keeping GL-based compositing
enabled even after all render-to-texture widgets in the window become
hidden. Due to the changes of how such widgets are gathered,
composeAndFlush() is not invoked anymore when no such widgets are
discovered for a given native parent. This is great since having
compositing enabled infinitely is an issue for applications like Qt
Creator that implement certain views with QQuickWidgets but cannot
afford the cost of texture uploads in other places (e.g. for the text
editor) on slower machines.
The openglwidget manual test is greatly enhanced to test various
situations (MDI, scroll areas, tab widgets, QOpenGLWidget as native
child, QOpenGLWidget with non-tlw native parent, etc.)
Task-number: QTBUG-48130
Task-number: QTBUG-49172
Change-Id: Iad098359c8bcf749f01c050da0853415e1550eda
Reviewed-by: Paul Olav Tvete <paul.tvete@theqtcompany.com>
Reviewed-by: Morten Johan Sørvig <morten.sorvig@theqtcompany.com>
2015-11-03 10:41:01 +00:00
|
|
|
d->m_interval = interval;
|
|
|
|
d->m_rotAxis = rotAxis;
|
|
|
|
if (interval > 0) {
|
|
|
|
QTimer *timer = new QTimer(this);
|
|
|
|
connect(timer, SIGNAL(timeout()), this, SLOT(update()));
|
|
|
|
timer->start(interval);
|
|
|
|
}
|
2013-12-18 13:48:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
OpenGLWidget::~OpenGLWidget()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLWidget::initializeGL()
|
|
|
|
{
|
|
|
|
// qDebug("*initializeGL*");
|
|
|
|
d->initialize();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLWidget::resizeGL(int w, int h)
|
|
|
|
{
|
|
|
|
// qDebug("*resizeGL*");
|
|
|
|
d->w = w;
|
|
|
|
d->h = h;
|
|
|
|
}
|
|
|
|
void OpenGLWidget::paintGL()
|
|
|
|
{
|
|
|
|
// qDebug("*paintGL* %d", d->m_frame);
|
|
|
|
d->render();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static const char *vertexShaderSource =
|
|
|
|
"attribute highp vec4 posAttr;\n"
|
|
|
|
"attribute lowp vec4 colAttr;\n"
|
|
|
|
"varying lowp vec4 col;\n"
|
|
|
|
"uniform highp mat4 matrix;\n"
|
|
|
|
"void main() {\n"
|
|
|
|
" col = colAttr;\n"
|
|
|
|
" gl_Position = matrix * posAttr;\n"
|
|
|
|
"}\n";
|
|
|
|
|
|
|
|
static const char *fragmentShaderSource =
|
|
|
|
"varying lowp vec4 col;\n"
|
|
|
|
"void main() {\n"
|
|
|
|
" gl_FragColor = col;\n"
|
|
|
|
"}\n";
|
|
|
|
|
|
|
|
void OpenGLWidgetPrivate::initialize()
|
|
|
|
{
|
2014-06-20 09:58:34 +00:00
|
|
|
initializeOpenGLFunctions();
|
2013-12-18 13:48:22 +00:00
|
|
|
m_program = new QOpenGLShaderProgram;
|
|
|
|
m_program->addShaderFromSourceCode(QOpenGLShader::Vertex, vertexShaderSource);
|
|
|
|
m_program->addShaderFromSourceCode(QOpenGLShader::Fragment, fragmentShaderSource);
|
|
|
|
m_program->link();
|
|
|
|
m_posAttr = m_program->attributeLocation("posAttr");
|
|
|
|
m_colAttr = m_program->attributeLocation("colAttr");
|
|
|
|
m_matrixUniform = m_program->uniformLocation("matrix");
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLWidgetPrivate::render()
|
|
|
|
{
|
2014-03-18 12:36:10 +00:00
|
|
|
const qreal retinaScale = q->devicePixelRatio();
|
2013-12-18 13:48:22 +00:00
|
|
|
glViewport(0, 0, width() * retinaScale, height() * retinaScale);
|
|
|
|
|
2016-01-27 17:12:44 +00:00
|
|
|
glClearColor(clearColor[0], clearColor[1], clearColor[2], 1.0f);
|
2013-12-18 13:48:22 +00:00
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
|
|
|
|
m_program->bind();
|
|
|
|
|
|
|
|
QMatrix4x4 matrix;
|
2014-06-20 09:58:34 +00:00
|
|
|
matrix.perspective(60.0f, 4.0f/3.0f, 0.1f, 100.0f);
|
2013-12-18 13:48:22 +00:00
|
|
|
matrix.translate(0, 0, -2);
|
Support mixing native child widgets with texture-based ones
Currently QOpenGLWidget and QQuickWidget do not support having native
child widgets inside the same top-level window. In some cases this is
inevitable, f.ex. multimedia may require native windows when used from
widget apps. winId() calls made for various (valid or invalid) reasons
are also problematic.
There are no blockers for supporting this setup, however. By storing
multiple texture lists (one for each subtree where the root is a
native widget), adding the missing markDirtyOnScreen calls, letting
each native widget access the correct texture list (i.e. the one
corresponding to its children) when they are (separately) flushed, and
fixing composeAndFlush() to take the update region and the (native
child) offset into account, it can all be made functional.
The change also fixes the issue of keeping GL-based compositing
enabled even after all render-to-texture widgets in the window become
hidden. Due to the changes of how such widgets are gathered,
composeAndFlush() is not invoked anymore when no such widgets are
discovered for a given native parent. This is great since having
compositing enabled infinitely is an issue for applications like Qt
Creator that implement certain views with QQuickWidgets but cannot
afford the cost of texture uploads in other places (e.g. for the text
editor) on slower machines.
The openglwidget manual test is greatly enhanced to test various
situations (MDI, scroll areas, tab widgets, QOpenGLWidget as native
child, QOpenGLWidget with non-tlw native parent, etc.)
Task-number: QTBUG-48130
Task-number: QTBUG-49172
Change-Id: Iad098359c8bcf749f01c050da0853415e1550eda
Reviewed-by: Paul Olav Tvete <paul.tvete@theqtcompany.com>
Reviewed-by: Morten Johan Sørvig <morten.sorvig@theqtcompany.com>
2015-11-03 10:41:01 +00:00
|
|
|
const qreal angle = 100.0f * m_frame / 30;
|
|
|
|
matrix.rotate(angle, m_rotAxis);
|
2013-12-18 13:48:22 +00:00
|
|
|
|
|
|
|
m_program->setUniformValue(m_matrixUniform, matrix);
|
|
|
|
|
|
|
|
GLfloat vertices[] = {
|
|
|
|
0.0f, 0.707f,
|
|
|
|
-0.5f, -0.5f,
|
|
|
|
0.5f, -0.5f
|
|
|
|
};
|
|
|
|
|
|
|
|
GLfloat colors[] = {
|
|
|
|
1.0f, 0.0f, 0.0f,
|
|
|
|
0.0f, 1.0f, 0.0f,
|
|
|
|
0.0f, 0.0f, 1.0f
|
|
|
|
};
|
|
|
|
|
|
|
|
glVertexAttribPointer(m_posAttr, 2, GL_FLOAT, GL_FALSE, 0, vertices);
|
|
|
|
glVertexAttribPointer(m_colAttr, 3, GL_FLOAT, GL_FALSE, 0, colors);
|
|
|
|
|
|
|
|
glEnableVertexAttribArray(0);
|
|
|
|
glEnableVertexAttribArray(1);
|
|
|
|
|
|
|
|
glDrawArrays(GL_TRIANGLES, 0, 3);
|
|
|
|
|
|
|
|
glDisableVertexAttribArray(1);
|
|
|
|
glDisableVertexAttribArray(0);
|
|
|
|
|
|
|
|
m_program->release();
|
|
|
|
|
|
|
|
++m_frame;
|
Support mixing native child widgets with texture-based ones
Currently QOpenGLWidget and QQuickWidget do not support having native
child widgets inside the same top-level window. In some cases this is
inevitable, f.ex. multimedia may require native windows when used from
widget apps. winId() calls made for various (valid or invalid) reasons
are also problematic.
There are no blockers for supporting this setup, however. By storing
multiple texture lists (one for each subtree where the root is a
native widget), adding the missing markDirtyOnScreen calls, letting
each native widget access the correct texture list (i.e. the one
corresponding to its children) when they are (separately) flushed, and
fixing composeAndFlush() to take the update region and the (native
child) offset into account, it can all be made functional.
The change also fixes the issue of keeping GL-based compositing
enabled even after all render-to-texture widgets in the window become
hidden. Due to the changes of how such widgets are gathered,
composeAndFlush() is not invoked anymore when no such widgets are
discovered for a given native parent. This is great since having
compositing enabled infinitely is an issue for applications like Qt
Creator that implement certain views with QQuickWidgets but cannot
afford the cost of texture uploads in other places (e.g. for the text
editor) on slower machines.
The openglwidget manual test is greatly enhanced to test various
situations (MDI, scroll areas, tab widgets, QOpenGLWidget as native
child, QOpenGLWidget with non-tlw native parent, etc.)
Task-number: QTBUG-48130
Task-number: QTBUG-49172
Change-Id: Iad098359c8bcf749f01c050da0853415e1550eda
Reviewed-by: Paul Olav Tvete <paul.tvete@theqtcompany.com>
Reviewed-by: Morten Johan Sørvig <morten.sorvig@theqtcompany.com>
2015-11-03 10:41:01 +00:00
|
|
|
|
|
|
|
if (m_interval <= 0)
|
|
|
|
q->update();
|
2013-12-18 13:48:22 +00:00
|
|
|
}
|
2016-01-27 17:12:44 +00:00
|
|
|
|
|
|
|
void OpenGLWidget::setClearColor(const float *c)
|
|
|
|
{
|
|
|
|
d->clearColor[0] = c[0];
|
|
|
|
d->clearColor[1] = c[1];
|
|
|
|
d->clearColor[2] = c[2];
|
|
|
|
}
|