Android 结合实际项目学会ListView局部刷新和相关知识《一》
转载本专栏博客,请注明出处:道龙的博客
最近在公司参与的项目中有一个界面需要做局部UI更新处理,把其化烦为简为Demoi形式写在这里。我们还是运行该Demo,知道ListView局部刷新的使用场景:(受时间限制,本篇就介绍这么多功能,下一篇会在这个Demo中加入更多的功能)
可以看到,点击每个Item的时候,ListView上的CheckBox会选中或取消选中。这里面用到知识点就是ListView的局部刷新,局部的意思就是脱离ListView,在每个Item位置进行更新UI操作。那么就从头一步步的学习如何实现这个功能:
一、实现基本功能。
模拟数据源,把最基本的功能先跑起来:
布局、自定义样式、自定义选择器都很简单,过多的代码就不再写了,文章只贴出业务逻辑重要的代码(文章最后会给出源代码可自行下载;由于实习公司使用的IDE还是Eclipse,本篇是基于eclipse写的)
主活动中代码也很常规简单:
public class MainActivity extends Activity { private ListView mListView;
List<ItemBean> mDatas = new ArrayList<ItemBean>();
private MyAdapter adapter; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mListView = (ListView) findViewById(R.id.listview);
initData();
adapter = new MyAdapter();
mListView.setAdapter(adapter);
} /**模拟数据*/
private void initData() {
ItemBean mItemBean = null;
for (int i = 0; i < 40; i++) {
//集合添加20个对象
mItemBean = new ItemBean();
mItemBean.setText("学习ListView局部刷新"+i);
mItemBean.setResIcon(R.drawable.directory_icon);
mDatas.add(mItemBean);
}
} private class MyAdapter extends BaseAdapter{ @Override
public int getCount() {
if(mDatas != null){
return mDatas.size();
}
return 0;
} @Override
public Object getItem(int position) {
if(mDatas != null){
return mDatas.get(position);
}
return null;
} @Override
public long getItemId(int position) {
return position;
} @Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if(convertView == null){
holder = new ViewHolder();
convertView = View.inflate(getApplicationContext(), R.layout.file_item, null);
holder.ivIcon = (ImageView) convertView.findViewById(R.id.iv_icon);
holder.tvText = (TextView) convertView.findViewById(R.id.tv_name);
holder.ivOption = (ImageView) convertView.findViewById(R.id.iv_option);
convertView.setTag(holder);
}else{
holder = (ViewHolder) convertView.getTag();
} //拿到数据
ItemBean mItem = (ItemBean) getItem(position);
//设置数据
holder.ivIcon.setImageResource(mItem.getResIcon());
holder.tvText.setText(mItem.getText());
return convertView;
} class ViewHolder{
ImageView ivIcon;
TextView tvText;
ImageView ivOption;
} }
}
此时运行程序:
接下来就对上边的常规代码做一下修饰了:
二、进入编辑模式
我们长按item的时候,希望整个布局改变一下进入编辑模式,让复选框图标展示,对应位置上ImageView图标消失,就可以这么写。
首先写一个方法,来控制布局的改变:
private boolean isEdit;
/**开始编辑模式*/
private void startEditMode() {
isEdit = true;//标志位置为true
//进入编辑模式,要改变IListView的界面
adapter.notifyDataSetChanged();
} /**结束编辑模式*/
private void stopEditMode(){
isEdit = false;
}
然后加入长按item的点击事件:
mListView.setOnItemLongClickListener(new OnItemLongClickListener() { @Override
public boolean onItemLongClick(AdapterView<?> parent, View view,
int position, long id) {
startEditMode();
return true;//返回true后,mListView.setOnItemClickListener就不会再去调用
}
});
三、ListView中CheckBox选中取消,实现局部的刷新
定义一个item局部刷新的方法:
/**定义一个item局部刷新的方法*/
public void updateItemView(View ItemView){
CheckBox itemCb = (CheckBox) ItemView.findViewById(R.id.cb_checkbox);
if(itemCb.isChecked()){
//点击item,从选中状态调到未选中状态
itemCb.setChecked(false);
}else{
itemCb.setChecked(true);
}
}
在item的点击事件的时候,调用这个方法,即可完成checkbox的选中与取消:
mListView.setOnItemClickListener(new OnItemClickListener() { @Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
if(isEdit){
//如果是编辑模式
adapter.updateItemView(view);
}else{
//不是编辑模式,点击item做其他逻辑处理
Toast.makeText(getApplicationContext(), "当前item"+position, 0).show();
}
}
});
目前我们仅仅是实现了界面的刷新,但是cb的选中与取消要记录起来,点击了哪个item也要记录起来,这样才能做接下来的操作,选中cb执行什么样的任务,因此我们需要使用javabean,记录每次点击选中cb的操作:
具体的局部刷新修改后的方法如下:
/**定义一个item局部刷新的方法*///position,为了拿到当前item的对象,记录状态
public void updateItemView(View ItemView,int position){
position = position - mListView.getHeaderViewsCount();
ItemBean itemBean = mDatas.get(position);
CheckBox itemCb = (CheckBox) ItemView.findViewById(R.id.cb_checkbox);
if(itemCb.isChecked()){
//点击item,从选中状态调到未选中状态
itemCb.setChecked(false);
//记录当前item未选中状态
itemBean.setChecked(false);
}else{
itemCb.setChecked(true);
//记录当前item选中状态
itemBean.setChecked(true);
} //通过集合把选中的item的对象itemBean记录起来
List<ItemBean> selectDatas = new ArrayList<ItemBean>();
for (ItemBean info : mDatas) {
if(info.isChecked()){//当前的item是选中的
selectDatas.add(info);
}
} /**测试局部刷新*/
for (ItemBean itemBean2 : selectDatas) {
System.out.println(itemBean2.getText());
}
}
然后该方法肯定要被调用咯,在哪里调用?点击item的时候嘛,那就在注册item的点击事件稍作修改就可以了:
mListView.setOnItemClickListener(new OnItemClickListener() { @Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
if(isEdit){
//如果是编辑模式
adapter.updateItemView(view,position);
}else{
//不是编辑模式,点击item做其他逻辑处理
Toast.makeText(getApplicationContext(), "当前item"+position, 0).show();
}
}
});
但是这个时候还是有许多问题的,比如第二页的数据会复用第一页的数据,那就先解决这个bug。
只需要在getView里面,对item上边的cb重新复制就可以了。
if(isEdit){//是编辑模式
// Ui层面进入编辑模式
holder.checkBox.setVisibility(View.VISIBLE);
holder.ivOption.setVisibility(View.GONE); /**解决复用问题,对每一个item上的cb都重新赋值*/
if (mItem.isChecked()) {
holder.checkBox.setChecked(true);
} else {
holder.checkBox.setChecked(false);
} }else{
holder.checkBox.setVisibility(View.GONE);
holder.ivOption.setVisibility(View.VISIBLE);
}
四、根据是否编辑状态,修改顶部标题栏文本信息
接下来的逻辑稍微复杂,就是通过当前状态,来修改顶端的标题栏状态。
首先,长按进入编辑模式的时候,要显示当前选中多少个。需要一个标志位mSelectedCount。进入编辑模式,显示标题:
//修改标题栏
mTextView.setText(String.format("已选择了%d个", mSelectedCount));
那对于标志的增加减少要放到,updateItemView里面
if(itemCb.isChecked()){
//点击item,从选中状态调到未选中状态
itemCb.setChecked(false);
//记录当前item未选中状态
itemBean.setChecked(false); mSelectedCount --;
}else{
itemCb.setChecked(true);
//记录当前item选中状态
itemBean.setChecked(true); mSelectedCount ++;
}
由于逻辑稍微复杂,也不是要介绍局部刷新这个技术点的范围了,就不再继续往下写了。我把Demo全部代码,打包上传了。可以直接去下载,学习。看看一下后边的复杂逻辑是如何实现的:
明天或者后天加入更多功能后,再把资源上传吧,今天就到这里了~
Android 结合实际项目学会ListView局部刷新和相关知识《一》的更多相关文章
- Android listview局部刷新和模拟应用下载(zhu)
在android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用notifyDataSetChanged()这个函数,但是它会更新listview中所有可 ...
- 转:android listview局部刷新和模拟应用下载
在 android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用 notifyDataSetChanged()这个函数,但是它会更新listview中所 ...
- Android开源项目发现---ListView篇(持续更新)
资料转载地址:https://github.com/Trinea/android-open-project 1. android-pulltorefresh 一个强大的拉动刷新开源项目,支持各种控件下 ...
- 【转】Android开源项目发现---ListView篇(持续更新)
原文网址:http://blog.csdn.net/krislight/article/details/20211045 资料转载地址:https://github.com/Trinea/androi ...
- jquerymobile listview 局部刷新
function onSuccess(data, status) { data = $.trim(data); // alert(data); // return; if (data) { $('#l ...
- android:ListView的局部刷新
1.简介 对于android中的ListView刷新机制,大多数的程序员都是很熟悉的,修改或者添加adapter中的数据源之后,然后调用notifyDataSetChanged()刷新ListView ...
- ListView实现Item局部刷新
对于ListView数据的刷新大家都知道,改变Adapter的数据源,然后调用Adapter的notifyDateSetChanged()方法即可. 但是博主在做公司项目的时候,有个下载模块,因为可 ...
- Android 它们的定义ListView实现底部和页下拉刷新刷新的顶
在项目开发.由于数据量过大,寻呼需要加载或下拉刷新.为了缓解长期等待-time负载.这个博客的评论中被自己的定义实例ListView实现底部的下拉刷新页面正在加载结果和顶部. 其效果图: 一.List ...
- ListView的局部刷新
有的列表可能notifyDataSetChanged()代价有点高,最好能局部刷新. 局部刷新的重点是,找到要更新的那项的View,然后再根据业务逻辑更新数据即可. private void upda ...
随机推荐
- JavaScript 中常见的内存泄露陷阱(摘)
内存泄露是每个开发者最终都不得不面对的问题.即便使用自动内存管理的语言,你还是会碰到一些内存泄漏的情况.内存泄露会导致一系列问题,比如:运行缓慢,崩溃,高延迟,甚至一些与其他应用相关的问题. 什么是内 ...
- 1102mysql关于SOCK文件的认识
自己原创,只为记录. 当本地登录MySQL的时候提示"Can't connect to local MySQL server through socket"的问题,其实代码是MyS ...
- javascript实现有限状态机
1.状态机描述 简单说,有限状态机是一种模型,模型都用来模拟事物,能够被有限状态机这种模型模拟的事物,一般都有以下特点: 1)可以用状态来描述事物,并且任一时刻,事物总是处于一种状态: 2)事物拥有的 ...
- 自动化运维:使用flask+mysql+highcharts搭建监控平台
1.前言 本来想做一个比较完善的监控平台,只需要做少许改动就可以直接拿来用,但是在做的过程中发现要实现这个目标所需的工作量太大,而当前的工作中对其需求又不是特别明显.所以就退而求其次,做了一个类似教程 ...
- mysql乱码配置
1.进入mysql show variables like "char%" 2.在/etc/mysql/my.cnf中增加以下内容 [client] default-c ...
- javascript的基础(1)
1.javascript是什么? 它是一门基于客户端的脚本语言,是相对于服务器而言,浏览器就是一个客户端软件,浏览器从服务器上将资源(html,css,js,图片等)请求下来 并且在本地利用浏览器去解 ...
- LGTB 与大数
LGTB 有一个非常大的数,并且他想对它进行Q 次操作 每次操作是把这个大数中的某种数字全部替换成一个数字串 他想知道Q 次操作之后得到的数对1000000007(109 + 7) 取模的结果,请输出 ...
- HNOI2018 滚粗记
day0 说好了不复习,于是复习了一下配置,没想到一下就记住了,咋不退役去搞英语竞赛捏 皇室一波攒RP,chicken chicken一波攒RP day1 机子坏了,换到了最后面,但这个时候已经 \( ...
- 51 nod 1421 最大MOD值
1421 最大MOD值 题目来源: CodeForces 基准时间限制:1 秒 空间限制:131072 KB 分值: 80 难度:5级算法题 有一个a数组,里面有n个整数.现在要从中找到两个数字(可以 ...
- hdu 5538(水)
Input The first line contains an integer T indicating the total number of test cases. First line of ...