Qt之动画框架

简述

Qt动画框架旨在为创建动画和平滑的GUI提供了一种简单的方法。通过Qt动画属性,该框架为部件和其它QObject对象的动画操作提供了非常大的自由性,框架也可以被用于图形视图框架中,动画框架中许多可用的概念也可以用于Qt Quick,它提供了一种声明式的方式定义动画。大部分关于动画框架学到的知识都可以应用于Qt Quick。

本篇,我们阐述了Qt动画框架的基本结构。同时,会展示最常见的技术示例,用于动画操作QObject和图形项。

  • 简述
  • 动画框架结构
  • 动画框架类
  • Qt动画属性
  • 动画和图形视图框架
  • 缓和曲线
  • 动画分组
  • 动画和状态
  • 更多参考

动画框架结构

本节中,我们将宏观了解Qt动画框架的体系结构,以及如何被用于Qt动画属性。下图展示了动画框架中最重要的类。

动画框架基础由基类QAbstractAnimation以及它的两个子类QVariantAnimation、QAnimationGroup组成。QAbstractAnimation是所有动画的祖先。它包含了一些在框架中被普遍使用的基本功能,尤其是启动、停止和暂停动画功能,它也接收定时触发通知。

Qt动画框架更是提供了QPropertyAnimation类,该类继承自QVariantAnimation,用于对Qt属性的动画操作(Qt属性系统是Qt元对象系统的一部分)。QPropertyAnimation类使用缓和曲线算法对属性进行插值演化操作。因此当你想使用动画改变一个值时,需要声明其为一个属性并且使该类继承自QObject。这给我们提供了很大的方便性,去动画操作现有的部件和其它的QObject对象。

复杂动画可以通过构建QAbstractAnimation树形结构来构造。该树主要使用QAnimationGroup,QAnimationGroup类是一个包含其它动画类的容器类;同时QAnimationGroup类也是QAbstractAnimation类的子类,因此一个容器可以包含其它容器。

Qt动画框架可以独立使用,但是也被设计为Qt状态机框架的一部分。状态机框架提供一个特殊的状态用来播放动画。当状态进入或者退出时,QState也可以改变属性。当这个动画状态提供了一个QPropertyAnimatio时,这个特殊的状态会在这些值之间进行篡改操作。后续我们将了解的更加仔细。

在幕后,动画被一个全局定时器控制着,该定时器对所有正在运行的动画发送更新命令。

要了解Qt动画框架中各个类的功能,请参考相应的类描述信息。

动画框架类

这些类提供了用于创建简单的和复杂的动画的框架

描述
QAbstractAnimation 所有动画类的基类
QAnimationGroup 动画容器类的抽象基类
QEasingCurve 动画控制的缓和曲线类
QParallelAnimationGroup 并行动画容器
QPauseAnimation QSequentialAnimationGroup暂停
QPropertyAnimation Qt的动画属性
QSequentialAnimationGroup 串行动画容器
QTimeLine 控制动画的时间轴类
QVariantAnimation 动画类的抽象基类

Qt动画属性

如上所述,QPropertyAnimation类能够修改Qt属性值,正是该类用于改变动画属性值。事实上,它的基类QVariantAnimation是一个抽象类,所以不能被直接使用。

选用Qt动画属性的一个主要原因,是因为它给我们很大的自由性去动画操作Qt API中已经存在的类,尤其是拥有bounds、colors等属性的QWidget类(能被嵌入到QGraphicsView中的QWidget)。

来看一个小例子:

QPushButton button("Animated Button");
button.show();

QPropertyAnimation animation(&button, "geometry");
animation.setDuration(10000);
animation.setStartValue(QRect(0, 0, 100, 30));
animation.setEndValue(QRect(250, 250, 100, 30));

animation.start();

上述代码,在10秒的持续时间把button从屏幕的左上角移动到(250, 250)点处。

