有时候我们会有这么一个需求,在Listview的某个Item上有个按钮,点击这个按钮之后呢,需要对其它的item做一些操作,就像下面这个:
采纳按钮点击之前:采纳按钮点击之后:
简单介绍一下这两张图的意思:
模拟了一个Listview,在每个Item上都有一个“采纳”按钮,当点击其中一个“采纳”之后,需要将其它“采纳”按钮隐藏,并且自身需要改为“已采纳”,如后面的图所示。
我们有一种简单的办法可以做到,就是更改数据的值,并且让Adapter重新刷新一遍,这是一种简单粗暴的方式,用户体验非常不好,会使整个页面都闪一下。
那么为了有没有什么更好的办法呢?有的,使用我们的观察者模式。
观察者模式在我们这里的基本实现方式是:当其中一个按钮被点击之后,通过回调更新其它控件即可。如何实现呢?以简单代码来说明:
package com.sahadev;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.os.Message;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ListView;
import com.lidroid.xutils.ViewUtils;
import com.lidroid.xutils.view.annotation.ViewInject;
import com.sahadev.renren.R;
public class MainActivity extends Activity {
@ViewInject(R.id.listview)
private ListView listview;
private BaseAdapter internalAdapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_zoom);
ViewUtils.inject(this);
internalAdapter = new InternalAdapter(this);
listview.setAdapter(internalAdapter);
}
/**
* 观察者模式的回调接口
*
* @author Sahadev
*
*/
public interface Callback {
public void update(Message msg);
}
private static class InternalAdapter extends BaseAdapter {
private int dataSize = 6;
private Map<View, Callback> maps;
private LayoutInflater inflate;
public InternalAdapter(Context context) {
super();
inflate = LayoutInflater.from(context);
maps = new HashMap<View, MainActivity2.Callback>();
}
@Override
public int getCount() {
return dataSize;
}
@Override
public Object getItem(int position) {
return null;
}
@Override
public long getItemId(int position) {
return 0;
}
@SuppressLint("InflateParams")
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = inflate.inflate(R.layout.listview_item_layout, null);
Button button = (Button) convertView.findViewById(R.id.button);
// 生成一个观察者
Callback callback = new Callback() {
@Override
public void update(Message msg) {
// 当其中一个按钮被点击时通过调用update方法来更新其它UI
// 先隐藏所有的采纳按钮
View view = (View) msg.obj;
if (msg.what == 0) {
view.setVisibility(View.INVISIBLE);
} else {
Button button = (Button) view;
button.setText("已采纳");
button.setEnabled(false);
}
}
};
// 将我们的需要更改的对象和观察者一并放入maps
maps.put(button, callback);
button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(final View v) {
// 遍历maps中所有的监听者
Set<Entry<View, Callback>> entrySet = maps.entrySet();
Iterator<Entry<View, Callback>> iterator = entrySet.iterator();
while (iterator.hasNext()) {
Entry<View, Callback> next = iterator.next();
// 使用value的update方法将key作为参数使得监听的地方对它进行修改
Message msg = new Message();
View temp = next.getKey();
msg.obj = temp;
if (temp.equals(v) || temp == v) {// 如果是点击的按钮本身,则需要对它进行特殊处理
msg.what = 1;
}
// 通知所有的观察者
next.getValue().update(msg);
}
}
});
}
return convertView;
}
}
}