QML之QQuickFramebufferObject

1.应用场景

在以QML和C++混合编程的项目开发中遇到涉及到视频播放的需求,预研过程中发现如果直接使用QML中Image来一帧帧显示图像,导致帧率相当感人,在嵌入式ARM-Linux环境中可能仅有2-3帧/秒。在寻找相关解决方法时偶然看到Qt中QQuickFramebufferObject类可以结合QOpenglFunctions来显示图片解决该问题,实际开发完成后测试基于上述解决方法在同样嵌入式环境中可以达到15帧/秒以上的显示,效果达到目标设计。

2.使用原理

在Qt5.12.12助手中是这样介绍QQuickFramebufferObject类的:"QQuickFramebufferObject类是一个便利类,用于将使用帧缓冲区对象(FBO)的OpenGL渲染与Qt Quick集成。在大多数平台上,渲染将在专用线程上进行。因此,QQuickFramebufferObject类强制在项目实现和FBO渲染之间进行严格分离。所有项目逻辑,如QML所需的属性和UI相关辅助函数,都应位于QQuickFramesbufferObject子类中。所有与渲染相关的内容都必须位于QQuickFramebufferObject::Renderer类中。"

在介绍中可以看出该类继承OpenGL的FBO到Qt Quick中来,再结合QQuickFramebufferObject::Renderer类可以实现显示功能。

3.使用方法

好在Qt官方给出相关的例子,在例子中搜索FBO可以看到相关代码,以下结合例子介绍相关使用方法。

cpp 复制代码
// 1.继承QQuickFramebufferObject,重写createRender
class FboInSGRenderer : public QQuickFramebufferObject {
    Q_OBJECT
public:	
	// 继承QQuickFramebufferObject,并重写createRender函数
    Renderer *createRenderer() const {
    	// LogoInFboRender类继承与Renderer类,重新实现Renderer
    	return new LogoInFboRenderer();
    }
};
// 2.该类重新实现了render和createFramebufferObject函数
class LogoInFboRenderer : public QQuickFramebufferObject::Renderer {
public:
    void render() override {
    	// LogoRenderer继承qopenglfunctions
        logo.render();
        update();
    }
    QOpenGLFramebufferObject *createFramebufferObject(const QSize &size) override {
        QOpenGLFramebufferObjectFormat format;
        format.setAttachment(QOpenGLFramebufferObject::CombinedDepthStencil);
        format.setSamples(4);
        return new QOpenGLFramebufferObject(size, format);
    }
    LogoRenderer logo;
};
// 3.opengl绘制图像
void LogoRenderer::render() {
	// 深度测试
    glDepthMask(true);
    // 清屏
    glClearColor(0.5f, 0.5f, 0.7f, 1.0f);
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
    // 纹理过滤设置
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR );
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR );
    glFrontFace(GL_CW);
    glCullFace(GL_FRONT);
    glEnable(GL_CULL_FACE);
    glEnable(GL_DEPTH_TEST);

    QMatrix4x4 modelview;
    modelview.rotate(m_fAngle, 0.0f, 1.0f, 0.0f);
    modelview.rotate(m_fAngle, 1.0f, 0.0f, 0.0f);
    modelview.rotate(m_fAngle, 0.0f, 0.0f, 1.0f);
    modelview.scale(m_fScale);
    modelview.translate(0.0f, -0.2f, 0.0f);
	// shader程序绑定
    program1.bind();
    program1.setUniformValue(matrixUniform1, modelview);
    //绘制Qt图标
    paintQtLogo();
    // 解绑
    program1.release();
    glDisable(GL_DEPTH_TEST);
    glDisable(GL_CULL_FACE);
    m_fAngle += 1.0f;
}
// 4.QML中注册
qmlRegisterType<FboInSGRenderer>("SceneGraphRendering", 1, 0, "Renderer");
// 5.QML中使用
    Renderer {
        id: renderer
        anchors.fill: parent
        anchors.margins: 10
        // The transform is just to show something interesting..
        transform: [
            Rotation { id: rotation; axis.x: 0; axis.z: 0; axis.y: 1; angle: 0; origin.x: renderer.width / 2; origin.y: renderer.height / 2; },
            Translate { id: txOut; x: -renderer.width / 2; y: -renderer.height / 2 },
            Scale { id: scale; },
            Translate { id: txIn; x: renderer.width / 2; y: renderer.height / 2 }
        ]
    }
相关推荐
看到我,请让我去学习6 小时前
QT - QT开发进阶合集
开发语言·qt
飞剑神17 小时前
qt svg缺失元素, 原因是不支持 rgba
开发语言·qt
眠りたいです18 小时前
Qt音频播放器项目实践:文件过滤、元数据提取与动态歌词显示实现
c++·qt·ui·音视频·媒体·qt5·mime
奇树谦1 天前
QT|windwos桌面端应用程序开发,当连接多个显示器的时候,如何获取屏幕编号?
开发语言·qt
Franklin1 天前
Python界面设计【QT-creator基础编程 - 01】如何让不同分辨率图像自动匹配graphicsView的窗口大小
开发语言·python·qt
郝学胜-神的一滴1 天前
深入理解QFlags:Qt中的位标志管理工具
开发语言·c++·qt·程序人生
看到我,请让我去学习2 天前
Qt— 布局综合项目(Splitter,Stacked,Dock)
开发语言·qt
创想未来CTF2 天前
Qt同步处理业务并禁用按钮
qt
谱写秋天2 天前
Qt 5.5 的安装与配置(使用 VSCode编辑)
开发语言·vscode·qt
前端市界2 天前
前端视角: PyQt6+Vue3 跨界开发实战
前端·qt·pyqt