Android 屬性動畫實戰

来源:https://www.cnblogs.com/lanxingren/archive/2019/08/06/11309747.html
-Advertisement-
Play Games

Android 屬性動畫初戰,通過屬性動畫實現類似於美團外賣購物車消失顯示的動效。 ...


什麼是屬性動畫?

屬性動畫可以通過直接更改 View 的屬性來實現 View 動畫。例如:

  1. 通過不斷的更改 View 的坐標來實現讓 View 移動的效果;
  2. 通過不斷的更改 View 的背景來實現讓 View 的背景漸變的效果;
  3. 通過不斷的更改 View 的寬高來實現讓 View 變形的效果;
  4. ...

由此可見,利用屬性動畫幾乎可以處理任何的涉及到 View 的動畫效果。

實戰

具體的細節就不多說了,網上相應的教程也不少。這篇博客主要是來實現類似於美團外賣購物車的效果。

分析

首先分析購物車動畫具體的細節:在滑動過程中,“購物車”向右移動,直至一半隱藏到右側;在手指停留在屏幕中時,“購物車”還隱藏在右側;當手指離開屏幕,“購物車”在一定時間後重新移動回來

以上的動畫細節可以分析出和 RecycleView 的滾動事件息息相關,因此動畫就應該在 RecycleView  的滾動事件中實現。

實現基本佈局

上圖的藍色圖片既是我們要處理的 View 。

給 RecycleView 加上滾動事件

接下來給 RecycleView 加上滾動事件,滑動或者飛翔時圖片消失,當停止滑動時圖片顯示。

 1 // 給rv加上滾動事件
 2 rv.addOnScrollListener(new RecyclerView.OnScrollListener() {
 3     @Override
 4     public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
 5         switch (newState) {
 6             case RecyclerView.SCROLL_STATE_DRAGGING:// 滾動中
 7             case RecyclerView.SCROLL_STATE_SETTLING:// 飛翔中
 8                 iv.setVisibility(View.GONE);
 9                 break;
10             case RecyclerView.SCROLL_STATE_IDLE:// 停止滾動
11                 iv.setVisibility(View.VISIBLE);
12                 break;
13         }
14     }
15 });

效果圖:

實現消失的動畫

根據上面的圖可以發現觸發時機基本是沒問題了,接下來要做的是讓消失不突兀,加上消失的動畫。

消失的實質是 View 的 x 坐標從當前位置一直往右直到變為隱藏了一半,下麵讓我們來實現這個效果:

 1 // 消失動畫的基本屬性(從iv當前的x坐標一直到出了屏幕右側一半)
 2 disappearAnimator = ValueAnimator.ofFloat(iv.getX(), (float) (Utils.getScreenWidth(this) - iv.getWidth() / 2.0));
 3 disappearAnimator.setDuration(400);// 動畫持續時間
 4 disappearAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
 5     @Override
 6     public void onAnimationUpdate(ValueAnimator animation) {// Value更新事件
 7         float curValue = (float) animation.getAnimatedValue();
 8         iv.setX(curValue);
 9     }
10 });
11 
12 // 給rv加上滾動事件
13 rv.addOnScrollListener(new RecyclerView.OnScrollListener() {
14     @Override
15     public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
16         switch (newState) {
17             case RecyclerView.SCROLL_STATE_DRAGGING:// 滾動中
18             case RecyclerView.SCROLL_STATE_SETTLING:// 飛翔中
19                 disappearAnimator.start();
20                 break;
21             case RecyclerView.SCROLL_STATE_IDLE:// 停止滾動
22                 iv.setVisibility(View.VISIBLE);
23                 break;
24         }
25     }
26 });

然而發現實際上動畫是這樣的:

發現他是從最左邊一直移動到了最右邊,與我們的需求不符。

經調試發現,在 onCreate 的時候 iv 尚未初始化完畢,因此寬高以及坐標都還不能獲取到。所以獲取到的坐標以及寬度都是0。

