Qt之對話框淡出、飛入、最小化的動畫實現

轉自:http://www.javashuo.com/article/p-rrnqqzjp-ex.htmlhtml

1、效果展現

  最近作了一個提示框消失的功能,覺着挺有意思,之前一直覺得Qt子窗口不能作淡出效果,其實Qt的淡出功能已經幫咱們封裝好了,咱們僅僅只須要幾行代碼就能夠作出酷炫的窗口關閉效果,寫此篇文章的時候,我特地瀏覽了下以前寫的兩篇文章(QPainterPath 不規則提示框QPainterPath 不規則提示框(二)),如今回想起來那會兒確實知之甚少,關於頂層窗口不能作圓角,其實幫助文檔裏已經說的很明確,解決辦法有多種,一種是重寫paintEvent函數,另外一種是把widget包裝一層,本篇文章就用的是後一種方式,如圖1所示窗口關閉動畫,實例程序中作了淡出、飛出、縮小等關閉窗口動畫,除此以外還包含了陰影、背景着色、濾鏡等特效。函數

圖1 窗口特效測試

2、功能

如圖1窗口特效所示,實例中總共包含了4個groupbox,這4個groupbox是分別用來展現不一樣特效,下面分別講述4個groupbox動畫

  • 背景色:主要針對窗口背景色進行了定製,就像groupbox中按鈕文字那樣,是紅色和綠色的背景提示框,其中紅色提示框使用了最小化關閉效果,綠色提示框使用了淡出特效
  • 飛出:這4個按鈕彈出的對話框都使用了飛出特效,4個按鈕分別展現了4種飛出的方式(左、上、右、下)
  • 自定義:支持自定義提示框別景色、提示框展現時長、消失動畫時長和消失模式
  • shortcut:主要是針對業務進行的功能定製,warning提示框體的圖標是進行單獨處理的,是一個歎號圖標

3、代碼實現

在講解代碼以前,先來認識幾個概念this

  • QPropertyAnimation:屬性動畫,能夠參考qt 窗口動畫
  • QGraphicsOpacityEffect:窗口透明度設置類,繼承自QGraphicsEffect
  • QGraphicsDropShadowEffect:窗口陰影,繼承自QGraphicsEffect
  • QGraphicsBlurEffect:濾鏡,繼承自QGraphicsEffect
  • QGraphicsColorizeEffect:着色,繼承自QGraphicsEffect

一、移出動畫,使用屬性動畫QPropertyAnimation類進行,propertyname的參數是窗口的屬性,詳情參見Q_PROPERTY屬性 。targetObject對象設置爲this內部單獨封裝的widget,這樣作的目的使得該提示框不須要依賴其餘窗口遮擋便可作出飛出效果url

1 void GMPOperateTip::MoveOut()
 2 {
 3     m_pAnimation->setTargetObject(m_pMoveWidget);
 4     m_pAnimation->setPropertyName("pos");
 5 
 6     m_pAnimation->setStartValue(QPoint());
 7     switch (m_eDirection)
 8     {
 9     case D_LEFT:
10         m_pAnimation->setEndValue(QPoint(-width(), 0));
11         break;
12     case D_TOP:
13         m_pAnimation->setEndValue(QPoint(0, -height()));
14         break;
15     case D_RIGHT:
16         m_pAnimation->setEndValue(QPoint(width(), 0));
17         break;
18     case D_BOTTOM:
19         m_pAnimation->setEndValue(QPoint(0, height()));
20         break;
21     default:
22         ;
23     }
24 }

二、淡出spa

1 m_pOpacity = new QGraphicsOpacityEffect(this);
 2 m_pOpacity->setOpacity(1);
 3 
 4 setGraphicsEffect(m_pOpacity);
 5 
 6 m_pAnimation->setTargetObject(m_pOpacity);
 7 m_pAnimation->setPropertyName("opacity");
 8 
 9 m_pAnimation->setStartValue(1);
10 m_pAnimation->setEndValue(0);

三、最小化.net

