这里说明:本人第一次写博客,可能写的不算太好。可是这个相关类型的研究与拓展,是项目中比較难得的。所以开一篇博客来总结和思考。先让我们看看项目需求。
项目需求说明:
1、须要在点击EditText的时候显示Button
2、点击其它条目的EditText的时候,必须在当前条目中显示Button,其它条目中的Button必须隐藏
3、滚动ListView的时候不可以显示错乱信息和错乱显示
4、在滚动的时候。假设条目不显示。那么必须隐藏Button
Content:
注意:布局文件就不须要细致去看了,主要还是依据需求来写相应的代码。完毕功能就好。
/**我们就看这个适配器里面的内容,在listView显示和ListView适配的时候,适配器始终是一个重要的相关内容。我们把适配器提取出来,细致研究究竟是怎么做到适配的。*/
public class ProductManagerAdapter extends BaseAdapter {
private List<Products> products;//数据源
private Context context;//上下文
private Handler handler;//主线程载体
//private static int temp = -1;//控件静态id来保持单一性
private static ViewHolder tempHolder = null;//控件静态重用对象。用来保持单一性
public ProductManagerAdapter(List<Products> products, Context context,
String type, Handler handler) {//构造方法,传递外界数据以及相应的上下文信息
super();
this.products = products;
this.context = context;
this.handler = handler;
}
public void updataAdapter(List<Products> products) {//公开方法,用来动态改变显示条目
if (products != null) {
this.products = products;
notifyDataSetChanged();
}
}
@Override
public int getCount() {
return products == null ? 0 : products.size();
}
@Override
public Object getItem(int position) {
return products.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
final Products product = products.get(position);
ViewHolder holder = null;
if (convertView == null) {
holder = new ViewHolder();
convertView = View.inflate(context, R.layout.item_products_manager,
null);
holder.price_save = (TextView) convertView
.findViewById(R.id.price_save);
holder.product_chengben = (EditText) convertView
.findViewById(R.id.product_chengben);
holder.product_price = (EditText) convertView
.findViewById(R.id.product_price);
holder.product_chengben.setTag(position); //保存相应的条目位置,防止被重用
holder.product_price.setTag(position); //保存相应的条目位置。防止被重用
**savingPosition**(holder, holder.product_chengben, "chengben", position);//重要方法A~參见后面具体分析
**savingPosition**(holder, holder.product_price, "price", position);
convertView.setTag(holder);
holder.price_save.setVisibility(View.GONE);//条目创建的时候为了摆脱焦点控制机制,所以使用强制性质要求不可以显示button
} else {
holder = (ViewHolder) convertView.getTag();
loseFoucus(holder);//失去焦点
holder.product_price.setTag(position);
holder.product_chengben.setTag(position);
holder.price_save.setVisibility(View.GONE);//滑动过程中,让显示的条目消失
}
return convertView;
}
class ViewHolder {
/** 商品价格 */
EditText product_price;
/** 商品成本 */
EditText product_chengben;
/** 保存按钮 */
TextView price_save;
}
在这之前的代码是完整的实现listView显示的控制代码。里面分别的集中几种方法非常重要,必须单独拧出来谈。
下面是重要方法A:savingPosition,下面我们具体分析。
/** edittext公共操作
* @param position **/
private void savingPosition(ViewHolder holder, final EditText ev,
final String type, int position) {
ev.addTextChangedListener(new ThisWatcher(holder) {//为EditText做监听
@Override
public void afterTextChanged(Editable s, ViewHolder holder) {
int p = (Integer) ev.getTag();//获取相应的position
savaData(p, s.toString(), type, holder);//重要方法B~參见后面具体分析
}
});
ev.setOnFocusChangeListener(new ForcusChangeDate(holder));//重要监听~參见后面具体分析
}
重要方法B:在保存相应的EditText信息中。使得EditText的内容可以保持在本地缓存中,不被由于页面转换而被消失改变
private void savaData(int p, String s, String tag, ViewHolder holder) {
if (p > products.size()) {//控制焦点
if ("chengben".equals(tag)) {
holder.product_chengben.clearFocus();
} else if ("price".equals(tag)) {
holder.product_price.clearFocus();
}
return;
} else {
Products product = products.get(p);
if ("chengben".equals(tag)) {
product.setCostprice(s);
} else if ("price".equals(tag)) {
product.setSaleprice(s);
}
products.set(p, product);
}
}
private abstract class ThisWatcher implements TextWatcher {
ViewHolder holder;
public ThisWatcher(ViewHolder holder) {
super();
this.holder = holder;
}
@Override
public void afterTextChanged(Editable s) {
afterTextChanged(s, holder);
}
@Override
public void beforeTextChanged(CharSequence s, int start, int count,
int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before,
int count) {
}
public abstract void afterTextChanged(Editable s, ViewHolder holder);
}
重要监听:这里是最为困难的地方,在写这段代码的时候。遇到非常多困难。正是由于这些困难。我保留了一些遇到的问题代码。来具体解说。
private class ForcusChangeDate implements OnFocusChangeListener {
ViewHolder holder;
public ForcusChangeDate(ViewHolder holder) {
super();
this.holder = holder;
}
/**这里出现了比較多的问题,须要具体的去注意细节控制button的显示*/
@Override
public void onFocusChange(View v, boolean hasFocus) {
/**A:这是正确的方法,将holoder控制成静态方法,然后依据每一个条目的不同来进行*/
if(hasFocus)
{
if(tempHolder == null)
{
//temp = v.getId();
tempHolder = holder;
holder.price_save.setVisibility(View.VISIBLE);
}
else if(tempHolder == holder)
{
holder.price_save.setVisibility(View.VISIBLE);
}
else if(tempHolder != holder)
{
tempHolder.price_save.setVisibility(View.GONE);
tempHolder=holder;
holder.price_save.setVisibility(View.VISIBLE);
}
}
/**B:这是当中的一种思路。将每一个EditText的ID来控制下来。重用显示,可是由于有两个EditText存在,所以此方法不合适*/
// if (hasFocus) {
// if (temp == -1) {
// holder.price_save.setVisibility(View.VISIBLE);
// if (v.getId() == holder.product_chengben.getId()) {
// temp = holder.product_chengben.getId();
// }else if (v.getId() == holder.product_price.getId()) {
// temp = holder.product_price.getId();
// }
// }else if (temp == holder.product_chengben.getId()|| temp == holder.product_price.getId()) {
// holder.price_save.setVisibility(View.VISIBLE);
// }
//
//
//
// }
/**C: 这是刚刚開始的方法,将两种进行推断,可是在实践的过程中发现Android5.0下面的机制可以实现。
Android5.0以上。将focus机制产生了变化。
使得此监听器一直处于运转状态,故一直在true和false中进行无限推断循环
所以不适用
*/
/*if (hasFocus) {
holder.price_save.setVisibility(View.VISIBLE);
} else {
holder.price_save.setVisibility(View.GONE);
}*/
}
}
/**让EditText失去焦点*/
private void loseFoucus(ViewHolder holder) {
holder.product_chengben.clearFocus();
holder.product_price.clearFocus();
}
}