Android自定義listview布局實(shí)現(xiàn)上拉加載下拉刷新功能
listview實(shí)現(xiàn)上拉加載以及下拉刷新的方式有很多。下面是我寫(xiě)的一種自定義的布局,復(fù)用性也比較的強(qiáng)。首先就是繼承的listview的自定義view。
AutoListView.Java:
package com.example.mic.testdemo.view; import android.annotation.TargetApi; import android.content.Context; import android.os.Build; import android.os.Bundle; import android.util.AttributeSet; import android.util.Log; import android.view.LayoutInflater; import android.view.MotionEvent; import android.view.View; import android.view.ViewGroup; import android.view.animation.Animation; import android.view.animation.LinearInterpolator; import android.view.animation.RotateAnimation; import android.widget.AbsListView; import android.widget.ImageView; import android.widget.ListView; import android.widget.ProgressBar; import android.widget.TextView; import com.example.mic.testdemo.R; import java.text.SimpleDateFormat; /** * Created by DFLENOVO on 2016/8/3. */ public class AutoListView extends ListView implements AbsListView.OnScrollListener { public static final String FOOTER = "FOOTER"; public static final String HEADER = "HEADER"; private static final int PULL = 1; private static final int NONE = 0; // private static final int RELEASE = 2; private static final int REFRESHING = 2; private static final int SPACE = 20; private static final String TAG = "AutoListView"; private RotateAnimation downAnimation; private RotateAnimation upAnimation; private Context context; private LayoutInflater inflater; private View footer; private View header; private int headerContentInitialHeight; private int headerContentHeight; private OnRefreshListener onRefreshListener; private boolean loadEnable; private OnLoadListener onLoadListener; private int firstVisibleItem; private boolean isLoadingMore = false; private int startY; private int state; private ImageView iv_pull; private int footerViewHeight; private ProgressBar mProgressBar; private TextView tvState; private TextView tvLastUpdateTime; private boolean isScrollToBottom; public AutoListView(Context context) { super(context); initView(context); } public AutoListView(Context context, AttributeSet attrs) { super(context, attrs); initView(context); } public AutoListView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); initView(context); } @TargetApi(Build.VERSION_CODES.LOLLIPOP) public AutoListView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { super(context, attrs, defStyleAttr, defStyleRes); initView(context); } private void initView(Context context) { initHeaderView(); initFooterView(); this.setOnScrollListener(this); // } private void initHeaderView() { header = View.inflate(getContext(), R.layout.pull_to_refresh_header, null); iv_pull = (ImageView) header .findViewById(R.id.iv_pull); mProgressBar = (ProgressBar) header .findViewById(R.id.pb_listview_header); tvState = (TextView) header .findViewById(R.id.tv_listview_header_state); tvLastUpdateTime = (TextView) header .findViewById(R.id.tv_listview_header_last_update_time); // 設(shè)置最后刷新時(shí)間 tvLastUpdateTime.setText("最后刷新時(shí)間: " + getLastUpdateTime()); header.measure(0, 0); // 系統(tǒng)會(huì)幫我們測(cè)量出headerView的高度 headerContentHeight = header.getMeasuredHeight(); header.setPadding(0, -headerContentHeight, 0, 0); this.addHeaderView(header,HEADER,true); // 向ListView的頂部添加一個(gè)view對(duì)象 initAnimation(); } private void initAnimation() { upAnimation = new RotateAnimation(0f, -180f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f); upAnimation.setDuration(500); upAnimation.setFillAfter(true); // 動(dòng)畫(huà)結(jié)束后, 停留在結(jié)束的位置上 downAnimation = new RotateAnimation(-180f, -360f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f); downAnimation.setDuration(500); downAnimation.setFillAfter(true); // 動(dòng)畫(huà)結(jié)束后, 停留在結(jié)束的位置上 } private String getLastUpdateTime() { SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); return sdf.format(System.currentTimeMillis()); } private void initFooterView() { footer = View.inflate(getContext(), R.layout.foot_view, null); footer.measure(0, 0); footerViewHeight = footer.getMeasuredHeight(); footer.setPadding(0, -footerViewHeight, 0, 0); this.addFooterView(footer,FOOTER,true); } private void measureView(View child) { ViewGroup.LayoutParams p = child.getLayoutParams(); if (p == null) { p = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT); } int childWidthSpec = ViewGroup.getChildMeasureSpec(0, 0 + 0, p.width); int lpHeight = p.height; int childHeightSpec; if (lpHeight > 0) { childHeightSpec = MeasureSpec.makeMeasureSpec(lpHeight, MeasureSpec.EXACTLY);//得到的是size。 } else { childHeightSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);//得到的是子布局的實(shí)際大小。 } child.measure(childWidthSpec, childHeightSpec); } private void topPadding(int topPadding) { header.setPadding(header.getPaddingLeft(), topPadding, header.getPaddingRight(), header.getPaddingBottom()); header.invalidate(); } @Override public void onScrollStateChanged(AbsListView absListView, int i) { if (i == SCROLL_STATE_IDLE || i == SCROLL_STATE_FLING) { // 判斷當(dāng)前是否已經(jīng)到了底部 if (isScrollToBottom && !isLoadingMore) { isLoadingMore = true; // 當(dāng)前到底部 Log.i(TAG, "加載更多數(shù)據(jù)"); footer.setPadding(0, 0, 0, 0); this.setSelection(this.getCount()); if (onLoadListener != null) { onLoadListener.onLoad(); } } } } @Override public void onScroll(AbsListView absListView, int i, int i1, int i2) { this.firstVisibleItem = i; if (getLastVisiblePosition() == (i2 - 1)) { isScrollToBottom = true; } else { isScrollToBottom = false; } } @Override public boolean onTouchEvent(MotionEvent ev) { switch (ev.getAction()) { // case MotionEvent.ACTION_DOWN: startY = (int) ev.getY(); break; case MotionEvent.ACTION_MOVE: int moveY = (int) ev.getY(); // 移動(dòng)中的y - 按下的y = 間距. int diff = (moveY - startY) / 2; // -頭布局的高度 + 間距 = paddingTop int paddingTop = -headerContentHeight + diff; // 如果: -頭布局的高度 > paddingTop的值 執(zhí)行super.onTouchEvent(ev); if (firstVisibleItem == 0 && -headerContentHeight < paddingTop) { if (paddingTop > 0 && state == NONE) { // 完全顯示了. Log.i(TAG, "松開(kāi)刷新"); state = PULL; refreshHeaderViewByState(); } else if (paddingTop < 0 && state == PULL) { // 沒(méi)有顯示完全 Log.i(TAG, "下拉刷新"); state = NONE; refreshHeaderViewByState(); } // 下拉頭布局 header.setPadding(0, paddingTop, 0, 0); } break; case MotionEvent.ACTION_UP: // 判斷當(dāng)前的狀態(tài)是松開(kāi)刷新還是下拉刷新 if (state == PULL) { Log.i(TAG, "刷新數(shù)據(jù)."); // 把頭布局設(shè)置為完全顯示狀態(tài) header.setPadding(0, 0, 0, 0); // 進(jìn)入到正在刷新中狀態(tài) state = REFRESHING; refreshHeaderViewByState(); if (onRefreshListener != null) { onRefreshListener.onRefresh(); // 調(diào)用使用者的監(jiān)聽(tīng)方法 } } else if (state == NONE) { // 隱藏頭布局 header.setPadding(0, -headerContentHeight, 0, 0); } break; default: break; } return super.onTouchEvent(ev); } // private void refreshHeaderViewByState() { switch (state) { // case NONE: // 下拉刷新?tīng)顟B(tài) tvState.setText("下拉刷新"); iv_pull.startAnimation(downAnimation); // 執(zhí)行向下旋轉(zhuǎn) break; case PULL: // 松開(kāi)刷新?tīng)顟B(tài) tvState.setText("松開(kāi)刷新"); iv_pull.startAnimation(upAnimation); // 執(zhí)行向上旋轉(zhuǎn) break; case REFRESHING: // 正在刷新中狀態(tài) iv_pull.clearAnimation(); iv_pull.setVisibility(View.GONE); mProgressBar.setVisibility(View.VISIBLE); tvState.setText("正在刷新中..."); break; default: break; } } public void hideHeaderView() { header.setPadding(0, -headerContentHeight, 0, 0); iv_pull.setVisibility(View.VISIBLE); mProgressBar.setVisibility(View.GONE); tvState.setText("下拉刷新"); tvLastUpdateTime.setText("最后刷新時(shí)間: " + getLastUpdateTime()); state = NONE; } public void hideFooterView() { footer.setPadding(0, -footerViewHeight, 0, 0); isLoadingMore = false; } public void setOnRefreshListener(OnRefreshListener onRefreshListener) { this.onRefreshListener = onRefreshListener; } public void setOnLoadListener(OnLoadListener onLoadListener) { this.onLoadListener = onLoadListener; } public void onRefresh() { if (onRefreshListener != null) { onRefreshListener.onRefresh(); } } public void onLoad() { if (onLoadListener != null) { onLoadListener.onLoad(); } } public interface OnRefreshListener {//定義下拉刷新接口 public void onRefresh(); } public interface OnLoadListener {//定義上拉加載更多 public void onLoad(); } }
上面的代碼就是實(shí)現(xiàn)的關(guān)鍵.下面是頭布局以及腳布局:
foot_view.xml:
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="加載更多" android:textSize="20dp" android:padding="10dp" android:layout_centerHorizontal="true"/> </RelativeLayout>
pull_to_refresh_header.xml:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="horizontal"> <FrameLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_margin="10dip" > <ImageView android:id="@+id/iv_pull" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center" android:minWidth="30dip" android:src="@mipmap/xlistview_arrow" /> <ProgressBar android:id="@+id/pb_listview_header" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center" android:visibility="gone" /> </FrameLayout> <LinearLayout android:layout_width="fill_parent" android:layout_height="wrap_content" android:layout_gravity="center_vertical" android:gravity="center_horizontal" android:orientation="vertical" > <TextView android:id="@+id/tv_listview_header_state" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="下拉刷新" android:textColor="#FF0000" android:textSize="18sp" /> <TextView android:id="@+id/tv_listview_header_last_update_time" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginTop="5dip" android:text="最后刷新時(shí)間: 2014-10-10 12:56:12" android:textColor="@android:color/black" android:textSize="14sp" /> </LinearLayout> </LinearLayout>
activity_main.xml:
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context="com.example.mic.testdemo.MainActivity"> <com.example.mic.testdemo.view.AutoListView android:id="@+id/lv" android:layout_width="match_parent" android:layout_height="match_parent"> </com.example.mic.testdemo.view.AutoListView> </RelativeLayout>
上面所完成的步驟就是在listview上下加載不同的布局,上面也是要復(fù)用的代碼,下面的代碼就是怎么使用的代碼了.
MainActivity.java:
public class MainActivity extends AppCompatActivity implements AutoListView.OnLoadListener,AutoListView.OnRefreshListener { private TextView textView; private String result; private AutoListView listView; private MyAdater adater; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); listView = (AutoListView) findViewById(R.id.lv);} listView.setOnRefreshListener(MainActivity.this); listView.setOnLoadListener(MainActivity.this); @Override public void onLoad() { } @Override public void onRefresh() {}
下面就是實(shí)現(xiàn),就是在你需要的地方設(shè)置監(jiān)聽(tīng),以及上面加載以及刷新需要的操作就可以了!
以上所述是小編給大家介紹的Android自定義listview布局實(shí)現(xiàn)上拉加載下拉刷新功能,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
- Android ExpandableListView實(shí)現(xiàn)下拉刷新和加載更多效果
- Android ListView實(shí)現(xiàn)下拉頂部圖片變大效果
- android使用SwipeRefreshLayout實(shí)現(xiàn)ListView下拉刷新上拉加載
- android使用PullToRefresh框架實(shí)現(xiàn)ListView下拉刷新上拉加載更多
- android ListView結(jié)合x(chóng)utils3仿微信實(shí)現(xiàn)下拉加載更多
- Android中ListView下拉刷新的實(shí)現(xiàn)代碼
- Android自定義漸變式炫酷ListView下拉刷新動(dòng)畫(huà)
- Android ListView實(shí)現(xiàn)上拉加載下拉刷新和滑動(dòng)刪除功能
- Android XListView下拉刷新和上拉加載更多
- Android自定義控件ListView下拉刷新的代碼
相關(guān)文章
Android實(shí)現(xiàn)向本地寫(xiě)入一個(gè)XML文件和解析XML文件
這篇文章主要介紹了Android實(shí)現(xiàn)向本地寫(xiě)入一個(gè)XML文件和解析XML文件,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Android應(yīng)用強(qiáng)制更新APP的示例代碼
本篇文章主要介紹了Android應(yīng)用強(qiáng)制更新APP的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02Android7.0中關(guān)于ContentProvider組件詳解
本文描述了Android7.0中關(guān)于ContentProvider組件實(shí)現(xiàn)原理以及ContentProvider發(fā)布者和調(diào)用者這兩在Framework層是如何實(shí)現(xiàn)的。2017-11-11Android實(shí)現(xiàn)Window彈窗效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)Window彈窗效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10Android編程實(shí)現(xiàn)QQ表情的發(fā)送和接收完整實(shí)例(附源碼)
這篇文章主要介紹了Android編程實(shí)現(xiàn)QQ表情的發(fā)送和接收的方法,涉及Android圖片資源、正則表達(dá)式及對(duì)話(huà)框的相關(guān)操作技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-11-11