1 m_pAnimation->setPropertyName("geometry");
2 
3 QRect startRect = rect();
4 startRect.moveTo(pos());
5 QRect stopRect = QRect(startRect.center(), QSize(0, 0));
6 
7 m_pAnimation->setStartValue(startRect);
8 m_pAnimation->setEndValue(stopRect);

四、動畫啓動機制htm

使用定時器控制動畫,當指定時間後啓動動畫,而且在動畫完成後關閉窗口對象

1 void InitializeConnect()
 2 {
 3     m_pAnimation = new QPropertyAnimation(this);
 4     m_pAnimation->setTargetObject(this);
 5 
 6     connect(m_pAnimation, &QPropertyAnimation::finished, this, &GMPOperateTip::close);
 7 
 8     connect(&m_StayTimer, &QTimer::timeout, this, [this]{
 9         m_pAnimation->setDuration(m_DurationTime);
10         switch (m_eMode)
11         {
12         case AM_FADEOUT:
13             FadeOut_p();
14             break;
15         case AM_FLYOUT:
16             MoveOut();
17             break;
18         case AM_ZOOMIN:
19             ZoomIn();
20             break;
21         default:
22             ;
23         }
24 
25         m_pAnimation->start();
26     });
27 }

窗口顯示時啓動定時器,而且將窗口隨機移動到屏幕一個位置

1 bool event(QEvent * e)
 2 {
 3     if (e->type() == QEvent::Show)
 4     {
 5         //QPoint pos = parentWidget()->rect().center() - this->rect().center();
 6         int wrand = qrand() % (parentWidget()->rect().width() - this->rect().width());
 7         int hrand = qrand() % (parentWidget()->rect().height() - this->rect().width());
 8         move(QPoint(wrand, hrand));
 9 
10         m_StayTimer.start(m_iStayDuration);
11     }
12 
13     return __super::event(e);
14 }

五、陰影

1 void setShadowEnable(bool enable)
 2 {
 3     if (!m_pShadow)
 4     {
 5         m_pShadow = new QGraphicsDropShadowEffect(this);
 6         m_pShadow->setColor(QColor(0, 0, 0, 85));
 7         m_pShadow->setBlurRadius(10);
 8         m_pShadow->setOffset(4, 4);
 9     }
10 
11     setGraphicsEffect(enable ? m_pShadow : nullptr);
12 }

六、着色

註釋中的代碼也能夠進行着色,可是窗體的一些特殊樣式不能完成,所以使用stylesheet來完成背景色修改

1 static const QString c_szStyleSheet = "QWidget{background-color:%1;\
2                                     border:1px solid %2;border-top:0;border-bottom-left-radius:3px;\
3                                     border-bottom-right-radius:3px;background-image: url();}";
1 void GMPOperateTip::setBackgroundColor(const QColor & color)
 2 {
 3     //if (!m_pColorize)
 4     //{
 5     //    m_pColorize = new QGraphicsColorizeEffect(this);
 6     //    m_pColorize->setStrength(1);
 7     //    
 8     //    setGraphicsEffect(m_pColorize);
 9     //}
10     //m_pColorize->setColor(color);
11 
12     QColor border = color;
13     border.setAlpha(255 * 0.1);
14     QString borderRgba = QString("rgba(%1,%2,%3,%4)").arg(border.red()).arg(border.green()).arg(border.blue()).arg(border.alpha());
15     setStyleSheet(c_szStyleSheet.arg(color.name()).arg(borderRgba));
16 }

 七、快捷調用接口,該接口都是類的靜態方法能夠直接調用

View Code

八、測試,因爲測試代碼較多,我只貼出2個

1 void tip::on_pushButton_success_clicked()
2 {
3     GMPOperateTip::Success(this, QStringLiteral("測a試º?,ê?測a試º?"), 1000, 1000);
4 }
5 
6 void tip::on_pushButton_warning_clicked()
7 {
8     GMPOperateTip::Waring(this, QStringLiteral("測a試º?,ê?測a試º?"), 1000, 1000);
9 }

4、demo程序

  VS版本:動畫提示框

      Qt Creator版本:動畫提示框 qt5.8

相關文章
相關標籤/搜索