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

  • 定义头部相关
    */
    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) {
// 准备刷新
Log.i(TAG, “paddingTop:” + paddingTop + “>>>准备刷新”);
thisUpdateStatusValue = PLAN_UPDATE;

updateHeaderState();

} else if (paddingTop < 0 && thisUpdateStatusValue == PLAN_UPDATE) {
// 正在刷新
Log.i(TAG, “paddingTop:” + paddingTop + “>>>正在刷新”);
thisUpdateStatusValue = DOWN_UPDATE;

updateHeaderState();
}

headerView.setPadding(0, paddingTop, 0, 0);
}
break;
default:
break;
}
return super.onTouchEvent(ev); // 不返回ture 而是去调用父类的方法,是保证ListView自身的滑动功能正常
}

private void updateHeaderState() {

switch (thisUpdateStatusValue) {
case DOWN_UPDATE:
ivHeaderArrow.startAnimation(downRotateAnimation);
tvHeaderState.setText(“下拉刷新”);
break;
case PLAN_UPDATE:
ivHeaderArrow.startAnimation(upRotateAnimation);
tvHeaderState.setText(“准备刷新”);
break;
case PROCESS_UPDATE:
ivHeaderArrow.setVisibility(INVISIBLE);
ivHeaderArrow.clearAnimation();
pbHeader.setVisibility(VISIBLE);
tvHeaderState.setText(“正在刷新中…”);

if (null != customUpdateListViewBack) {
customUpdateListViewBack.downUpdateListData();
}
break;
default:
break;
}
}

private ICustomUpdateListViewBack customUpdateListViewBack;

public void setCallback(ICustomUpdateListViewBack back) {
this.customUpdateListViewBack = back;
}

public void updateHeaderResult() {
headerView.setPadding(0, -headerViewHeight, 0, 0);

// 状态还原
ivHeaderArrow.clearAnimation();
tvHeaderState.setText(“下拉刷新”);

ivHeaderArrow.setVisibility(VISIBLE);
pbHeader.setVisibility(INVISIBLE);

tvHeaderLastUpdateTime.setText(getThisTiem());

thisUpdateStatusValue = DOWN_UPDATE;
}

private String getThisTiem() {
SimpleDateFormat simpleDateFormat = new SimpleDateFormat(“yyyy年MM月dd日 HH:mm:ss”);// HH:mm:ss
// 获取当前时间
Date date = new Date(System.currentTimeMillis());
return simpleDateFormat.format(date);
}

public void updateBottomResult() {
/tvBottomState.setText(“加载成功”);
tvBottomState.setTextColor(Color.GREEN);
/

new android.os.Handler().postDelayed(new Runnable() {
@Override
public void run() {
bottomView.setPadding(0, -bottomViewHeight, 0, 0);
}
}, 2000);
}
}

三:把ProgressBar的风格修改,从白色演变成红色的形式
custom_progressbar.xml 文件:

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

<rotate xmlns:android=“http://schemas.android.com/apk/res/android”

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级安卓工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Android移动开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
img

学习分享

在当下这个信息共享的时代,很多资源都可以在网络上找到,只取决于你愿不愿意找或是找的方法对不对了

很多朋友不是没有资料,大多都是有几十上百个G,但是杂乱无章,不知道怎么看从哪看起,甚至是看后就忘

如果大家觉得自己在网上找的资料非常杂乱、不成体系的话,我也分享一套给大家,比较系统,我平常自己也会经常研读。

2021最新上万页的大厂面试真题

七大模块学习资料:如NDK模块开发、Android框架体系架构…

只有系统,有方向的学习,才能在段时间内迅速提高自己的技术。

这份体系学习笔记,适应人群:
**第一,**学习知识比较碎片化,没有合理的学习路线与进阶方向。
**第二,**开发几年,不知道如何进阶更进一步,比较迷茫。
**第三,**到了合适的年纪,后续不知道该如何发展,转型管理,还是加强技术研究。如果你有需要,我这里恰好有为什么,不来领取!说不定能改变你现在的状态呢!

由于文章内容比较多,篇幅不允许,部分未展示内容以截图方式展示 。如有需要获取完整的资料文档的朋友点击我的【GitHub】免费获取。

的技术。

这份体系学习笔记,适应人群:
**第一,**学习知识比较碎片化,没有合理的学习路线与进阶方向。
**第二,**开发几年,不知道如何进阶更进一步,比较迷茫。
**第三,**到了合适的年纪,后续不知道该如何发展,转型管理,还是加强技术研究。如果你有需要,我这里恰好有为什么,不来领取!说不定能改变你现在的状态呢!

由于文章内容比较多,篇幅不允许,部分未展示内容以截图方式展示 。如有需要获取完整的资料文档的朋友点击我的【GitHub】免费获取。

  • 4
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值