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