博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
qml demo分析(threadedanimation-线程动画)
阅读量:6438 次
发布时间:2019-06-23

本文共 6769 字,大约阅读时间需要 22 分钟。

一、效果预览

  使用过qml的同学都知道,使用qml做动画效果是非常简单的,再也不需要像QWidget那样,自己模拟一个动画,费时又费力,往往还达不到效果。今天我们就来分析下qml的两种动画实现方式,如图1所示,窗口底部的提示信息文本“This application shows two spinners. The one to the right is animated on the scene graph thread (when applicable) and the left one is using the normal Qt Quick animation system.”意思就是左边的窗口使用quick animation实现,右侧窗口使用opengl scene graph实现,细心的同学就会发现,当Not blocked字样变为Blocked时,左侧的旋转动画每过大概400ms就会停顿下,而右侧窗口丝毫不受影响。好了废话不多说,我们这就来分析下这个示例程序的源码,这个现象就不难理解啦。

图1 旋转动画

二、源码分析

  首先我们先来分析下该示例代码的工程目录,如图1所示,spinner一个C++类,其重写了QQuickItem组件,QQuickItem其实在qml中就相当于Item组件,重写QQuickItem是为了注册到qml系统中,把spinner当qml的一个组件使用,在spinner的实现过程中,使用了opengl的方式来进行绘图。

图2 工程目录

  该示例代码的main.cpp文件与文章中分析的一样,都是注册了一个自定义QQuickItem对象到qml上下文中,并使用QQuickView加载main.qml文件并显示,在此就不贴代码啦。

1、opengl scene graph

  由于这是第一次分析opengl的示例程序,这里我将会把spinner的头文件和实现代码都贴上。首先先来看下spinner类头文件

1 #ifndef SPINNER_H 2 #define SPINNER_H 3  4 #include 
5 6 class Spinner : public QQuickItem 7 { 8 Q_OBJECT 9 10 Q_PROPERTY(bool spinning READ spinning WRITE setSpinning NOTIFY spinningChanged)11 12 public:13 Spinner();14 15 bool spinning() const { return m_spinning; }16 void setSpinning(bool spinning);17 18 protected:19 QSGNode *updatePaintNode(QSGNode *, UpdatePaintNodeData *);20 21 signals:22 void spinningChanged();23 24 private:25 bool m_spinning;26 };

  头文件中声明了一个SPinner类,继承自QQuickItem类,重写该类是为了使用opengl绘制旋转效果。代码中使用了Q_PROPERTY宏将m_spinning成员变量注册到了qml系统中,qml代码直接可以通过spinning:值的方式修改该变量,该宏在之前的文章中也有简单的介绍,具体参见。

1 Spinner::Spinner() 2     : m_spinning(false) 3 { 4     setSize(QSize(64, 64)); 5     setFlag(ItemHasContents); 6 } 7  8 void Spinner::setSpinning(bool spinning) 9 {10     if (spinning == m_spinning)11         return;12     m_spinning = spinning;13     emit spinningChanged();14     update();15 }16 17 QSGNode *Spinner::updatePaintNode(QSGNode *old, UpdatePaintNodeData *)18 {19     SpinnerNode *n = static_cast
(old);20 if (!n)21 n = new SpinnerNode(window());22 23 n->setSpinning(m_spinning);24 25 return n;26 }

  在Spinner类实现中,updatePaintNode接口中构造了一个SpinnerNode类,具体的绘制操作就是在SpinnerNode类中实现。

