我需要覆盖来自类ArrayAdapter
的getFilter()
方法,并且我找到了来自here in the github的源代码
//package name
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import android.content.Context;
import android.util.Log;
import android.widget.ArrayAdapter;
import android.widget.Filter;
import android.widget.Filterable;
public class CustomAdapter<T> extends ArrayAdapter<T> implements Filterable{
private ArrayList<T> mOriginalValues;
private List<T> mObjects;
private CustomFilter mFilter;
private final Object mLock = new Object();
public CustomAdapter(Context context, int textViewResourceId, T[] objects) {
super(context, textViewResourceId, objects);
mObjects = Arrays.asList(objects);
// TODO Auto-generated constructor stub
}
@Override
public Filter getFilter() {
// TODO Auto-generated method stub
if (mFilter == null) {
mFilter = new CustomFilter();
}
return mFilter;
}
private class CustomFilter extends Filter {
@Override
protected FilterResults performFiltering(CharSequence prefix) {
FilterResults results = new FilterResults();
Log.d("bajji", "its ---> " + prefix);
if (mOriginalValues == null) {
synchronized (mLock) {
mOriginalValues = new ArrayList<T>(mObjects);
}
}
if (prefix == null || prefix.length() == 0) {
ArrayList<T> list;
synchronized (mLock) {
list = new ArrayList<T>(mOriginalValues);
}
results.values = list;
results.count = list.size();
} else {
String prefixString = prefix.toString().toLowerCase();
ArrayList<T> values;
synchronized (mLock) {
values = new ArrayList<T>(mOriginalValues);
}
final int count = values.size();
final ArrayList<T> newValues = new ArrayList<T>();
final ArrayList<T> approxValues = new ArrayList<T>();
final ArrayList<T> secondApproxValues = new ArrayList<T>();
for (int i = 0; i < count; i++) {
final T value = values.get(i);
final String valueText = value.toString().toLowerCase();
boolean flag = true;
// First match against the whole, non-splitted value
if (valueText.startsWith(prefixString)) {
newValues.add(value);
flag = false;
} else {
final String[] words = valueText.split(" ");
final int wordCount = words.length;
// Start at index 0, in case valueText starts with space(s)
for (int k = 0; k < wordCount; k++) {
if (words[k].startsWith(prefixString)) {
newValues.add(value);
flag = false;
break;
}
}
}
if(flag) {
if(approxMatch(valueText, prefixString) <= 3) { //change the stuff and do a levi work
approxValues.add(value);
}
else {
final String[] words = valueText.split(" ");
final int wordCount = words.length;
// Start at index 0, in case valueText starts with space(s)
for (int k = 0; k < wordCount; k++) {
if(approxMatch(words[k], prefixString) <= 3) {
//leve work
secondApproxValues.add(value);
break;
}
}
}
}
}
newValues.addAll(approxValues);
newValues.addAll(secondApproxValues);
results.values = newValues;
results.count = newValues.size();
}
return results;
}
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
//noinspection unchecked
mObjects = (List<T>) results.values;
if (results.count > 0) {
notifyDataSetChanged();
} else {
notifyDataSetInvalidated();
}
}
}
private int approxMatch (String s, String t) {
// an approxmimate string matching algo
return p;
}
}
问题是getFilter方法有一个私有内部类ArrayFilter
的对象,它有一个peformFiltering
方法,我需要在那里放一个不同的代码,所以我必须重写这个类。我在这个方法中得到了一个异常。
在扩展ArrayAdapter的派生类中,我创建了一个类似于ArrayFilter
的私有内部类,并将其命名为MyFilter
,并且在方法performFiltering
中再次得到相同的异常。
我找到一个解决方案来解决我的问题。我复制了ArrayAdapter
类中的所有代码,创建了一个名为MyAdapter
的新类,并修改了内部类ArrayFilter
中的一些代码,应用程序按我想要的方式运行。但我觉得这不是最好的解决方案。
Android有不同的api级别,所以如果在不同的api级别中更改数组适配器,那么我必须在我的代码中添加这些更改。所以我觉得最好的方法是扩展ArrayAdapter
类来创建MyAdapter
,而不是仅仅从ArrayAdapter
中复制和粘贴代码
如何重写父类的内部私有类?
编辑:我得到的异常..
Edit2:现在我在问题中添加了完整的代码。如果我复制和编辑数组适配器,它就能完美地工作。问题是只有当我扩展..!!现在,代码和搜索都运行得很完美。我和Log.i
确认过了..但UI中自动完成建议的下拉列表不起作用。我只得到我输入的第一个字符,下一个字符过滤会发生,但UI更新不会发生。
发布于 2012-05-08 12:30:52
在stackoverflow社区的一些帮助下,我删除了异常,后来我发现返回的建议并没有真正改变,因为mObjects
是从超类或基类返回的。问题是有两个公共方法getCount()
和getItem(int position)
,它们获取计数并从基类的mObjects中获取列表。所以我只需要在我的类中添加这两个方法。
public int getCount() {
return mObjects.size();
}
public T getItem(int position) {
return mObjects.get(position);
}
现在将返回派生类的mObjects
。它们是在UI的下拉列表中更新的。
我不是java专家,但这解决了我的问题。如果您有任何建议可以使代码更好,请将其添加到评论中或随时编辑答案。
发布于 2012-05-04 14:13:06
更新:
您应该覆盖T
类中的toString()
方法,并返回筛选条件字符串。
例如,
class T {
String firstName;
String lastName;
...
...
@override
String toString() {
return firstName + lastName;
}
}
缺省情况下,Object
类的toString()
实现返回getClass().getName() + '@' + Integer.toHexString(hashCode())
原件:
应将数组列表引用传递给ArrayAdapter
构造函数。我认为你没有这样做,因此它不能获得一个列表进行过滤。
在此示例代码中,objects
数组引用被传递给超级构造函数。
public CustomAdapter(Context context, int tvResId, ArrayList<String> objects) {
super(context, textViewResourceId, objects);
this.objects = objects;
}
如果您的过滤是基于列表项中的简单字符串,那么您可以按照@Iiorry的建议实现TextWatcher
。但是,如果过滤比这复杂一点,那么您还需要实现Filterable
。
https://stackoverflow.com/questions/10379261
复制相似问题