Расширение ArrayAdapter в Android
Мне нужно переопределить метод getFilter()
из класса ArrayAdapter
, и я нашел исходный код здесь, в github
//package name
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import android.content.Context;
import android.util.Log;
import android.widget.ArrayAdapter;
import android.widget.Filter;
import android.widget.Filterable;
public class CustomAdapter<T> extends ArrayAdapter<T> implements Filterable{
private ArrayList<T> mOriginalValues;
private List<T> mObjects;
private CustomFilter mFilter;
private final Object mLock = new Object();
public CustomAdapter(Context context, int textViewResourceId, T[] objects) {
super(context, textViewResourceId, objects);
mObjects = Arrays.asList(objects);
// TODO Auto-generated constructor stub
}
@Override
public Filter getFilter() {
// TODO Auto-generated method stub
if (mFilter == null) {
mFilter = new CustomFilter();
}
return mFilter;
}
private class CustomFilter extends Filter {
@Override
protected FilterResults performFiltering(CharSequence prefix) {
FilterResults results = new FilterResults();
Log.d("bajji", "its ---> " + prefix);
if (mOriginalValues == null) {
synchronized (mLock) {
mOriginalValues = new ArrayList<T>(mObjects);
}
}
if (prefix == null || prefix.length() == 0) {
ArrayList<T> list;
synchronized (mLock) {
list = new ArrayList<T>(mOriginalValues);
}
results.values = list;
results.count = list.size();
} else {
String prefixString = prefix.toString().toLowerCase();
ArrayList<T> values;
synchronized (mLock) {
values = new ArrayList<T>(mOriginalValues);
}
final int count = values.size();
final ArrayList<T> newValues = new ArrayList<T>();
final ArrayList<T> approxValues = new ArrayList<T>();
final ArrayList<T> secondApproxValues = new ArrayList<T>();
for (int i = 0; i < count; i++) {
final T value = values.get(i);
final String valueText = value.toString().toLowerCase();
boolean flag = true;
// First match against the whole, non-splitted value
if (valueText.startsWith(prefixString)) {
newValues.add(value);
flag = false;
} else {
final String[] words = valueText.split(" ");
final int wordCount = words.length;
// Start at index 0, in case valueText starts with space(s)
for (int k = 0; k < wordCount; k++) {
if (words[k].startsWith(prefixString)) {
newValues.add(value);
flag = false;
break;
}
}
}
if(flag) {
if(approxMatch(valueText, prefixString) <= 3) { //change the stuff and do a levi work
approxValues.add(value);
}
else {
final String[] words = valueText.split(" ");
final int wordCount = words.length;
// Start at index 0, in case valueText starts with space(s)
for (int k = 0; k < wordCount; k++) {
if(approxMatch(words[k], prefixString) <= 3) {
//leve work
secondApproxValues.add(value);
break;
}
}
}
}
}
newValues.addAll(approxValues);
newValues.addAll(secondApproxValues);
results.values = newValues;
results.count = newValues.size();
}
return results;
}
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
//noinspection unchecked
mObjects = (List<T>) results.values;
if (results.count > 0) {
notifyDataSetChanged();
} else {
notifyDataSetInvalidated();
}
}
}
private int approxMatch (String s, String t) {
// an approxmimate string matching algo
return p;
}
}
Проблема заключается в том, что метод getFilter имеет объект частного внутреннего класса ArrayFilter
, который имеет метод peformFiltering
, и мне нужно поместить туда другой код, поэтому мне нужно переопределить класс. И я получаю исключение в методе.
В производном классе, который расширяет ArrayAdapter, я создал частный внутренний класс, который похож на ArrayFilter
и называет его MyFilter
, и я снова получаю то же исключение в методе performFiltering
.
Я нашел решение для решения моей проблемы. Я скопировал весь код в классе ArrayAdapter
и создал новый класс под названием MyAdapter
, и я изменил код внутри внутреннего класса ArrayFilter
, и приложение работает так, как я этого хотел. Но я чувствую, что это не лучшее решение.
Android имеет различные уровни api, поэтому, если адаптер массива изменен на другом уровне api, тогда мне нужно добавить эти изменения в мои коды. Поэтому я считаю, что лучший способ - расширить класс ArrayAdapter
, чтобы создать MyAdapter
, а не просто копировать и вставлять код из ArrayAdapter
Как я могу переопределить внутренний частный класс родительского класса.?
Изменить: исключение я получаю..
![Exception]()
Edit2: Теперь я добавил полный код в вопрос. и он отлично работает, если я копирую и редактирую адаптер массива. Проблема только в том, что я расширяю..!! теперь код и поиск работают отлично. Я проверил его с помощью Log.i
.., но выпадающий список для автоматического завершения предложения в пользовательском интерфейсе не работает. Я только получаю для первого символа, который я набираю, следующая фильтрация символов происходит, но обновление пользовательского интерфейса не происходит.
Ответы
Ответ 1
После некоторой помощи сообщества stackoverflow я удалил исключение, и позже я узнал, что возвращаемые предложения не меняются, потому что mObjects
был возвращен из суперкласса или базового класса. Проблема заключалась в том, что были два общедоступных метода getCount()
и getItem(int position)
, которые получают подсчет и выбирают список из mObjects из базового класса. Поэтому я просто должен добавить эти два метода в свой класс.
public int getCount() {
return mObjects.size();
}
public T getItem(int position) {
return mObjects.get(position);
}
Теперь возвращается mObjects
производного класса. Какие из них обновлены в раскрывающемся списке в пользовательском интерфейсе.
Я не эксперт по Java, но это решает мою проблему. Если у вас есть предложения по улучшению кода, добавьте его в комментарии или не стесняйтесь редактировать ответ.
Ответ 2
Обновление:
Вы должны переопределить метод toString()
в своем классе T
и вернуть строку критерия фильтра.
Например,
class T {
String firstName;
String lastName;
...
...
@override
String toString() {
return firstName + lastName;
}
}
По умолчанию реализация toString()
класса Object
возвращает getClass().getName() + '@' + Integer.toHexString(hashCode())
Оригинал:
Ссылка на список массива должна быть передана конструктору ArrayAdapter
. Я думаю, вы не делаете этого и, следовательно, не можете получить список для фильтрации.
В этом примере кода objects
ссылка массива передается супер-конструктору.
public CustomAdapter(Context context, int tvResId, ArrayList<String> objects) {
super(context, textViewResourceId, objects);
this.objects = objects;
}
Если ваша фильтрация основана на простых строках в элементах списка, вы можете реализовать TextWatcher
, как было предложено @Iiorry. Но если фильтрация немного сложна, тогда вам необходимо реализовать Filterable
.
Ответ 3
Вам не нужно реализовывать свой собственный фильтр... просто переопределите getFilter как предлагаемый @Waqas.
Вот что я сделал и это работает.
Не стесняйтесь приспосабливать его к вашим потребностям...
Надеюсь, что это поможет.
EditText (searchFilder):
searchFilter.addTextChangedListener(mOnSearchBoxTextChanged);
private TextWatcher mOnSearchBoxTextChanged = new TextWatcher() {
public void afterTextChanged(Editable s) {
}
public void beforeTextChanged(CharSequence s, int start, int count,
int after) {
}
public void onTextChanged(CharSequence s, int start, int before,
int count) {
mAdapter.getFilter().filter(s.toString());
}
};
Адаптер getFilter():
class MyAdapter extends BaseAdapter implements Filterable {
//the rest of the adapter....
@Override
public Filter getFilter() {
if (newFilter == null) {
newFilter = new Filter() {
@Override
protected void publishResults(CharSequence prefix,
FilterResults results) {
CLog.logD( "Friends list filtered");
mAdapter.notifyDataSetChanged();
}
@Override
protected FilterResults performFiltering(CharSequence constraint) {
constraint = constraint.toString().toLowerCase();
filteredFriendList = new ArrayList<FriendsListItemView>();
if (constraint!= null && constraint.toString().length() > 0) {
for (int i = 0; i < friendsList.size(); i++) {
FriendsListItemView newFriend = friendsList.get(i);
String name = newFriend.userName.getText().toString();
if (name.toLowerCase().contains(constraint)) {
filteredFriendList.add(newFriend);
}
}
} else {
if (filteredFriendList.size() == 0) {
filteredFriendList = friendsList;
}
}
FilterResults newFilterResults = new FilterResults();
newFilterResults.count = filteredFriendList.size();
newFilterResults.values = filteredFriendList;
return newFilterResults;
}
};
}
return newFilter;
}
}
Ответ 4
Возможно, это может вам помочь:
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import android.content.Context;
import android.util.Log;
import android.widget.ArrayAdapter;
import android.widget.Filter;
public class CustomAdapter<T> extends ArrayAdapter<T> {
private final ArrayList<T> mOriginalValues;
private List<T> mObjects;
private CustomFilter mFilter;
private final Object mLock = new Object();
public CustomAdapter(Context context, int textViewResourceId, T[] objects) {
super(context, textViewResourceId, objects);
mObjects = Arrays.asList(objects);
mOriginalValues = (ArrayList<T>) Arrays.asList(objects);
// TODO Auto-generated constructor stub
}
@Override
public Filter getFilter() {
// TODO Auto-generated method stub
if (mFilter == null) {
mFilter = new CustomFilter();
}
return mFilter;
}
private class CustomFilter extends Filter {
@Override
protected FilterResults performFiltering(CharSequence prefix) {
FilterResults results = new FilterResults();
Log.d("bajji", "its ---> " + prefix);
if (prefix == null || prefix.toString().trim().length() == 0) {
ArrayList<T> list;
synchronized (mLock) {
list = new ArrayList<T>(mOriginalValues);
}
results.values = list;
results.count = list.size();
} else {
String prefixString = prefix.toString().toLowerCase();
ArrayList<T> values;
synchronized (mLock) {
values = new ArrayList<T>(mOriginalValues);
}
final int count = values.size();
final ArrayList<T> newValues = new ArrayList<T>();
final ArrayList<T> approxValues = new ArrayList<T>();
final ArrayList<T> secondApproxValues = new ArrayList<T>();
for (int i = 0; i < count; i++) {
final T value = values.get(i);
final String valueText = value.toString().toLowerCase();
boolean flag = true;
// First match against the whole, non-splitted value
if (valueText.startsWith(prefixString)) {
newValues.add(value);
flag = false;
} else {
final String[] words = valueText.split(" ");
final int wordCount = words.length;
// Start at index 0, in case valueText starts with space(s)
for (int k = 0; k < wordCount; k++) {
if (words[k].startsWith(prefixString)) {
newValues.add(value);
flag = false;
break;
}
}
}
if(flag) {
if(approxMatch(valueText, prefixString) <= 3) { //change the stuff and do a levi work
approxValues.add(value);
}
else {
final String[] words = valueText.split(" ");
final int wordCount = words.length;
// Start at index 0, in case valueText starts with space(s)
for (int k = 0; k < wordCount; k++) {
if(approxMatch(words[k], prefixString) <= 3) {
//leve work
secondApproxValues.add(value);
break;
}
}
}
}
}
newValues.addAll(approxValues);
newValues.addAll(secondApproxValues);
results.values = newValues;
results.count = newValues.size();
}
return results;
}
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
//noinspection unchecked
mObjects = (List<T>) results.values;
notifyDataSetChanged();
clear();
for(T tmp : mObjects){
add(tmp);
}
notifyDataSetChanged();
}
}
private int approxMatch (String s, String t) {
// an approxmimate string matching algo
return p;
}
}
Теперь в адаптере getView
вам нужно обратиться к объекту mObjects, чтобы получить последние доступные значения для ListView