1 class SpinnerNode : public QObject, public QSGTransformNode 2 { 3     Q_OBJECT 4 public: 5     SpinnerNode(QQuickWindow *window) 6         : m_rotation(0) 7         , m_spinning(false) 8         , m_window(window) 9     {10         connect(window, &QQuickWindow::beforeRendering, this, &SpinnerNode::maybeRotate);11         connect(window, &QQuickWindow::frameSwapped, this, &SpinnerNode::maybeUpdate);12 13         QImage image(":/scenegraph/threadedanimation/spinner.png");14         m_texture = window->createTextureFromImage(image);//创建一个纹理15         QSGSimpleTextureNode *textureNode = new QSGSimpleTextureNode();//简单纹理节点  加入场景前必须有一个纹理16         textureNode->setTexture(m_texture);//设置纹理17         textureNode->setRect(0, 0, image.width(), image.height());18         textureNode->setFiltering(QSGTexture::Linear);19         appendChildNode(textureNode);20     }21 22     ~SpinnerNode() {23         delete m_texture;24     }25 26     void setSpinning(bool spinning)27     {28         m_spinning = spinning;29     }30 31 public slots:32     void maybeRotate() {33         if (m_spinning) {34             m_rotation += (360 / m_window->screen()->refreshRate());35             QMatrix4x4 matrix;36             matrix.translate(32, 32);37             matrix.rotate(m_rotation, 0, 0, 1);//绕z轴旋转38             matrix.translate(-32, -32);39             setMatrix(matrix);40         }41     }42 43     void maybeUpdate() {44         if (m_spinning) {45             m_window->update();46         }47     }48 49 private:50     qreal m_rotation;51     bool m_spinning;52     QSGTexture *m_texture;53     QQuickWindow *m_window;54 };

  SpinnerNode类继承自QSGTransformNode,构造函数中使用图片构造了一个纹理,并将纹理设置到QSGSimpleTextureNode对象中,进而将该对象添加到场景,添加到场景的顺序决定他们在场景中被绘制的顺序。

  SpinnerNode类中使用maybeRotate方法将纹理进行了旋转,代码如上32-41行所示。

2、quick animation

  使用qml属性动画进行旋转操作相对来说比较简单,只需要使用NumberAnimation on rotation即可,具体代码如下64-66行

1 Rectangle { 2  3     width: 320 4     height: 480 5  6     gradient: Gradient {
//设置背景色 7 GradientStop { position: 0; color: "lightsteelblue" } 8 GradientStop { position: 1; color: "black" } 9 }10 11 Rectangle {
//与blockingLabel组成一个提示框12 color: Qt.rgba(1, 1, 1, 0.7);13 radius: 1014 border.width: 115 border.color: "white"16 anchors.fill: blockingLabel;17 anchors.margins: -1018 }19 20 Text {21 id: blockingLabel22 color: blocker.running ? "red" : "black"//根据定时器状态 修改提示框文本信息和字体颜色23 text: blocker.running ? "Blocked!" : "Not blocked"24 anchors.horizontalCenter: parent.horizontalCenter25 anchors.top: parent.top26 anchors.topMargin: 10027 }28 29 Timer {30 id: blocker31 interval: 357//触发间隔32 running: false;//默认没有启动33 repeat: true34 onTriggered: {
//该槽的作用就是为了模拟一个延迟 说明opengl的渲染不在主线程进行 而quick animation在主线程进行渲染35 var d = new Date();36 var x = 0;37 var wait = 50 + Math.random() * 200;38 while ((new Date().getTime() - d.getTime()) < 100) {39 x += 1;40 }41 }42 }43 44 Timer {45 id: blockerEnabler46 interval: 400047 running: true48 repeat: true49 onTriggered: {50 blocker.running = !blocker.running//定时器状态修改 4s中置反51 }52 }53 54 Spinner {
//opengl55 anchors.centerIn: parent56 anchors.horizontalCenterOffset: 8057 spinning: true//通过setSpinning接口设置该属性值58 }59 60 Image {
//quick animation61 anchors.centerIn: parent62 anchors.horizontalCenterOffset: -8063 source: "spinner.png"64 NumberAnimation on rotation {65 from: 0; to: 360; duration: 1000; loops: Animation.Infinite//1s转动一圈66 }67 }68 69 Rectangle {
//与label文本组成底部提示框70 color: Qt.rgba(1, 1, 1, 0.7)71 radius: 1072 border.width: 173 border.color: "white"74 anchors.fill: label75 anchors.margins: -1076 }77 78 Text {79 id: label80 color: "black"81 wrapMode: Text.WordWrap82 text: "This application shows two spinners. The one to the right is animated on the scene graph thread (when applicable) and the left one is using the normal Qt Quick animation system."83 anchors.right: parent.right84 anchors.left: parent.left85 anchors.bottom: parent.bottom86 anchors.margins: 2087 }88 }

 

总结:使用qml有两种方式实现旋转,往大的说就是动画,有两种方式:opengl scene or quick animation。

opengl scene:实现比较麻烦,但是线程独立,不会因为复杂计算而阻塞ui绘制

quick animation:使用简单,但是是在主线程进行绘制。

转载地址:http://yozwo.baihongyu.com/

你可能感兴趣的文章
ZendStudio10.6.1如何安装最新的集成svn小工具?
查看>>
PHP中$_SERVER的详细参数与说明
查看>>
jquery easyui datagrid mvc server端分页排序筛选的实现
查看>>
去了大公司就一定能学到很牛的技术么?
查看>>
methanol 模块化的可定制的网页爬虫软件,主要的优点是速度快。
查看>>
IOS开发之表视图(UITableView)
查看>>
Notepad++去除代码行号的几种方法
查看>>
polay定理总结
查看>>
IIS如何配置可以下载APK、IPA文件
查看>>
CodeForces 396C 树状数组 + DFS
查看>>
[sharepoint]rest api文档库文件上传,下载,拷贝,剪切,删除文件,创建文件夹,修改文件夹属性,删除文件夹,获取文档列表...
查看>>
远程桌面退出全屏/不能全屏/全屏切换的技巧
查看>>
【Java】Float计算不准确
查看>>
mybatis在xml文件中处理大于号小于号的方法
查看>>
Codeforces Codeforces Round #319 (Div. 2) A. Multiplication Table 水题
查看>>
各大浏览器CSS Hack收集
查看>>
再谈 $* 和 $@ 在 Bash 中的表现
查看>>
Apache Commons工具集简介
查看>>
【翻译】Nginx的反向代理
查看>>
htm、html、shtml网页区别
查看>>