3年以上勿进!最简单的Android自定义ListView下拉刷新与上拉加载,代码直接拿去用(1)

android:layout_height=“wrap_content”
android:text=“下拉刷新”
android:textColor=“#FF0000”
android:textSize=“18sp” />


底部布局文件:

<?xml version="1.0" encoding="utf-8"?>

二:自定义ListView需要的接口回调给UI,告诉UI ListView执行了下拉加载/上拉加载动作

public interface ICustomUpdateListViewBack {

public void downUpdateListData();

public void upUpdateListData();

}

自定义ListView:

public class CustomUpdateListView extends ListView implements AbsListView.OnScrollListener{

private static final String TAG = CustomUpdateListView.class.getSimpleName();

/**

  • 下拉刷新
    */
    private static final int DOWN_UPDATE = 111;

/**

  • 准备刷新
    */
    private static final int PLAN_UPDATE = 112;

/**

  • 正在刷新
    */
    private static final int PROCESS_UPDATE = 113;

private int thisUpdateStatusValue = DOWN_UPDATE; // 默认一直是下拉刷新

public CustomUpdateListView(Context context, AttributeSet attrs) {
super(context, attrs);

setOnScrollListener(this);

initHeader();
initBottom();
}

/**

  • 定义头部相关
    */
    private View headerView;
    private int headerViewHeight;

private ImageView ivHeaderArrow;
private ProgressBar pbHeader;
private TextView tvHeaderState;
private TextView tvHeaderLastUpdateTime;

/**

  • 定义底部相关
    /
    private View bottomView;
    private int bottomViewHeight;
    private TextView tvBottomState;
    /
    *
  • 初始化头部 布局View相关
    */
    private void initHeader() {
    // 从布局中拿到一个View
    headerView = View.inflate(getContext(), R.layout.listview_header, null);

// 获取头部各个控件的值
ivHeaderArrow = headerView.findViewById(R.id.iv_listview_header_arrow);
pbHeader = headerView.findViewById(R.id.pb_listview_header);
tvHeaderState = headerView.findViewById(R.id.tv_listview_header_state);
tvHeaderLastUpdateTime = headerView.findViewById(R.id.tv_listview_header_last_update_time);

tvHeaderLastUpdateTime.setText(getThisTiem());

// getHieight(); 方法只能获取到控件显示后的高度
// int headerViewHeight = headerView.getHeight();
// 结果 headerViewHeight: 0

// View的绘制流程:测量 onLayout onDraw

// 所以先测量后,就能得到测量后的高度了
headerView.measure(0, 0); // 注意:传0系统会自动去测量View高度

// 得到测量后的高度
headerViewHeight = headerView.getMeasuredHeight();
Log.i(TAG, “headerViewHeight:” + headerViewHeight);

headerView.setPadding(0, -headerViewHeight, 0 ,0);

addHeaderView(headerView);

initHeaderAnimation();
}

private void initBottom() {
bottomView = View.inflate(getContext(), R.layout.listview_bottom, null);

tvBottomState = bottomView.findViewById(R.id.tv_bottom_state);

// 先测量
bottomView.measure(0, 0);

// 获取高度
bottomViewHeight = bottomView.getMeasuredHeight();

bottomView.setPadding(0, -bottomViewHeight, 0, 0);

addFooterView(bottomView);

}

private RotateAnimation upRotateAnimation;
private RotateAnimation downRotateAnimation;

private void initHeaderAnimation() {
upRotateAnimation = new RotateAnimation(
0, 180,
Animation.RELATIVE_TO_SELF, 0.5f,
Animation.RELATIVE_TO_SELF, 0.5f);
upRotateAnimation.setDuration(500);
upRotateAnimation.setFillAfter(true);

downRotateAnimation = new RotateAnimation(
180, 360,
Animation.RELATIVE_TO_SELF, 0.5f,
Animation.RELATIVE_TO_SELF, 0.5f);
downRotateAnimation.setDuration(500);
downRotateAnimation.setFillAfter(true);
}

/**

  • 滑动的状态改变
  • @param view
  • @param scrollState 有三种状态
  •                SCROLL_STATE_IDLE 代表 滑动停止状态类似于手指松开UP
    
  •                SCROLL_STATE_TOUCH_SCROLL 代表滑动触摸状态
    
  •                SCROLL_STATE_FLING 快速滑动 猛的一滑
    

*/
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
// 如果是猛地滑动 或者 手指松开UP 才显示底部布局View
if (scrollState == SCROLL_STATE_IDLE || scrollState == SCROLL_STATE_FLING) {
// 判断必须是底部的Item的时候
if (getLastVisiblePosition() == (getCount() -1)) {
bottomView.setPadding(0, 0, 0, 0);

// 回调接口方法
if (null != customUpdateListViewBack) {
customUpdateListViewBack.upUpdateListData();
}
}
}
}

