ListView 中的自定义过滤 ArrayAdapter
Posted
技术标签:
【中文标题】ListView 中的自定义过滤 ArrayAdapter【英文标题】:Custom Filtering ArrayAdapter in ListView 【发布时间】:2011-09-23 10:44:59 【问题描述】:我是 android 的初学者,但我尝试进行自定义列表视图过滤,但它以某种方式工作。我唯一的问题是我保留所有值的 ArrayList (“原始” ArrayList )在每次过滤中的项目上越来越低。我无法解释这一点,但我认为您可以以某种方式帮助我。
无论如何这里是自定义 ArrayAdaptor:
public class PkmnAdapter extends ArrayAdapter<Pkmn>
private ArrayList<Pkmn> original;
private ArrayList<Pkmn> fitems;
private Filter filter;
public PkmnAdapter(Context context, int textViewResourceId, ArrayList<Pkmn> items)
super(context, textViewResourceId, items);
this.original = items;//new ArrayList<Pkmn>();
this.fitems = items;//new ArrayList<Pkmn>();
@Override
public void add(Pkmn item)
original.add(item);
@Override
public View getView(int position, View convertView, ViewGroup parent)
View v = convertView;
if (v == null)
LayoutInflater vi = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
v = vi.inflate(R.layout.row, null);
Pkmn pkmn = original.get(position);
if (pkmn != null)
TextView tt = (TextView) v.findViewById(R.id.RlabPName);
TextView dex = (TextView)v.findViewById(R.id.RlabDex);
ImageView img = (ImageView)v.findViewById(R.id.RimgPkmn);
if (tt != null) tt.setText(pkmn.getName());
if (dex != null) dex.setText(CalcDex(pkmn.getId()));
if (img != null)
int resId = getContext().getResources().getIdentifier("dex" + pkmn.getId(), "drawable", "com.compileguy.pokebwteam");
img.setImageResource(resId);
return v;
@Override
public Filter getFilter()
if (filter == null)
filter = new PkmnNameFilter();
return filter;
private class PkmnNameFilter extends Filter
@Override
protected FilterResults performFiltering(CharSequence constraint)
FilterResults results = new FilterResults();
String prefix = constraint.toString().toLowerCase();
if (prefix == null || prefix.length() == 0)
ArrayList<Pkmn> list = new ArrayList<Pkmn>(original);
results.values = list;
results.count = list.size();
else
final ArrayList<Pkmn> list = original;
int count = list.size();
final ArrayList<Pkmn> nlist = new ArrayList<Pkmn>(count);
for (int i=0; i<count; i++)
final Pkmn pkmn = list.get(i);
final String value = pkmn.getName().toLowerCase();
if (value.startsWith(prefix))
nlist.add(pkmn);
results.values = nlist;
results.count = nlist.size();
return results;
@SuppressWarnings("unchecked")
@Override
protected void publishResults(CharSequence constraint, FilterResults results)
fitems = (ArrayList<Pkmn>)results.values;
clear();
int count = fitems.size();
for (int i=0; i<count; i++)
Pkmn pkmn = (Pkmn)fitems.get(i);
add(pkmn);
if (fitems.size() > 0)
notifyDataSetChanged();
else
notifyDataSetInvalidated();
private String CalcDex(int id)
String s = String.valueOf(id);
if (s.length() == 1)
s = "00"+s;
else if (s.length() == 2)
s = "0"+s;
return '#'+s;
注意:列表视图正确显示了项目,但是当我删除编辑框中的一个字母(触发过滤)时,这就是问题的开始。
--- 编辑 ---
@Janusz:非常感谢您的回答。这解决了我的问题。
这是对我有用的源代码,所以如果有人有同样的问题,他们可以试试这个:
private ArrayList<Pkmn> original;
private ArrayList<Pkmn> fitems;
private Filter filter;
public PkmnAdapter(Context context, int textViewResourceId, ArrayList<Pkmn> items)
super(context, textViewResourceId, items);
this.original = new ArrayList<Pkmn>(items);
this.fitems = new ArrayList<Pkmn>(items);
@Override
public View getView(int position, View convertView, ViewGroup parent)
View v = convertView;
if (v == null)
LayoutInflater vi = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
v = vi.inflate(R.layout.row, null);
Pkmn pkmn = fitems.get(position);
if (pkmn != null)
TextView tt = (TextView) v.findViewById(R.id.RlabPName);
TextView dex = (TextView)v.findViewById(R.id.RlabDex);
ImageView img = (ImageView)v.findViewById(R.id.RimgPkmn);
if (tt != null) tt.setText(pkmn.getName());
if (dex != null) dex.setText(CalcDex(pkmn.getId()));
if (img != null)
int resId = getContext().getResources().getIdentifier("dex" + pkmn.getId(), "drawable", "com.compileguy.pokebwteam");
img.setImageResource(resId);
return v;
@Override
public Filter getFilter()
if (filter == null)
filter = new PkmnNameFilter();
return filter;
private class PkmnNameFilter extends Filter
@Override
protected FilterResults performFiltering(CharSequence constraint)
FilterResults results = new FilterResults();
String prefix = constraint.toString().toLowerCase();
if (prefix == null || prefix.length() == 0)
ArrayList<Pkmn> list = new ArrayList<Pkmn>(original);
results.values = list;
results.count = list.size();
else
final ArrayList<Pkmn> list = new ArrayList<Pkmn>(original);
final ArrayList<Pkmn> nlist = new ArrayList<Pkmn>();
int count = list.size();
for (int i=0; i<count; i++)
final Pkmn pkmn = list.get(i);
final String value = pkmn.getName().toLowerCase();
if (value.startsWith(prefix))
nlist.add(pkmn);
results.values = nlist;
results.count = nlist.size();
return results;
@SuppressWarnings("unchecked")
@Override
protected void publishResults(CharSequence constraint, FilterResults results)
fitems = (ArrayList<Pkmn>)results.values;
clear();
int count = fitems.size();
for (int i=0; i<count; i++)
Pkmn pkmn = (Pkmn)fitems.get(i);
add(pkmn);
【问题讨论】:
'clear()' 和 'add()' 函数的作用是什么。 @compileGuy,“Pkmn”是什么意思?我无法运行此示例。你能完整上传代码吗? 优秀的实现@compileguy,添加和清除是适配器的功能,你不需要覆盖它们。 pkmn 是一个模型类,用于保存列表中的数据值 您的代码与我的 (***.com/questions/20524417/…) 有何匹配?我还是有问题:/ 如何在列表视图中设置适配器 @kkanellis 【参考方案1】:你的问题是这几行:
this.original = items;
this.fitems = items;
Items 是您用于 ListView 的列表,将它放在两个不同的变量中不会产生两个不同的列表。您只给列表项两个不同的名称。
你可以使用:
this.fitems = new ArrayList(items);
这应该生成一个新的列表,并且对该列表的更改只会更改 fitems 列表。
【讨论】:
不适合我,我的 ListView 消失了,当我删除整个字符串时它不会再回来了。 ***.com/questions/20524417/… @SiKni8 我也得到了同样的结果。请帮忙【参考方案2】:您可以通过在 Pkmn 类上创建一个 toString()
方法来实现相同的效果,该方法返回您要过滤的值。
【讨论】:
【参考方案3】:我发现过滤 ArrayAdapter 的最佳方法是创建自己的过滤器类:
private class MyFilter extends Filter
然后在该函数中创建新的对象数组以在过滤器之后显示(您可以在class ArrayAdapter
的源代码中找到很好的实现)
@Override
protected FilterResults performFiltering(CharSequence prefix)
现在诀窍就在这个方法中
@Override
protected void publishResults(CharSequence constraint, FilterResults results)
当您使用阵列适配器时,您不能这样做:
myAdapterData = results.values
从那时起您将数据与超级数据断开连接,您必须这样做以保持对超级原始数据数组的引用:
data.clear();
data.addAll((List<YourType>) results.values);
然后覆盖
getFilter()
在您的适配器中,例如:
@Override
public Filter getFilter()
if (filter == null)
filter = new MyFilter();
return filter;
【讨论】:
【参考方案4】:他们在这里介绍了它: https://www.mysamplecode.com/2012/07/android-listview-custom-layout-filter.html
您需要在 arrrat 适配器上调用 clear() 来避免在每个元素上都遇到 indexBoundException 和 add(),因为 addAll() 仅在 honeycomb 之后才有效。
@SuppressWarnings("未选中") @Override protected void publishResults(CharSequence 约束, FilterResults 结果)
countryList = (ArrayList<Country>)results.values;
notifyDataSetChanged();
clear();
for(int i = 0, l = countryList.size(); i < l; i++)
add(countryList.get(i));
notifyDataSetInvalidated();
【讨论】:
以上是关于ListView 中的自定义过滤 ArrayAdapter的主要内容,如果未能解决你的问题,请参考以下文章
ListView 中自定义 ArrayAdapter 的自定义过滤
带有过滤器 Android 的自定义 Listview 适配器
未调用 DrawerLayout 中的自定义 BaseAdapter ListView onItemClickListener