上面的例子在开始值与结束值之间做了线性插值。当然,设置的值在开始处与结束处之间的数值也是合理的,那么插值衍化就沿这些点进行。

QPushButton button("Animated Button");
button.show();

QPropertyAnimation animation(&button, "geometry");
animation.setDuration(10000);

animation.setKeyValueAt(0, QRect(0, 0, 100, 30));
animation.setKeyValueAt(0.8, QRect(250, 250, 100, 30));
animation.setKeyValueAt(1, QRect(0, 0, 100, 30));

animation.start();

这个例子中,在8秒的持续时间将button移到(250, 250),然后在剩下的2秒再移回至初始位置;这些点之间的移动都是通过线性插值的。

你也可以动画操作没有声明动画属性的QObject对象中的值,但是唯一的条件是该值有个能进行修改的设置函数。所以可以进行子类化,在该类中包含声明属性的值并且有个设置函数。每个Qt属性需要一个获取值的访问函数,因此如果类本身没提供对该值的访问函数的话,你自己就需要提供一个。

class MyGraphicsRectItem : public QObject, public QGraphicsRectItem
{
    Q_OBJECT
    Q_PROPERTY(QRectF geometry READ geometry WRITE setGeometry)
};

如上所示的代码例子中,我们子类化QGraphicsRectItem类,并且定义了”geometry”属性。即使QGraphicsRectItem没有提供”geometry”属性,我们也可以动画操作MyGraphicsRectItem的位置信息了。

动画和图形视图框架

当你想动画操作QGraphicsItem时,也可以使用QPropertyAnimation类。然而,QGraphicsItem并不继承于QObject。一个好的解决办法是子类化一个你需要的图形项,同时这个类也继承自QObject。通过这种方式,QPropertyAnimation类就能适用于QGraphicsItem。下面的代码例子展示了这是如何实现的。另一种可行性是只继承于QGraphicsWidget,因为QGraphicsWidget继承于QObject。

