12

我正在使用带有 AutoCompleteTextView 的自定义适配器。它在模拟器和我的平板电脑上运行良好。但是,我的手机在横向模式下存在问题。在此模式下显示的自动完成提示是对象信息而不是文本。但是,当我选择任何项目时,字段会在相应字段中正确填充文本。

基于 Android Stock Array Adapter 的其他字段的自动完成工作正常。

我必须在我的自定义适配器中为此做些什么吗?我在 SO 上只看到一个类似的问题。对该问题的一个回答是谈论覆盖 toString 方法,但我无法充分理解它以在我的代码中实现。

任何指导将不胜感激?如果您需要更多信息,请告诉我。

编辑:添加了我的自定义适配器源代码....

  public class Part_Mstr_Info
  {
    private long part_id;
    private String name, desg, org, dept;

    public Part_Mstr_Info(long part_id, String name, String desg, String org, String dept)
    {
        this.part_id = part_id;
        this.name = name;
        this.desg = desg;
        this.org = org;
        this.dept = dept;
    }
    public long get_part_id() { return part_id; }
    public String get_name() { return name; }
    public String get_desg() { return desg; }
    public String get_org() { return org; }
    public String get_dept() { return dept; }
}

public class CustomAdapter extends ArrayAdapter<Part_Mstr_Info> implements Filterable{
   private List<Part_Mstr_Info> entries;
   private ArrayList<Part_Mstr_Info> orig;
   private Activity activity;
   private ArrayFilter myFilter;

   public CustomAdapter(Activity a, int textViewResourceId, ArrayList<Part_Mstr_Info> entries) {
    super(a, textViewResourceId, entries);
    this.entries = entries;
    this.activity = a;
}

public static class ViewHolder{
    public TextView tv_ac_name;
    public TextView tv_ac_desg;
    public TextView tv_ac_org;
    public TextView tv_ac_dept;
}

@Override
public int getCount(){
      return entries!=null ? entries.size() : 0;
}

@Override
public Part_Mstr_Info getItem(int index) {
    return entries.get(index);
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    View v = convertView;
    ViewHolder holder;
    if (v == null) {
        LayoutInflater vi =
            (LayoutInflater)activity.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        v = vi.inflate(R.layout.ac_name_list, null);
        holder = new ViewHolder();
        holder.tv_ac_name = (TextView) v.findViewById(R.id.ac_name);
        holder.tv_ac_desg = (TextView) v.findViewById(R.id.ac_desg);
        holder.tv_ac_org = (TextView) v.findViewById(R.id.ac_org);
        holder.tv_ac_dept = (TextView) v.findViewById(R.id.ac_dept);
        v.setTag(holder);
    }
    else
        holder=(ViewHolder)v.getTag();

    final Part_Mstr_Info custom = entries.get(position);
    if (custom != null) {
        holder.tv_ac_name.setText(custom.get_name());
        holder.tv_ac_desg.setText(custom.get_desg());
        holder.tv_ac_org.setText(custom.get_org());
        holder.tv_ac_dept.setText(custom.get_dept());
    }
    return v;
}

@Override
public Filter getFilter() {
    if (myFilter == null){
        myFilter = new ArrayFilter();
    }
    return myFilter;
}

@Override
public String toString() {
    String temp = getClass().getName();
    return temp;
}

private class ArrayFilter extends Filter {
    @Override
    protected FilterResults performFiltering(CharSequence constraint) {
        FilterResults results = new FilterResults();
        if (orig == null)
            orig = new ArrayList<Part_Mstr_Info>(entries);
        if (constraint != null && constraint.length() != 0) {
            ArrayList<Part_Mstr_Info> resultsSuggestions = new ArrayList<Part_Mstr_Info>();
            for (int i = 0; i < orig.size(); i++) {
                if(orig.get(i).get_name().toLowerCase().startsWith(constraint.toString().toLowerCase())){
                    resultsSuggestions.add(orig.get(i));
                }
            }

            results.values = resultsSuggestions;
            results.count = resultsSuggestions.size();

        }
        else {
            ArrayList <Part_Mstr_Info> list = new ArrayList <Part_Mstr_Info>(orig);
            results.values = list;
            results.count = list.size();
        }
        return results;
    }

