自定义AutoCompleteTextView行为

时间:2012-01-27 11:35:44

标签: android filtering android-arrayadapter autocompletetextview

开箱即用,AutoCompleteTextView窗口小部件似乎无法匹配列表值中间的输入字符串 - 匹配始终在开头进行;例如,输入“ar”匹配“argentina”,但不匹配“hungary”。

如何搜索单词中间的文字?谁能给我一个想法?

提前致谢!

4 个答案:

答案 0 :(得分:7)

您需要编写自定义Filter类并自行实施performFiltering方法。此方法采用CharSequence参数,您可以使用该参数执行所需的任何String操作,以便从数据集生成匹配列表(在您的情况下,您可以使用String.contains而不是{{ 1}})。 String.startsWith函数未在UI线程上运行。

然后将您的匹配列表作为FilterResults对象返回,其中包含performFiltering (您的匹配列表,可能是Object)和一个ArrayList 计数,这是您的匹配列表的大小。

最后,实现publishResults回调方法,该方法在工作线程生成匹配列表后返回,允许您在AutoCompleteTextView的适配器上调用int,以便它可以显示结果。< / p>

答案 1 :(得分:4)

老问题,但仍然相关。在一些其他问题的指导下实现了使用可过滤的自定义适配器。我做了一个简单的通用适配器,用contains搜索。关于它的快速说明:

我使用的是butterknife,但很容易用findviewbyid做viewHolder。

布局R.layout.list_item_simple是一个带有textview R.id.text_view_simple的简单布局。

该对象需要一个将要比较的toString。

public class SimpleContainsAutocompleteAdapter <T> extends ArrayAdapter<T> implements Filterable {
    private List <T> listObjects;
    List<T> suggestions = new ArrayList<>();
    private int resource;

    private Filter mFilter = new Filter(){
        @Override
        protected FilterResults performFiltering(CharSequence constraint) {
            FilterResults filterResults = new FilterResults();

            if(constraint != null) {
                suggestions.clear();
                for(T object : listObjects){
                    if(object.toString().toLowerCase().contains(constraint.toString().toLowerCase())){
                        suggestions.add(object);
                    }
                }

                filterResults.values = suggestions;
                filterResults.count = suggestions.size();
            }

            return filterResults;
        }

        @Override
        protected void publishResults(CharSequence contraint, FilterResults results) {
            if(results == null){
                return;
            }

            List<T> filteredList = (List<T>) results.values;
            if(results.count > 0) {
                clear();
                for (T filteredObject : filteredList) {
                    add(filteredObject);
                }
                notifyDataSetChanged();
            }
        }
    };

    public SimpleContainsAutocompleteAdapter(Context context, List<T> listObjects) {
        super(context, R.layout.list_item_simple, listObjects);
        this.listObjects = new ArrayList<>(listObjects);
        this.resource = R.layout.list_item_simple;
    }

    @Override
    public Filter getFilter() {
        return mFilter;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        Object listObject = getItem(position);
        viewHolder holder;
        if(convertView != null) {
            holder = (viewHolder) convertView.getTag();
        }else{
            convertView = LayoutInflater.from(getContext()).inflate(resource, parent, false);
            holder = new viewHolder(convertView);
            convertView.setTag(holder);
        }

        holder.name.setText(listObject.toString());

        return convertView;
    }


    static class viewHolder {
        @InjectView(R.id.text_view_simple) TextView name;

        public viewHolder(View view) {
            ButterKnife.inject(this, view);
        }
    }
}

答案 2 :(得分:2)

public class AutoCompleteAdapter <T> extends ArrayAdapter<T> implements Filterable {
    private List <T> listObjects;
    List<T> suggestions = new ArrayList<>();
    private Context context;
    public AutoCompleteAdapter(Context context, List<T> listObjects) {
        super(context, R.layout.list_item_simple, listObjects);
        this.listObjects = new ArrayList<>(listObjects);
        this.context = context;
    }
    private Filter mFilter = new Filter(){
        @Override
        protected FilterResults performFiltering(CharSequence constraint) {
            FilterResults filterResults = new FilterResults();
            if(constraint != null) {
            suggestions.clear();
                for(T object : listObjects){  
                    if(object.toString().toLowerCase().contains(constraint.toString().toLowerCase())){                        suggestions.add(object);
                }
            }

            filterResults.values = suggestions;
            filterResults.count = suggestions.size();
        }
        return filterResults;
    }

    @Override
    protected void publishResults(CharSequence constraint, FilterResults results) {
        if(results == null){
            return;
        }
        List<T> filteredList = (List<T>) results.values;
        if(results.count > 0) {
            clear();
            for (T filteredObject : filteredList) {
                add(filteredObject);
            }
            notifyDataSetChanged();
        }
    }
};
@Override
public Filter getFilter() {
    return mFilter;
}
private static class ViewHolder {
    TextView title;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
    Object listObject = getItem(position);
    final ViewHolder viewHolder; // view lookup cache stored in tag
    if (convertView == null) {
        viewHolder = new ViewHolder();
        LayoutInflater inflater = LayoutInflater.from(getContext());
        convertView = inflater.inflate(R.layout.list_item_simple, parent, false);
        viewHolder.title = (TextView) convertView.findViewById(R.id.title);
        convertView.setTag(viewHolder);
    } else {
        viewHolder = (ViewHolder) convertView.getTag();
    }
    viewHolder.title.setText(listObject.toString());
    return convertView;
}

}

答案 3 :(得分:0)

我的建议是将字符串解析为字符数组。 然后迭代每个字符,直到找到字符串。

例如,假设您的搜索想要返回所有带有“ate”的单词并且单词列表是...

状态 特征 谯呵 晚

你的算法应该是这样的

获取字符串并将其解析为字符数组 循环遍历数组并找到第一个“正确的字符”(在我们的例子中是'a') 找到该字符后,检查下一个字符,继续检查每个字符是否匹配,直到搜索到的值完成为止。如果字符不匹配,则退出数组迭代并转到下一个单词。