开箱即用,AutoCompleteTextView
窗口小部件似乎无法匹配列表值中间的输入字符串 - 匹配始终在开头进行;例如,输入“ar
”匹配“argentina
”,但不匹配“hungary
”。
如何搜索单词中间的文字?谁能给我一个想法?
提前致谢!
答案 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') 找到该字符后,检查下一个字符,继续检查每个字符是否匹配,直到搜索到的值完成为止。如果字符不匹配,则退出数组迭代并转到下一个单词。