    @Override
    @SuppressWarnings("unchecked")
    protected void publishResults(CharSequence constraint, FilterResults results) {
        clear();
        ArrayList<Part_Mstr_Info> newValues = (ArrayList<Part_Mstr_Info>) results.values;
        if(newValues !=null) {
            for (int i = 0; i < newValues.size(); i++) {
                add(newValues.get(i));
            }
            if(results.count>0) {
                notifyDataSetChanged();
            } else {
                notifyDataSetInvalidated();
            }   
        }    

    }

}
4

5 回答 5

10

正确的方法是覆盖ArrayFilter 私有类中的方法Filter#convertResultToString(Object) 。

如Android文档中所述

公共 CharSequence convertResultToString(对象结果值)

将过滤集中的值转换为 CharSequence。子类应重写此方法以转换其结果。默认实现为空值返回一个空字符串或该值的默认字符串表示。

在你的情况下,它应该是这样的:

private class ArrayFilter extends Filter {
    @Override
    public CharSequence convertResultToString(Object resultValue) {
        return ((Part_Mstr_Info) resultValue).get_name();
    }

添加此方法将允许 AutoCompleteTextView 提供更正提示(在横向视图中)而不是对象引用或默认 toString

于 2015-05-06T11:38:18.927 回答
9

是的,只需覆盖 POJO 上的 toString() 方法。

public class MyObject {
    ...

    @Override
    public String toString() {
        return showThisStringOnAdapter;
    }
}
于 2012-10-02T01:14:36.490 回答
3

创建扩展的自定义类<AutoCompleteTextView />。例如:

package lt.irisas.akcija.utils;

import lt.irisas.akcija.data.ItemsModel;
import android.content.Context;
import android.database.Cursor;
import android.util.AttributeSet;
import android.widget.AutoCompleteTextView;

/** Customizing AutoCompleteTextView to return Cursor column info as String
*/
public class CustomAutoCompleteTextView extends AutoCompleteTextView {

    public CustomAutoCompleteTextView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }
    /* Overriding this method and returning String type solves the problem */
    @Override
    protected CharSequence convertSelectionToString(Object selectedItem) {
        Cursor c = (Cursor) selectedItem;

        return c.getString(c.getColumnIndex("COLUMN_NAME"));
    }
}

但是,<AutoCompleteTextView/>您必须使用自定义的而不是。例如:

    <LinearLayout
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal" >

        <lt.irisas.akcija.utils.CustomAutoCompleteTextView
            android:id="@+id/search_field"
            android:layout_width="0dip"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:singleLine="true"
            android:imeOptions="actionSearch" >
        </lt.irisas.akcija.utils.CustomAutoCompleteTextView>
</LinearLayout>

希望能帮助到你。至少它对我有用。

于 2012-10-17T12:54:12.140 回答
0

这是你的答案:

adapter.setCursorToStringConverter(new CursorToStringConverter() {
    @Override
    public CharSequence convertToString(Cursor c) {
        return c.getString(c.getColumnIndexOrThrow("Column"));
    }
});
于 2014-01-14T19:33:49.260 回答
0

我上次找到了答案。我有同样的问题,似乎自动完成在横向模式下使用 convertToString() 方法而不是 newView() 和 bindView() 方法。您必须将所有光标数据返回到一个没有“\n”的字符串中,就像项目将在软键盘上水平显示一样。

在您的 AutoCompleteTextViewAdapter 示例中,添加方法:

        @Override
    public CharSequence convertToString(Cursor cursor) {


        String name = cursor.getString(0);
        String number = cursor.getString(1);


        return name+" : "+number;
    }

我用我的手机 Xperia U 和另一个 Galaxy Note 对其进行了测试,它可以工作。就是这么简单。当然,您将根据您希望它显示的内容填充此方法。

于 2013-01-25T10:10:50.243 回答