所以可以在滾動事件中獲取 iv 的坐標以及寬高,更改後的代碼如下:

 1 // 給rv加上滾動事件
 2 rv.addOnScrollListener(new RecyclerView.OnScrollListener() {
 3     @Override
 4     public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
 5         // 獲取iv的坐標以及寬高
 6         if (0 == originX) {
 7             originX = iv.getX();
 8             ivWidth = iv.getWidth();
 9         }
10 
11         switch (newState) {
12             case RecyclerView.SCROLL_STATE_DRAGGING:// 滾動中
13             case RecyclerView.SCROLL_STATE_SETTLING:// 飛翔中
14                 // 消失動畫的基本屬性(從iv當前的x坐標一直到出了屏幕右側一半)
15                 if (disappearAnimator == null) {
16                     disappearAnimator = ValueAnimator.ofFloat(originX, (float) (screenWidth - ivWidth / 2.0));
17                     disappearAnimator.setDuration(400);// 動畫持續時間
18                     disappearAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
19                         @Override
20                         public void onAnimationUpdate(ValueAnimator animation) {// Value更新事件
21                             float curValue = (float) animation.getAnimatedValue();
22                             iv.setX(curValue);
23                         }
24                     });
25                 }
26 
27                 disappearAnimator.start();
28                 break;
29             case RecyclerView.SCROLL_STATE_IDLE:// 停止滾動
30                 iv.setVisibility(View.VISIBLE);
31                 break;
32         }
33     }
34 });

效果如下圖:

實現出現的動畫

既然已經實現了消失的動畫,那出現的動畫也就不難了。出現的實質是 View 的 x 坐標從右側一半一直運動到原始位置。

 1 case RecyclerView.SCROLL_STATE_IDLE:// 停止滾動
 2     // 出現動畫的基本屬性(從屏幕右側一半到原始位置)
 3     if (appearAnimator == null) {
 4         appearAnimator = ValueAnimator.ofFloat((float) (screenWidth - ivWidth / 2.0), originX);
 5         appearAnimator.setDuration(400);// 動畫持續時間
 6         appearAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
 7             @Override
 8             public void onAnimationUpdate(ValueAnimator animation) {// Value更新事件
 9                 float curValue = (float) animation.getAnimatedValue();
10                 iv.setX(curValue);
11             }
12         });
13     }
14 
15     appearAnimator.start();
16     break;

效果圖如下:

但是發現如果頻繁的滑動暫停的話動畫會衝突,因此需要做一些判定,如果動畫正在運行則不再重新開始動畫。改動後的代碼如下:

 1 switch (newState) {
 2     case RecyclerView.SCROLL_STATE_DRAGGING:// 滾動中
 3     case RecyclerView.SCROLL_STATE_SETTLING:// 飛翔中
 4         // 消失動畫的基本屬性(從iv當前的x坐標一直到出了屏幕右側一半)
 5         if (disappearAnimator == null) {
 6             disappearAnimator = ValueAnimator.ofFloat(originX, (float) (screenWidth - ivWidth / 2.0));
 7             disappearAnimator.setDuration(400);// 動畫持續時間
 8             disappearAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
 9                 @Override
10                 public void onAnimationUpdate(ValueAnimator animation) {// Value更新事件
11                     float curValue = (float) animation.getAnimatedValue();
12                     iv.setX(curValue);
13                 }
14             });
15         }
16 
17         // 如果消失動畫還未開始執行並且iv的位置在原始位置,則執行
18         if (!disappearAnimator.isStarted() && originX == iv.getX()) {
19             disappearAnimator.start();
20         }
21         break;
22     case RecyclerView.SCROLL_STATE_IDLE:// 停止滾動
23         // 出現動畫的基本屬性(從屏幕右側一半到原始位置)
24         if (appearAnimator == null) {
25             appearAnimator = ValueAnimator.ofFloat((float) (screenWidth - ivWidth / 2.0), originX);
26             appearAnimator.setDuration(400);// 動畫持續時間
27             appearAnimator.setStartDelay(700);// 延遲時間
28             appearAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
29                 @Override
30                 public void onAnimationUpdate(ValueAnimator animation) {// Value更新事件
31                     float curValue = (float) animation.getAnimatedValue();
32                     iv.setX(curValue);
33                 }
34             });
35         }
36 
37         // 如果出現動畫還未開始執行,則執行
38         if (!appearAnimator.isStarted()) {
39             appearAnimator.start();
40         }
41         break;
42 }

但是發現還是會有衝突,經檢測,發現是出現動畫和消失動畫互相干擾的緣故。當滑動已暫停但出現動畫還未執行完畢,此時重新滑動會觸發消失動畫