class Pixmap : public QObject, public QGraphicsPixmapItem
{
    Q_OBJECT
    Q_PROPERTY(QPointF pos READ pos WRITE setPos)
    ...

如上所述,我们定义了一个需要动画操作的属性值。

注意:出于元对象系统的要求,QObject必须是第一个继承者。

缓和曲线

QPropertyAnimation在开始与结束之间执行插值操作。除了为动画添加更多的键值外,你也可以使用缓和曲线,缓和曲线控制着在0与1之间的插值速度,如果你想在没有改变插值路径的情况下改变动画速度,那么缓和曲线是很有用的。

QPushButton button("Animated Button");
button.show();

QPropertyAnimation animation(&button, "geometry");
animation.setDuration(3000);
animation.setStartValue(QRect(0, 0, 100, 30));
animation.setEndValue(QRect(250, 250, 100, 30));

animation.setEasingCurve(QEasingCurve::OutBounce);

animation.start();

这里,动画即沿着OutBounce曲线,该曲线样式是到结束处会弹跳起来像个弹跳球。QEasingCurve类有大量供选择的曲线,它们被定义成QEasingCurve::Type枚举。如果你需要另外的曲线样式,也可以自己实现一个,然后用QEasingCurve注册它既可。

动画分组

一个应用程序常常包含多个动画。例如,你或许希望同时移动不止一个图形项或者一个接一个的顺序移动它们。

QAnimationGroup(QSequentialAnimationGroup和QParallelAnimationGroup)的子类是动画容器类,因此多个动画可以被串行或者并行执行。QAnimationGroup类就是一个例子,其不操作动画属性,但是它能周期性的获得定时通知,这使得它能把定时通知应用于动画中,从而进行控制。

下面我们来看看使用QSequentialAnimationGroup和QParallelAnimationGroup的例子:

QPushButton *bonnie = new QPushButton("Bonnie");
bonnie->show();

QPushButton *clyde = new QPushButton("Clyde");
clyde->show();

QPropertyAnimation *anim1 = new QPropertyAnimation(bonnie, "geometry");
// Set up anim1

QPropertyAnimation *anim2 = new QPropertyAnimation(clyde, "geometry");
// Set up anim2

QParallelAnimationGroup *group = new QParallelAnimationGroup;
group->addAnimation(anim1);
group->addAnimation(anim2);

group->start();

并行容器内的动画是同时进行的,调用它的start()函数即开始操作它所管理的所有动画。

QPushButton button("Animated Button");
button.show();

QPropertyAnimation anim1(&button, "geometry");
anim1.setDuration(3000);
anim1.setStartValue(QRect(0, 0, 100, 30));
anim1.setEndValue(QRect(500, 500, 100, 30));

QPropertyAnimation anim2(&button, "geometry");
anim2.setDuration(3000);
anim2.setStartValue(QRect(500, 500, 100, 30));
anim2.setEndValue(QRect(1000, 500, 100, 30));

QSequentialAnimationGroup group;

group.addAnimation(&anim1);
group.addAnimation(&anim2);

group.start();

毫无疑问你已经猜到了,QSequentialAnimationGroup串行的操作它所管理的动画。

因为动画容器类也是动画,所以你可以将其加入到其它动画容器里;用这种方式,就可以建造一个动画树结构,该结构指定了动画彼此之间运行的关系。

动画和状态

当使用状态机时,我们可以使用QSignalTransition或QEventTransition类将一个或者多个动画与状态之间的切换中进行关联。这些类继承于QAbstractTransition,QAbstractTransition类提供了便利的函数addAnimation(),该函数在状态切换发生的情况下能触发一个或多个被附加的动画。

我们也可以和状态进行属性关联,而不是自己设置开始和结束值,下面就是一段完整的动画操作QPushButton位置的代码例子:

QPushButton *button = new QPushButton("Animated Button");
button->show();

QStateMachine *machine = new QStateMachine;

QState *state1 = new QState(machine);
state1->assignProperty(button, "geometry", QRect(0, 0, 100, 30));
machine->setInitialState(state1);

QState *state2 = new QState(machine);
state2->assignProperty(button, "geometry", QRect(250, 250, 100, 30));

QSignalTransition *transition1 = state1->addTransition(button,
    SIGNAL(clicked()), state2);
transition1->addAnimation(new QPropertyAnimation(button, "geometry"));

QSignalTransition *transition2 = state2->addTransition(button,
    SIGNAL(clicked()), state1);
transition2->addAnimation(new QPropertyAnimation(button, "geometry"));

machine->start();

有关如何使用动画状态机框架更全面的示例,请参见状态示例(位于examples/animation/states目录)。

更多参考

