简述

很多时候,我们在实现页面页面显示时,经常要 show 或者 hide 页面,直接操作显得很生硬,体验度不好。所以很想在这期间,加上一个动画,用动画过度,让页面流畅,舒适。那么QTimeLine 类你不可错过。QTimeLine 类提供了用于控制动画的时间轴通常用于通过定期调用一个槽函数来动画一个 GUI 控件,一个动画是由很多张静态画面组成而每一个画面就是一帧图像。根据人是觉得暂留效应,每隔一定时间间隔就显示一帧图像当该间隔较短时人眼就感觉不出来了觉得看到的是连续的影像。

使用流程

构造对象

QTimeLine在构造函数中,你可以传递一个毫秒级的参数,设定动画的运行时间。如:

  1. m_timLine = std::make_shared<QTimeLine>(600,this);

这里设置持续时间是600毫秒。这里是有默认值的,默认值是1000毫秒。

参数设置

1、设置帧数

  1. m_timLine->setFrameRange(0,100);

这样表示动画分为100帧,然后每帧刷新的时候会产生一个信号, frameChanged() 这样,你就可以通过链接你的槽函数做你想做的事情。
2、设置增长曲线模式。

  1. m_timLine->setCurveShape(QTimeLine::SineCurve);

这里设置为正弦曲线增长方式,其他的方式,请见后面参数详解。
3、信号绑定

  1. connect(m_timLine.get(),&QTimeLine::valueChanged,this, &QtWidgetsDesktop::updateStep);

4、启动
最后一切准备就绪之后,调用start函数:

  1. m_timLine->start();

参数详解

QTimeLine :: CurveShape

该枚举描述了QTimeLine值曲线的默认形状。默认形状是EaseInOutCurve。曲线定义了值和时间线之间的关系。

不变 描述
QTimeLine::EaseInCurve 0 该值开始缓慢增长,然后增加速度。
QTimeLine::EaseOutCurve 1 该值开始稳定增长,然后缓慢结束。
QTimeLine::EaseInOutCurve 2 值开始缓慢增长,然后稳定运行,然后再次缓慢增长。
QTimeLine::LinearCurve 3 该值线性增长(例如,如果持续时间为1000 ms,则时间500 ms的值为0.5)。
QTimeLine::SineCurve 4 该值以正弦形式增长。
QTimeLine::CosineCurve 5 该值呈余弦形增长。

通过 setCurveShape() 进行设置. curveShape() 获取当前状态.

QTimeLine :: Direction

该枚举描述了处于 运行 状态时时间轴的方向。

不变 描述
QTimeLine::Forward 0 时间轴的当前时间随时间增加(即,从0移到结束/持续时间)。
QTimeLine::Backward 1 时间轴的当前时间随时间减少(即,从结束/持续时间向0移动)。

通过 setDirection() 进行设置。 direction() 获取当前状态.

QTimeLine :: State

该枚举描述了时间轴的状态。

不变 描述
QTimeLine::NotRunning 0 时间轴未运行。这是QTimeLine的初始状态,完成后将重新进入QTimeLine状态。当前时间,帧和值将保持不变,直到调用setCurrentTime()或通过调用start()启动时间轴。
QTimeLine::Paused 1 时间线已暂停(即暂时暂停)。调用setPaused(false)将恢复时间轴活动。
QTimeLine::Running 2 时间轴正在运行。当控件处于事件循环中时,QTimeLine将定期更新其当前时间,并在适当的时候发出valueChanged()和frameChanged()。

通过 state() 获取当前状态, stateChanged() 获取改变状态。

使用案例

代码展示

这里实现了一个 QWidget ,当鼠标进入时,粉色图形从左往右,宽度渐渐增长,直到和父窗口相等。紫色图形,从右往左,宽度渐渐减小,直到为0。但鼠标离开时,情况正好相反。以下是代码。

  1. //QtWidgetsDesktop.h
  2. #pragma once
  3. #include <QWidget>
  4. #include <QFrame>
  5. #include <memory>
  6. class QLabel;
  7. class QTimeLine;
  8. class QtWidgetsDesktop : public QFrame
  9. {
  10. Q_OBJECT
  11. public:
  12. QtWidgetsDesktop(QFrame* parent = Q_NULLPTR);
  13. protected slots:
  14. void updateStep(qreal val);
  15. protected:
  16. void enterEvent(QEvent* event);
  17. void leaveEvent(QEvent* event);
  18. private:
  19. std::shared_ptr<QTimeLine> m_timLine;
  20. std::shared_ptr<QLabel> m_labShowA;
  21. std::shared_ptr<QLabel> m_labShowB;
  22. };
  1. //QtWidgetsDesktop.cpp
  2. #include "QtWidgetsDesktop.h"
  3. #include <QStyleOption>
  4. #include <QPainter>
  5. #include <QPushButton>
  6. #include <QLabel>
  7. #include <QTimeLine>
  8. QtWidgetsDesktop::QtWidgetsDesktop(QFrame*parent)
  9. : QFrame(parent)
  10. {
  11. this->resize(500, 309);
  12. this->setWindowFlags(Qt::FramelessWindowHint);
  13. this->setStyleSheet("background: #76AA71;");
  14. m_labShowA = std::make_shared<QLabel>(this);
  15. m_labShowB = std::make_shared<QLabel>(this);
  16. m_labShowA->setStyleSheet("background: #FA8694;");
  17. m_labShowB->setStyleSheet("background: #BDA1E2;");
  18. m_labShowB->resize(this->width(), this->height() / 5 * 2);
  19. m_labShowB->move(0, this->height() / 5 * 3);
  20. m_timLine = std::make_shared<QTimeLine>(600,this);
  21. m_timLine->setFrameRange(0, 100);
  22. connect(m_timLine.get(),&QTimeLine::valueChanged,this, &QtWidgetsDesktop::updateStep);
  23. }
  24. void QtWidgetsDesktop::updateStep(qreal val)
  25. {
  26. QTimeLine::Direction type = m_timLine->direction();
  27. if (type == QTimeLine::Forward)
  28. {
  29. m_labShowA->resize(val * this->width(), this->height() / 5 * 2);
  30. m_labShowB->resize((this->width() - val * this->width()), this->height() / 5 * 2);
  31. }
  32. else {
  33. m_labShowA->resize(val * this->width(), this->height() / 5 * 2);
  34. m_labShowB->resize((this->width() - val * this->width()), this->height() / 5 * 2);
  35. }
  36. }
  37. void QtWidgetsDesktop::enterEvent(QEvent* event)
  38. {
  39. m_timLine->setDirection(QTimeLine::Forward);
  40. if (m_timLine->state() != QTimeLine::Running)
  41. {
  42. m_timLine->start();
  43. }
  44. }
  45. void QtWidgetsDesktop::leaveEvent(QEvent* event)
  46. {
  47. m_timLine->setDirection(QTimeLine::Backward);
  48. if (m_timLine->state() != QTimeLine::Running)
  49. {
  50. m_timLine->start();
  51. }
  52. }

结果展示

CurveShape 设置为: EaseOutCurve
EaseOutCurve.gif
CurveShape 设置为: LinearCurve
LinearCurve.gif
CurveShape 设置为: SineCurve
SineCurve.gif