Android AutoCompleteTextView показывает информацию об объекте вместо текста в ландшафтном режиме

Я использую пользовательский адаптер с 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();
            }   
        }    

    }

}

5 ответов

Правильный способ сделать это - переопределить метод Filter#convertResultToString(Object) в вашем закрытом классе ArrayFilter.

Как сказано в документации Android

public CharSequence convertResultToString (Object resultValue)

Преобразует значение из отфильтрованного набора в последовательность CharSequence. Подклассы должны переопределить этот метод, чтобы преобразовать их результаты. Реализация по умолчанию возвращает пустую строку для значений NULL или строковое представление значения по умолчанию.

В вашем случае это должно быть что-то вроде этого:

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

Добавление этого метода позволит AutoCompleteTextView предоставлять подсказки исправления (в альбомной ориентации) вместо ссылок на объекты или по умолчанию toString

Да, просто переопределите метод toString() в вашем POJO.

public class MyObject {
    ...

    @Override
    public String toString() {
        return showThisStringOnAdapter;
    }
}

Создайте собственный класс, который расширяет <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>

Надеюсь, поможет. Ну, по крайней мере, это для меня.

Это ваш ответ:

adapter.setCursorToStringConverter(new CursorToStringConverter() {
    @Override
    public CharSequence convertToString(Cursor c) {
        return c.getString(c.getColumnIndexOrThrow("Column"));
    }
});

Я нашел ответ в прошлый раз. У меня была та же проблема, и кажется, что автозаполнение использует метод 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, и он работает. Это так просто. Конечно, вы будете заполнять этот метод в зависимости от того, что вы хотите, чтобы он отображался.

Другие вопросы по тегам