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

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

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

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

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip204888 (备注Android)
img

正文

/**

  • 定义底部相关
    /
    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"?>



四:如何使用自定义的ListView:

<heima.custom.CustomUpdateListView
android:id=“@+id/custom_listview”
android:layout_width=“match_parent”
android:layout_height=“match_parent”
android:layout_below=“@id/ll”>
</heima.custom.CustomUpdateListView>

当把数据查询出来后,执行:

listView.updateHeaderResult(); // 更新头部布局复原
listView.updateBottomResult(); // 更新底部布局复原

listView.setCallback(new ICustomUpdateListViewBack() {
@Override
public void downUpdateListData() {
new AsyncTask<Void, Void, Void>() {

@Override
protected Void doInBackground(Void… voids) {
SystemClock.sleep(3000);
mData.add(0, “1最新加载出来的数据”);
mData.add(1, “2最新加载出来的数据”);
mData.add(2, “3最新加载出来的数据”);
mData.add(3, “4最新加载出来的数据”);
mData.add(4, “5最新加载出来的数据”);
mData.add(5, “6最新加载出来的数据”);
return null;
}

@Override
protected void onPostExecute(Void aVoid) {
// super.onPostExecute(aVoid);
adapter.notifyDataSetChanged();
listView.updateHeaderResult();
}
}.execute(new Void[]{});
}

总结

【Android 详细知识点思维脑图(技能树)】

我个人是做Android开发,已经有十来年了,目前在某创业公司任职CTO兼系统架构师。虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。

这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。

由于篇幅有限,这里以图片的形式给大家展示一小部分。

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

最后,赠与大家一句话,共勉!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注Android)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

最后,赠与大家一句话,共勉!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注Android)
[外链图片转存中…(img-kecXh5E6-1713610071517)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值