因此需要給出現動畫加一個延遲,並且如果處於非暫停狀態需要取消出現動畫。(也許美團外賣暫停一段時間才開始出現的原因就是防止用戶不停的滑動暫停吧。)

修改後的代碼如下:

case RecyclerView.SCROLL_STATE_DRAGGING:// 滾動中
case RecyclerView.SCROLL_STATE_SETTLING:// 飛翔中
    // 消失動畫的基本屬性(從iv當前的x坐標一直到出了屏幕右側一半)
    if (disappearAnimator == null) {
        disappearAnimator = ValueAnimator.ofFloat(originX, (float) (screenWidth - ivWidth / 2.0));
        disappearAnimator.setDuration(400);// 動畫持續時間
        disappearAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {// Value更新事件
                float curValue = (float) animation.getAnimatedValue();
                iv.setX(curValue);
            }
        });
    }

    // 如果此時出現動畫已開始,則取消
    if (appearAnimator != null && appearAnimator.isStarted()) {
        appearAnimator.cancel();
    }

    // 如果消失動畫還未開始執行並且iv的位置在原始位置,則執行
    if (!disappearAnimator.isStarted() && originX == iv.getX()) {
        disappearAnimator.start();
    }
    break;

最終效果如圖所示:

總結

  • 如果要實現其他的效果,例如淡入淡出等同理就可以實現;
  • 多個動畫對統一個 View 做變換時一定要註意動畫之間的衝突;
  • 屬性動畫+函數方程可以實現一些豐富多變的效果,待研究;
  • 本文的實現還是比較簡陋,最好的效果是動畫可以被打斷,由於比較麻煩,所以沒有實現。

Github地址:屬性動畫初戰

 

大家如果有什麼疑問或者建議可以通過評論或者郵件的方式聯繫我,歡迎大家的評論~


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • zookeeper是幹嘛的呢 Zookeeper的作用1.可以為客戶端管理少量的數據kvkey:是以路徑的形式表示的,那就意味著,各key之間有父子關係,比如/ 是頂層key用戶建的key只能在/ 下作為子節點,比如建一個key: /aa 這個key可以帶value數據也可以建一個key: /bb也 ...
  • 通過https://www.cnblogs.com/tree1123/p/11243668.html 已經對consumer有了一定的瞭解。producer比consumer要簡單一些。 一、舊版本producer 0.9.0.0版本以前,是由scala編寫的舊版本producer。 入口類:kaf ...
  • 恢復內容開始 Windows下備份mysql 第一步 編寫腳本 --user 用戶名 --password 密碼 --host 地址 --port 埠 --default-character-set 字元編碼 --all-databases 備份整個資料庫 (單單備份一個庫可用--database ...
  • 前言 由於最近在學習node+express,學習到持久化存儲章節需要連接mongodb資料庫,然後之前也有試過安裝mongodb但是失敗了,這次就找了很多資料,終於安裝完成了,故此記錄下來安裝步驟,提供給有需要的人. 安裝流程 1. 下載mongodb安裝包 官網地址:https://www.mo ...
  • Redis事務 事務提供了一種"將多個命令打包,一次性提交並按順序執行"的機制,提交後在事務執行中不會中斷。只有在執行完所有命令後才會繼續執行來自其他客戶的消息。 Redis中的使用 Redis通過multi,exec,discard,watch實現事務功能。 1. multi:開始事務 2. ex ...
  • ==註:wm_concat(str1) 11g 後不支持使用== LISTAGG函數用法 ...
  • mysql資料庫相關流程圖/原理圖 1.mysql主從複製原理圖 mysql主從複製原理是大廠後端的高頻面試題,瞭解mysql主從複製原理非常有必要。 主從複製原理,簡言之,就三步曲,如下: 主資料庫有個bin-log二進位文件,紀錄了所有增刪改Sql語句。(binlog線程) 從資料庫把主資料庫的 ...
  • 1、結構化查詢語言——SQL,關係型資料庫通信的標準語言; 2、關係型資料庫:表的邏輯單元組成,這些表在內部彼此關聯,組成了關係型資料庫; 3、SQL會話:用戶用SQL命令語句與關係型資料庫進行交互時發生的事情。當用戶與資料庫建立鏈接時,會話就建立了,當用戶與資料庫斷開連接時,會話就結束了。 con ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...