private int firstVisibleItem;

/**

  • ListView滑动的监听方法
  • @param view 当前ListView
  • @param firstVisibleItem 当前屏幕的第一个显示的Item
  • @param visibleItemCount 当前屏幕显示的Item数量
  • @param totalItemCount 总共Item数量
    */
    @Override
    public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
    this.firstVisibleItem = firstVisibleItem;
    }

private int downY;

@Override
public boolean onTouchEvent(MotionEvent ev) {
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:
downY = (int) ev.getY();
break;
case MotionEvent.ACTION_UP:
if (thisUpdateStatusValue == DOWN_UPDATE) {
headerView.setPadding(0, -headerViewHeight ,0 ,0);
} else {
headerView.setPadding(0, 0, 0, 0);
thisUpdateStatusValue = PROCESS_UPDATE;
updateHeaderState();
}
break;
case MotionEvent.ACTION_MOVE:
int cha = (int) ev.getY() - downY;
if (this.firstVisibleItem == 0 && cha > 0) {
int paddingTop = -headerViewHeight + cha;
// Log.i(TAG, “paddingTop:” + paddingTop);

if (thisUpdateStatusValue == PROCESS_UPDATE) {
break;
}

if (paddingTop > 0 && thisUpdateStatusValue == DOWN_UPDATE) {

最后

针对Android程序员,我这边给大家整理了一些资料,包括不限于高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter等全方面的Android进阶实践技术;希望能帮助到大家,也节省大家在网上搜索资料的时间来学习,也可以分享动态给身边好友一起学习!

  • Android前沿技术大纲

  • 全套体系化高级架构视频

资料领取:点赞+点击GitHub免费获取

往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、混合式开发(ReactNative+Weex)全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。

30972028)]

  • 全套体系化高级架构视频

    [外链图片转存中…(img-Gv2NmYeW-1710330972029)]

资料领取:点赞+点击GitHub免费获取

往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、混合式开发(ReactNative+Weex)全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。

  • 18
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
实现上刷新、下ListView,可以使用第三方库,如 Google 推荐的 SwipeRefreshLayout,或者自己实现。 下面是自己实现的一种方法: 1. 在布局文件中ListView 和 ProgressBar: ``` <RelativeLayout android:layout_width="match_parent" android:layout_height="match_parent"> <ListView android:id="@+id/listView" android:layout_width="match_parent" android:layout_height="match_parent" /> <ProgressBar android:id="@+id/progressBar" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerInParent="true" android:visibility="gone" /> </RelativeLayout> ``` 2. 在 Activity 或 Fragment 中初始化 ListView 和适配器: ``` listView = findViewById(R.id.listView); listView.setAdapter(adapter); ``` 3. 给 ListView 设置滚动监听器,当滚动到底部时,执行更多的操作: ``` listView.setOnScrollListener(new AbsListView.OnScrollListener() { @Override public void onScrollStateChanged(AbsListView view, int scrollState) { if (scrollState == AbsListView.OnScrollListener.SCROLL_STATE_IDLE) { if (view.getLastVisiblePosition() == view.getCount() - 1) { // 滚动到底部,执行更多操作 loadMoreData(); } } } @Override public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) { } }); ``` 4. 在更多的操作中,显示 ProgressBar,数据后更新适配器,隐藏 ProgressBar: ``` private void loadMoreData() { progressBar.setVisibility(View.VISIBLE); // 数据 // 更新适配器 progressBar.setVisibility(View.GONE); } ``` 5. 实现下刷新,可以使用 SwipeRefreshLayout,在布局文件中入 SwipeRefreshLayout 和 ListView: ``` <android.support.v4.widget.SwipeRefreshLayout android:id="@+id/swipeRefreshLayout" android:layout_width="match_parent" android:layout_height="match_parent"> <ListView android:id="@+id/listView" android:layout_width="match_parent" android:layout_height="match_parent" /> </android.support.v4.widget.SwipeRefreshLayout> ``` 6. 在 Activity 或 Fragment 中初始化 SwipeRefreshLayout 和 ListView,并给 SwipeRefreshLayout 设置刷新监听器: ``` swipeRefreshLayout = findViewById(R.id.swipeRefreshLayout); swipeRefreshLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() { @Override public void onRefresh() { // 下刷新,重新数据 refreshData(); } }); ``` 7. 在刷新数据的操作中,显示 ProgressBar,数据后更新适配器,隐藏 ProgressBar 和 SwipeRefreshLayout: ``` private void refreshData() { progressBar.setVisibility(View.VISIBLE); // 数据 // 更新适配器 progressBar.setVisibility(View.GONE); swipeRefreshLayout.setRefreshing(false); } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值