Android之自定義控制項-下拉刷新

来源:http://www.cnblogs.com/xmcx1995/archive/2016/10/12/5951772.html
-Advertisement-
Play Games

實現效果: 圖片素材: --> 首先, 寫先下拉刷新時的刷新佈局 pull_to_refresh.xml: 1 <resources> 2 <string name="app_name">PullToRefreshTest</string> 3 <string name="pull_to_refre ...


實現效果:

 

 

圖片素材:        

 

--> 首先, 寫先下拉刷新時的刷新佈局 pull_to_refresh.xml:

 1 <resources>
 2     <string name="app_name">PullToRefreshTest</string>
 3     <string name="pull_to_refresh">下拉可以刷新</string>
 4     <string name="release_to_refresh">釋放立即刷新</string>
 5     <string name="refreshing">正在刷新...</string>
 6     <string name="not_updated_yet">暫未更新過</string>
 7     <string name="updated_at">上次更新於%1$s前</string>
 8     <string name="updated_just_now">剛剛更新</string>
 9     <string name="time_error">時間有問題</string>
10 </resources>
strings
 1 <?xml version="1.0" encoding="utf-8"?>
 2 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
 3     android:id="@+id/pull_to_refresh_head"
 4     android:layout_width="match_parent"
 5     android:layout_height="60dp">
 6 
 7     <LinearLayout
 8         android:layout_width="200dp"
 9         android:layout_height="60dp"
10         android:layout_centerInParent="true"
11         android:orientation="horizontal">
12 
13         <RelativeLayout
14             android:layout_width="0dp"
15             android:layout_height="60dp"
16             android:layout_weight="3">
17 
18             <ImageView
19                 android:id="@+id/arrow"
20                 android:layout_width="wrap_content"
21                 android:layout_height="wrap_content"
22                 android:layout_centerInParent="true"
23                 android:src="@mipmap/indicator_arrow" />
24 
25             <ProgressBar
26                 android:id="@+id/progress_bar"
27                 android:layout_width="30dp"
28                 android:layout_height="30dp"
29                 android:layout_centerInParent="true"
30                 android:visibility="gone" />
31         </RelativeLayout>
32 
33         <LinearLayout
34             android:layout_width="0dp"
35             android:layout_height="60dp"
36             android:layout_weight="12"
37             android:orientation="vertical">
38 
39             <TextView
40                 android:id="@+id/description"
41                 android:layout_width="match_parent"
42                 android:layout_height="0dp"
43                 android:layout_weight="1"
44                 android:gravity="center_horizontal|bottom"
45                 android:text="@string/pull_to_refresh" />
46 
47             <TextView
48                 android:id="@+id/updated_at"
49                 android:layout_width="match_parent"
50                 android:layout_height="0dp"
51                 android:layout_weight="1"
52                 android:gravity="center_horizontal|top"
53                 android:text="@string/updated_at" />
54         </LinearLayout>
55     </LinearLayout>
56 
57 </RelativeLayout>
pull_to_refresh

 

