qt-各种惊艳动画实例

本文详细介绍了使用Qt开发的一款BounceIndicator类,展示了如何创建具有惊艳动画效果的控件,包括关键代码片段和下载链接。
摘要由CSDN通过智能技术生成


一、演示效果

请添加图片描述

请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述

二、关键代码

#include "bounceindicator.h"

#include <QPainter>
#include <QPropertyAnimation>
#include <QParallelAnimationGroup>
#include <QTime>

inline QColor randomColor()
{
    srand(QTime::currentTime().msecsSinceStartOfDay());
    return QColor(qrand() % 255, qrand() % 255, qrand() % 255);
}

static const QSize ItemSize = QSize(30, 30);
static const int TopMargin = 10;
static const int BounceDuration = 200;
static const QEasingCurve::Type EasingCurve = QEasingCurve::Linear;

class BounceItem : public QWidget
{
    Q_OBJECT
public:
    explicit BounceItem(QWidget* parent = nullptr);
    void setColor(const QColor& clr) { m_color = clr; update(); }
    QColor color() const { return m_color; }

protected:
    void paintEvent(QPaintEvent* event);

private:
    QColor m_color = Qt::blue;
};

BounceItem::BounceItem(QWidget *parent) : QWidget(parent)
{
    setFixedSize(ItemSize);
}

void BounceItem::paintEvent(QPaintEvent *event)
{
    Q_UNUSED(event)
    QPainter painter(this);
    painter.setRenderHints(QPainter::Antialiasing);
    painter.setPen(Qt::NoPen);
    painter.setBrush(QBrush(m_color));
    painter.drawEllipse(rect());
}

class BounceIndicatorPrivate
{
public:
    void layoutItems();
    void makeBounce(int index);

    BounceIndicator* q_ptr;
    bool started = false;
    QList<BounceItem*> items;
    QPropertyAnimation* headAnimation = nullptr;
    QParallelAnimationGroup* parallelAnimGroup = nullptr;
};

void BounceIndicatorPrivate::layoutItems()
{
    int itemCount = items.size();
    static const int ItemSpace = 30;
    int horSpace = ItemSpace;
    for (int i = 0; i < items.size(); ++i) {
        items.at(i)->move(horSpace, (q_ptr->height() / 2)  - (ItemSize.height() / 2));
        horSpace += ItemSize.width() + ItemSpace;
    }
}

void BounceIndicatorPrivate::makeBounce(int index)
{
    if (index >= items.size())
        return;

    BounceItem* item = items.at(index);
    int x = item->x();
    int end = 20;
    int start = 50;

    QPropertyAnimation* anim = new QPropertyAnimation(item, "pos");
    anim->setEasingCurve(EasingCurve);
    anim->setDuration(BounceDuration);
    anim->setStartValue(QPoint(x, start));
    anim->setEndValue(QPoint(x, end));
    anim->start();

    if (index == 0 && !headAnimation)
        headAnimation = anim;

    q_ptr->connect(anim, &QAbstractAnimation::finished, [this, item, x, index, anim, start, end] () {
        QPropertyAnimation* reverseAnim = new QPropertyAnimation(item, "pos");
        reverseAnim->setEasingCurve(EasingCurve);
        reverseAnim->setDuration(BounceDuration);
        reverseAnim->setStartValue(QPoint(x, end));
        reverseAnim->setEndValue(QPoint(x, start));
        reverseAnim->start();

        makeBounce(index + 1);

        if (index == (items.size() - 1))
            q_ptr->connect(reverseAnim, SIGNAL(finished()), headAnimation, SLOT(start()));

        q_ptr->connect(reverseAnim, &QAbstractAnimation::finished, [anim] () {
//            anim->start();
        });
    });
}

BounceIndicator::BounceIndicator(QWidget *parent) : QWidget(parent)
{
    d_ptr = new BounceIndicatorPrivate;
    d_ptr->q_ptr = this;
    setNumberOfItems(7);
    setWindowFlags(Qt::FramelessWindowHint);
    setAttribute(Qt::WA_TranslucentBackground);
}

BounceIndicator::~BounceIndicator()
{
    delete d_ptr;
}

void BounceIndicator::setAnimationStarted(bool start)
{
    if (start) {

        if (d_ptr->items.isEmpty())
            return;

        int startIndex = 0;
        d_ptr->makeBounce(startIndex);

    } else {

    }
}

bool BounceIndicator::animationStarted() const
{
    return false;
}

void BounceIndicator::setNumberOfItems(int number)
{
    if (d_ptr->items.size() != number) {
        int delta = number - d_ptr->items.size();
        if (delta > 0) {
            for (int i = 0; i < delta; ++i) {
                BounceItem* item = new BounceItem(this);
                item->setColor(randomColor());
                d_ptr->items.append(item);
            }
        } else {
            for (int i = 0 ; i < qAbs(delta); ++i) {
                BounceItem* bi = d_ptr->items.takeLast();
                delete bi;
                d_ptr->items.pop_back();
            }
        }

        d_ptr->layoutItems();
    }
}

int BounceIndicator::numberOfItems() const
{
    return d_ptr->items.size();
}

QSize BounceIndicator::sizeHint() const
{
    return QSize(300, 80);
}

void BounceIndicator::resizeEvent(QResizeEvent* event)
{
    QWidget::resizeEvent(event);
    d_ptr->layoutItems();
}

#include "bounceindicator.moc"

三、下载链接

https://download.csdn.net/download/u013083044/88846392

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

进击的大海贼

联系博主,为您提供有价值的资源

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值