  • The Animation Framework(助手)
时间: 2024-10-07 01:57:21

Qt之动画框架的相关文章

详解Qt的动画框架(一)

Qt的动画框架是在4.6版本引入的.通过Qt动画属性,Qt动画框架为部件和其他QObject对象的动画操作提供了非常大的自由性.Qt动画框架也能用于图形视图框架中.以下是Qt中的有关动画框架类的类视图: Qt的动画框架的Base是由QAbstactionAnimation以及它的两个子类QVariantAnimation和QAnimationGroup组成.QAbstractAnimation类是所有动画类的祖先.它包含了一些在框架中被普遍使用的基本功能:尤其是启动.停止和暂停动画功能.它也接收

详解Qt动画框架(2)--- 实现网易云音乐tag切换

在详解Qt的动画框架(一)介绍了关于Qt动画框架一些基础知识,在这一节中,将会实际的看到有关与动画框架的实现,该案例主要实现的具体是网易云音乐tag的切换,网易云音乐中的切换如图所示: 本文介绍的方法也可以达到这种切换的简易效果. 设计动画框架 首先我们需要设计对于动画框架,其需要的动画效果是什么?对于上图,我们需要的是三个tag可以不停的切换,可以抽象为左移以及右移,即一个tag从一个矩形区域,移动到另外的矩形区域,那么对于Tag的承载体选为什么较为合适呢?因为我们仅仅只需要图片的显示,因此Q

Qt移动应用开发(二):使用动画框架

上一篇博客介绍了如何使用Qt的QML来对屏幕分辨率大小进行适应,事实上,不同分辨率的适应是一个非常棘手的问题,除了分辨率不同外,宽高比(aspect ratio)也不尽相同.有些平板在硬件上做得和IPad一样是Retina屏(2048×1536),有些低端的手机分辨率只有320×480,这样宽高比又不一样了,所以在设计App的过程一定要对内容布局有所规划.采用锚布局的方法可以帮我们解决一定的问题,同时也要善用Screen类的成员来获得系统分辨率的更多信息. 这篇文章主要介绍的是QtQuick的动

[转]IFTTT开源Swift编写的帧动画框架--RazzleDazzle

RazzleDazzle 是IFTTT开源的一个iOS帧动画框架,用Swift编写,非常适用于APP初次使用时的介绍和引导信息.RazzleDazzle由IFTTT此前开源的一款Objective-C滚动帧动画库JazzHands发展而来.JazzHands是UIKit一个简单的关键帧基础动画框架,可通过手势.scrollview.KVO或者ReactiveCocoa控制动画,被IFTTT应用在IFTTT for iPhone上.多款知名应用程序都使用了JazzHands这个框架,目前其在git

qt 窗口动画

窗口动画 编辑删除转载 2015-10-10 14:50:27 标签:qt渐变动画 一个应用程序通常包含多个动画,例如,你可能希望同时移动许多graphic items或者一个个按照串行的方式的移动他们 QAnimationGroup:动画组父类,以包含其它animations,这样这些animations可以串行或者并行的触发了 QSequentialAnimationGroup:串行动画 QParallelAnimationGroup:并行动画 QPropertyAnimation?:动画类

js动画框架设计

题记: 当你不再依赖JQuery时,当你已经厌倦了引入js类库实现一些动画效果的方式,当你想实现一个简单而实用的动画框架......下面介绍下愚人设计的动画框架:支持动画缓动算法函数,如Linear.Cubic.Back.Bounce,支持改变高度,宽度,透明度,边框,外边距的基本动画,支持动画的回调函数,如开始.暂停.完成的callback等. Section One 游戏动画,Flash动画里一个比较重要的概念是帧频,即每秒播放多少帧动画,一般动画是30帧/秒,单位为fps(frames p

DirectUI通用动画框架

在编写VC界面时,编写动画比较困难,代码重用性不高.编写一个临时动画需要创建定时器或者线程来驱动改变渲染状态,来达到画面实时改变的目的.但是定时器和线程都是比较难以维护的,处理不好很容易造成资源浪费甚至程序崩溃. Skilla在上一周整理好了skillcore库,这一次又给它增添了通用动画框架.这个动画框架本身没有渲染功能,主要是提供动画的驱动事件,使用时需要自己去处理动画事件去完成动画渲染.该框架比较简单,动画由线程来驱动,下面展示一下具体的构成. 根据动画的特点,就像播放动画片一样,Skil

Rebound动画框架简单介绍

Rebound动画框架简单介绍 Android菜鸟一枚,有不对的地方希望大家指出,谢谢. 最近在接手了一个老项目,发现里面动画框架用的是facebook中的Rebound框架,由于以前没听说过,放假时闲得蛋痛,看看了源码,就顺手写这一篇吧. 写了一个小Demo,具体效果如下: 代码很简单,这是xml布局: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools=&

CSS3动画框架 Animate.css

CSS3的动画应用越来越多了,Animate.css一个从名字上就知道干什么的动画框架. github上的访问地址:http://daneden.github.io/animate.css/ 使用方法 引用css <link rel="stylesheet" href="animate.min.css" /> 然后在需要动画效果的元素上添加animated和网页上显示的API的名字即可,比如: <div class="animated b