Jeśli ktoś jest nadal zainteresowany tym tematem, uważam, że najlepszym podejściem do filtrowania list jest utworzenie ogólnej klasy Filter i użycie jej z pewnymi podstawowymi technikami refleksji / generycznymi zawartymi w pakiecie Java old school SDK. Oto co zrobiłem:
public class GenericListFilter<T> extends Filter {
public GenericListFilter (List<T> list, String reflectMethodName, ArrayAdapter<T> adapter) {
super ();
mInternalList = new ArrayList<>(list);
mAdapterUsed = adapter;
try {
ParameterizedType stringListType = (ParameterizedType)
getClass().getField("mInternalList").getGenericType();
mCompairMethod =
stringListType.getActualTypeArguments()[0].getClass().getMethod(reflectMethodName);
}
catch (Exception ex) {
Log.w("GenericListFilter", ex.getMessage(), ex);
try {
if (mInternalList.size() > 0) {
T type = mInternalList.get(0);
mCompairMethod = type.getClass().getMethod(reflectMethodName);
}
}
catch (Exception e) {
Log.e("GenericListFilter", e.getMessage(), e);
}
}
}
@Override protected FilterResults performFiltering(CharSequence constraint) {
FilterResults results = new FilterResults();
List<T> filteredContents = new ArrayList<>();
if ( constraint.length() > 0 ) {
try {
for (T obj : mInternalList) {
String result = (String) mCompairMethod.invoke(obj);
if (result.toLowerCase().startsWith(constraint.toString().toLowerCase())) {
filteredContents.add(obj);
}
}
}
catch (Exception ex) {
Log.e("GenericListFilter", ex.getMessage(), ex);
}
}
else {
filteredContents.addAll(mInternalList);
}
results.values = filteredContents;
results.count = filteredContents.size();
return results;
}
@Override protected void publishResults(CharSequence constraint, FilterResults results) {
mAdapterUsed.clear();
mAdapterUsed.addAll((List<T>) results.values);
if ( results.count == 0 ) {
mAdapterUsed.notifyDataSetInvalidated();
}
else {
mAdapterUsed.notifyDataSetChanged();
}
}
private ArrayAdapter<T> mAdapterUsed;
private List<T> mInternalList;
private Method mCompairMethod;
}
Następnie jedyne, co musisz zrobić, to utworzyć filtr jako klasę składową (prawdopodobnie w ramach elementu „onCreate” widoku), przekazując odniesienie do adaptera, listę i metodę, która ma zostać wywołana w celu filtrowania:
this.mFilter = new GenericFilter<MyObjectBean> (list, "getName", adapter);
Jedyne, czego teraz brakuje, to przesłonięcie metody „getFilter” w klasie adaptera:
@Override public Filter getFilter () {
return MyViewClass.this.mFilter;
}
Gotowe! Powinieneś pomyślnie przefiltrować swoją listę - Oczywiście powinieneś również zaimplementować algorytm filtru w najlepszy sposób, który opisuje Twoje potrzeby, poniższy kod jest tylko przykładem. . Mam nadzieję, że to pomogło, uważaj.