--> 然後, 也是主要的, 自定義下拉刷新的 View (包含下拉刷新所有操作) RefreshView.java:

  1 package com.dragon.android.tofreshlayout;
  2 
  3 import android.content.Context;
  4 import android.content.SharedPreferences;
  5 import android.os.AsyncTask;
  6 import android.os.SystemClock;
  7 import android.preference.PreferenceManager;
  8 import android.util.AttributeSet;
  9 import android.view.LayoutInflater;
 10 import android.view.MotionEvent;
 11 import android.view.View;
 12 import android.view.ViewConfiguration;
 13 import android.view.animation.RotateAnimation;
 14 import android.widget.ImageView;
 15 import android.widget.LinearLayout;
 16 import android.widget.ListView;
 17 import android.widget.ProgressBar;
 18 import android.widget.TextView;
 19 
 20 public class RefreshView extends LinearLayout implements View.OnTouchListener {
 21 
 22     private static final String TAG = RefreshView.class.getSimpleName();
 23 
 24     public enum PULL_STATUS {
 25         STATUS_PULL_TO_REFRESH(0), // 下拉狀態
 26         STATUS_RELEASE_TO_REFRESH(1), // 釋放立即刷新狀態
 27         STATUS_REFRESHING(2), // 正在刷新狀態
 28         STATUS_REFRESH_FINISHED(3); // 刷新完成或未刷新狀態
 29 
 30         private int status; // 狀態
 31 
 32         PULL_STATUS(int value) {
 33             this.status = value;
 34         }
 35 
 36         public int getValue() {
 37             return this.status;
 38         }
 39     }
 40 
 41     // 下拉頭部回滾的速度
 42     public static final int SCROLL_SPEED = -20;
 43     // 一分鐘的毫秒值,用於判斷上次的更新時間
 44     public static final long ONE_MINUTE = 60 * 1000;
 45     // 一小時的毫秒值,用於判斷上次的更新時間
 46     public static final long ONE_HOUR = 60 * ONE_MINUTE;
 47     // 一天的毫秒值,用於判斷上次的更新時間
 48     public static final long ONE_DAY = 24 * ONE_HOUR;
 49     // 一月的毫秒值,用於判斷上次的更新時間
 50     public static final long ONE_MONTH = 30 * ONE_DAY;
 51     // 一年的毫秒值,用於判斷上次的更新時間
 52     public static final long ONE_YEAR = 12 * ONE_MONTH;
 53     // 上次更新時間的字元串常量,用於作為 SharedPreferences 的鍵值
 54     private static final String UPDATED_AT = "updated_at";
 55 
 56     // 下拉刷新的回調介面
 57     private PullToRefreshListener mListener;
 58 
 59     private SharedPreferences preferences; // 用於存儲上次更新時間
 60     private View header; // 下拉頭的View
 61     private ListView listView; // 需要去下拉刷新的ListView
 62 
 63     private ProgressBar progressBar; // 刷新時顯示的進度條
 64     private ImageView arrow; // 指示下拉和釋放的箭頭
 65     private TextView description; // 指示下拉和釋放的文字描述
 66     private TextView updateAt; // 上次更新時間的文字描述
 67 
 68     private MarginLayoutParams headerLayoutParams; // 下拉頭的佈局參數
 69     private long lastUpdateTime; // 上次更新時間的毫秒值
 70 
 71     // 為了防止不同界面的下拉刷新在上次更新時間上互相有衝突,使用id來做區分
 72     private int mId = -1;
 73 
 74     private int hideHeaderHeight; // 下拉頭的高度
 75 
 76     /**
 77      * 當前處理什麼狀態,可選值有 STATUS_PULL_TO_REFRESH, STATUS_RELEASE_TO_REFRESH, STATUS_REFRESHING 和 STATUS_REFRESH_FINISHED
 78      */
 79     private PULL_STATUS currentStatus = PULL_STATUS.STATUS_REFRESH_FINISHED;
 80 
 81     // 記錄上一次的狀態是什麼,避免進行重覆操作
 82     private PULL_STATUS lastStatus = currentStatus;
 83 
 84     private float yDown; // 手指按下時的屏幕縱坐標
 85 
 86     private int touchSlop; // 在被判定為滾動之前用戶手指可以移動的最大值。
 87 
 88     private boolean loadOnce; // 是否已載入過一次layout,這裡onLayout中的初始化只需載入一次
 89 
 90     private boolean ableToPull; // 當前是否可以下拉,只有ListView滾動到頭的時候才允許下拉
 91 
 92     /**
 93      * 下拉刷新控制項的構造函數,會在運行時動態添加一個下拉頭的佈局
 94      */
 95     public RefreshView(Context context, AttributeSet attrs) {
 96         super(context, attrs);
 97 
 98         preferences = PreferenceManager.getDefaultSharedPreferences(context);
 99         header = LayoutInflater.from(context).inflate(R.layout.pull_to_refresh, null, true);
100         progressBar = (ProgressBar) header.findViewById(R.id.progress_bar);
101         arrow = (ImageView) header.findViewById(R.id.arrow);
102         description = (TextView) header.findViewById(R.id.description);
103         updateAt = (TextView) header.findViewById(R.id.updated_at);
104         touchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
105 
106         refreshUpdatedAtValue();
107         setOrientation(VERTICAL);
108         addView(header, 0);
109 
110         //Log.d(TAG, "RefreshView Constructor() getChildAt(0): " + getChildAt(0));
111         //Log.d(TAG, "RefreshView Constructor() getChildAt(0): " + getChildAt(1));
112 
113 //        listView = (ListView) getChildAt(1);
114 //        listView.setOnTouchListener(this);
115     }
116 
117     /**
118      * 進行一些關鍵性的初始化操作,比如:將下拉頭向上偏移進行隱藏,給 ListView 註冊 touch 事件
119      */
120     @Override
121     protected void onLayout(boolean changed, int l, int t, int r, int b) {
122         super.onLayout(changed, l, t, r, b);
123         if (changed && !loadOnce) {
124             hideHeaderHeight = -header.getHeight();
125 
126             headerLayoutParams = (MarginLayoutParams) header.getLayoutParams();
127             headerLayoutParams.topMargin = hideHeaderHeight;
128             listView = (ListView) getChildAt(1);
129             //Log.d(TAG, "onLayout() getChildAt(0): " + getChildAt(0));
130             //Log.d(TAG, "onLayout() listView: " + listView);
131             listView.setOnTouchListener(this);
132             loadOnce = true;
133         }
134     }
135 
136     /**
137      * 當 ListView 被觸摸時調用,其中處理了各種下拉刷新的具體邏輯
138      */
139     @Override
140     public boolean onTouch(View v, MotionEvent event) {
141         setCanAbleToPull(event); // 判斷是否可以下拉
142         if (ableToPull) {
143             switch (event.getAction()) {
144                 case MotionEvent.ACTION_DOWN:
145                     yDown = event.getRawY();
146                     break;
147                 case MotionEvent.ACTION_MOVE:
148                     // 獲取移動中的 Y 軸的位置
149                     float yMove = event.getRawY();
150                     // 獲取從按下到移動過程中移動的距離
151                     int distance = (int) (yMove - yDown);
152 
153                     // 如果手指是上滑狀態,並且下拉頭是完全隱藏的,就屏蔽下拉事件
154                     if (distance <= 0 && headerLayoutParams.topMargin <= hideHeaderHeight) {
155                         return false;
156                     }
157                     if (distance < touchSlop) {
158                         return false;
159                     }
160                     // 判斷是否已經在刷新狀態
161                     if (currentStatus != PULL_STATUS.STATUS_REFRESHING) {
162                         // 判斷設置的 topMargin 是否 > 0, 預設初始設置為 -header.getHeight()
163                         if (headerLayoutParams.topMargin > 0) {
164                             currentStatus = PULL_STATUS.STATUS_RELEASE_TO_REFRESH;
165                         } else {
166                             // 否則狀態為下拉中的狀態
167                             currentStatus = PULL_STATUS.STATUS_PULL_TO_REFRESH;
168                         }
169                         // 通過偏移下拉頭的 topMargin 值,來實現下拉效果
170                         headerLayoutParams.topMargin = (distance / 2) + hideHeaderHeight;
171                         header.setLayoutParams(headerLayoutParams);
172                     }
173                     break;
174                 case MotionEvent.ACTION_UP:
175                 default:
176                     if (currentStatus == PULL_STATUS.STATUS_RELEASE_TO_REFRESH) {
177                         // 鬆手時如果是釋放立即刷新狀態,就去調用正在刷新的任務
178                         new RefreshingTask().execute();
179                     } else if (currentStatus == PULL_STATUS.STATUS_PULL_TO_REFRESH) {
180                         // 鬆手時如果是下拉狀態,就去調用隱藏下拉頭的任務
181                         new HideHeaderTask().execute();
182                     }
183                     break;
184             }
185             // 時刻記得更新下拉頭中的信息
186             if (currentStatus == PULL_STATUS.STATUS_PULL_TO_REFRESH
187                     || currentStatus == PULL_STATUS.STATUS_RELEASE_TO_REFRESH) {
188                 updateHeaderView();
189                 // 當前正處於下拉或釋放狀態,要讓 ListView 失去焦點,否則被點擊的那一項會一直處於選中狀態
190                 listView.setPressed(false);
191                 listView.setFocusable(false);
192                 listView.setFocusableInTouchMode(false);
193                 lastStatus = currentStatus;
194                 // 當前正處於下拉或釋放狀態,通過返回 true 屏蔽掉 ListView 的滾動事件
195                 return true;
196             }
197         }
198         return false;
199     }
200 
201     /**
202      * 給下拉刷新控制項註冊一個監聽器
203      *
204      * @param listener 監聽器的實現
205      * @param id       為了防止不同界面的下拉刷新在上次更新時間上互相有衝突,不同界面在註冊下拉刷新監聽器時一定要傳入不同的 id
206      */
207     public void setOnRefreshListener(PullToRefreshListener listener, int id) {
208         mListener = listener;
209         mId = id;
210     }
211 
212     /**
213      * 當所有的刷新邏輯完成後,記錄調用一下,否則你的 ListView 將一直處於正在刷新狀態
214      */
215     public void finishRefreshing() {
216         currentStatus = PULL_STATUS.STATUS_REFRESH_FINISHED;
217         preferences.edit().putLong(UPDATED_AT + mId, System.currentTimeMillis()).commit();
218         new HideHeaderTask().execute();
219     }
220 
221     /**
222      * 根據當前 ListView 的滾動狀態來設定 {@link #ableToPull}
223      * 的值,每次都需要在 onTouch 中第一個執行,這樣可以判斷出當前應該是滾動 ListView,還是應該進行下拉
224      */
225     private void setCanAbleToPull(MotionEvent event) {
226         View firstChild = listView.getChildAt(0);
227         if (firstChild != null) {
228             // 獲取 ListView 中第一個Item的位置
229             int firstVisiblePos = listView.getFirstVisiblePosition();
230             // 判斷第一個子控制項的 Top 是否和第一個 Item 位置相等
231             if (firstVisiblePos == 0 && firstChild.getTop() == 0) {
232                 if (!ableToPull) {
233                     // getRawY() 獲得的是相對屏幕 Y 方向的位置
234                     yDown = event.getRawY();
235                 }
236                 // 如果首個元素的上邊緣,距離父佈局值為 0,就說明 ListView 滾動到了最頂部,此時應該允許下拉刷新
237                 ableToPull = true;
238             } else {
239                 if (headerLayoutParams.topMargin != hideHeaderHeight) {
240                     headerLayoutParams.topMargin = hideHeaderHeight;
241                     header.setLayoutParams(headerLayoutParams);
242                 }
243                 ableToPull = false;
244             }
245         } else {
246             // 如果 ListView 中沒有元素,也應該允許下拉刷新
247             ableToPull = true;
248         }
249     }
250 
251     /**
252      * 更新下拉頭中的信息
253      */
254     private void updateHeaderView() {
255         if (lastStatus != currentStatus) {
256             if (currentStatus == PULL_STATUS.STATUS_PULL_TO_REFRESH) {
257                 description.setText(getResources().getString(R.string.pull_to_refresh));
258                 arrow.setVisibility(View.VISIBLE);
259                 progressBar.setVisibility(View.GONE);
260                 rotateArrow();
261             } else if (currentStatus == PULL_STATUS.STATUS_RELEASE_TO_REFRESH) {
262                 description.setText(getResources().getString(R.string.release_to_refresh));
263                 arrow.setVisibility(View.VISIBLE);
264                 progressBar.setVisibility(View.GONE);
265                 rotateArrow();
266             } else if (currentStatus == PULL_STATUS.STATUS_REFRESHING) {
267                 description.setText(getResources().getString(R.string.refreshing));
268                 progressBar.setVisibility(View.VISIBLE);
269                 arrow.clearAnimation();
270                 arrow.setVisibility(View.GONE);
271             }
272             refreshUpdatedAtValue();
273         }
274     }
275 
276     /**
277      * 根據當前的狀態來旋轉箭頭
278      */
279     private void rotateArrow() {
280         float pivotX = arrow.getWidth() / 2f;
281         float pivotY = arrow.getHeight() / 2f;
282         float fromDegrees = 0f;
283         float toDegrees = 0f;
284         if (currentStatus == PULL_STATUS.STATUS_PULL_TO_REFRESH) {
285             fromDegrees = 180f;
286             toDegrees = 360f;
287         } else if (currentStatus == PULL_STATUS.STATUS_RELEASE_TO_REFRESH) {
288             fromDegrees = 0f;
289             toDegrees = 180f;
290         }
291         RotateAnimation animation = new RotateAnimation(fromDegrees, toDegrees, pivotX, pivotY);
292         animation.setDuration(100);
293         animation.setFillAfter(true);
294         arrow.startAnimation(animation);
295     }
296 
297     /**
298      * 刷新下拉頭中上次更新時間的文字描述
299      */
300     private void refreshUpdatedAtValue() {
301         lastUpdateTime = preferences.getLong(UPDATED_AT + mId, -1);
302         long currentTime = System.currentTimeMillis();
303         long timePassed = currentTime - lastUpdateTime;
304         long timeIntoFormat;
305         String updateAtValue;
306         if (lastUpdateTime == -1) {
307             updateAtValue = getResources().getString(R.string.not_updated_yet);
308         } else if (timePassed < 0) {
309             updateAtValue = getResources().getString(R.string.time_error);
310         } else if (timePassed < ONE_MINUTE) {
311             updateAtValue = getResources().getString(R.string.updated_just_now);
312         } else if (timePassed < ONE_HOUR) {
313             timeIntoFormat = timePassed / ONE_MINUTE;
314             String value = timeIntoFormat + "分鐘";
315             updateAtValue = String.format(getResources().getString(R.string.updated_at), value);
316         } else if (timePassed < ONE_DAY) {
317             timeIntoFormat = timePassed / ONE_HOUR;
318             String value = timeIntoFormat + "小時";
319             updateAtValue = String.format(getResources().getString(R.string.updated_at), value);
320         } else if (timePassed < ONE_MONTH) {
321             timeIntoFormat = timePassed / ONE_DAY;
322             String value = timeIntoFormat + "天";
323             updateAtValue = String.format(getResources().getString(R.string.updated_at), value);
324         } else if (timePassed < ONE_YEAR) {
325             timeIntoFormat = timePassed / ONE_MONTH;
326             String value = timeIntoFormat + "個月";
327             updateAtValue = String.format(getResources().getString(R.string.updated_at), value);
328         } else {
329             timeIntoFormat = timePassed / ONE_YEAR;
330             String value = timeIntoFormat + "年";
331             updateAtValue = String.format(getResources().getString(R.string.updated_at), value);
332         }
333         updateAt.setText(updateAtValue);
334     }
335 
336     /**
337      * 正在刷新的任務,在此任務中會去回調註冊進來的下拉刷新監聽器
338      */
339     class RefreshingTask extends AsyncTask<Void, Integer, Void> {
340 
341         @Override
342         protected Void doInBackground(Void... params) {
343             int topMargin = headerLayoutParams.topMargin;
344             while (true) {
345                 

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

-Advertisement-
Play Games
更多相關文章
  • Android Weekly Issue #226筆記. 本期內容包括: 用Firebase做A/B Test; 用RxJava做動畫; MVP; proguardFiles; RxJava和Android Data Binding的結合; Mockito的更新; Gradle configurat... ...
  • 最近在開發一個大項目的時候遇到一個很頭疼的問題,由於項目代碼較多,每次都要編譯鏈接1分鐘左右,調試的時候很浪費時間,於是研究了一下如何提高編譯鏈接的速度,在這裡分享給大家。 提升編譯鏈接的速度主要有以下三個方式: 1. 提高XCode編譯時使用的線程數 [plain] view plain copy ...
  • Android啟動Service有兩種方法,一種是startService,一種是bindService。生命周期如下: 執行startService時,調用者如果沒有stopService,Service會一直在後臺運行。多次調用startService,該Service只能被創建一次,即該Ser ...
  • adb命令下pull的作用是從手機端向電腦端拷文件。 adb命令下push的作用是從電腦端向手機端拷文件。 adb push /Users/edwin/Downloads/**.apk ...
  • platform :ios, '7.0' xcodeproj 'Coach.xcodeproj' target 'Coach' do pod 'MBProgressHUD', '~> 0.9.2' pod 'AFNetworking', '~> 3.1.0' pod 'MJRefresh', '~> ...
  • 前面的文章已經實現相關的佈局,本文接著進行相關的功能實現 本文地址:http://www.cnblogs.com/wuyudong/p/5951794.html,轉載請註明出處。 讀取系統聯繫人 當點擊“選擇聯繫人”按鈕後,彈出聯繫人列表,讀取系統聯繫人分如下幾個步驟: 系統聯繫人提供了一個內容提供 ...
  • 2. 創建LocationManager類 3.獲取位置 ...
  • 距上一篇博客"APP引導頁的高度集成 - DHGuidePageHUD - ①"的發佈有一段時間了, 後來又在SDK中補充了一些新的內容進去但是一直沒來得及跟大家分享, 今天來跟大家分享一下, 還是一行代碼搞定APP引導頁, 廢話不多說直接進入主題! 如果還沒來得及看上一篇博客的話, 請大家點擊這裡 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...