Niestandardowe filtrowanie ArrayAdapter w ListView
Jestem początkujący w Androidzie, ale próbowałem zrobić niestandardowe filtrowanie listview i jakoś to działało. Jedynym problemem jest to, że ArrayList, że zachowałem wszystkie wartości ("oryginalna" ArrayList ), jest coraz niżej na pozycji w każdym filtrowaniu. Nie potrafię tego wyjaśnić, ale pomyślałem, że możesz mi jakoś pomóc .
W każdym razie tutaj jest Niestandardowy ArrayAdaptor:
public class PkmnAdapter extends ArrayAdapter<Pkmn> {
private ArrayList<Pkmn> original;
private ArrayList<Pkmn> fitems;
private Filter filter;
public PkmnAdapter(Context context, int textViewResourceId, ArrayList<Pkmn> items) {
super(context, textViewResourceId, items);
this.original = items;//new ArrayList<Pkmn>();
this.fitems = items;//new ArrayList<Pkmn>();
}
@Override
public void add(Pkmn item){
original.add(item);
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View v = convertView;
if (v == null) {
LayoutInflater vi = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
v = vi.inflate(R.layout.row, null);
}
Pkmn pkmn = original.get(position);
if (pkmn != null) {
TextView tt = (TextView) v.findViewById(R.id.RlabPName);
TextView dex = (TextView)v.findViewById(R.id.RlabDex);
ImageView img = (ImageView)v.findViewById(R.id.RimgPkmn);
if (tt != null) { tt.setText(pkmn.getName()); }
if (dex != null){ dex.setText(CalcDex(pkmn.getId())); }
if (img != null){
int resId = getContext().getResources().getIdentifier("dex" + pkmn.getId(), "drawable", "com.compileguy.pokebwteam");
img.setImageResource(resId);
}
}
return v;
}
@Override
public Filter getFilter()
{
if (filter == null)
filter = new PkmnNameFilter();
return filter;
}
private class PkmnNameFilter extends Filter
{
@Override
protected FilterResults performFiltering(CharSequence constraint)
{
FilterResults results = new FilterResults();
String prefix = constraint.toString().toLowerCase();
if (prefix == null || prefix.length() == 0)
{
ArrayList<Pkmn> list = new ArrayList<Pkmn>(original);
results.values = list;
results.count = list.size();
}
else
{
final ArrayList<Pkmn> list = original;
int count = list.size();
final ArrayList<Pkmn> nlist = new ArrayList<Pkmn>(count);
for (int i=0; i<count; i++)
{
final Pkmn pkmn = list.get(i);
final String value = pkmn.getName().toLowerCase();
if (value.startsWith(prefix))
{
nlist.add(pkmn);
}
}
results.values = nlist;
results.count = nlist.size();
}
return results;
}
@SuppressWarnings("unchecked")
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
fitems = (ArrayList<Pkmn>)results.values;
clear();
int count = fitems.size();
for (int i=0; i<count; i++)
{
Pkmn pkmn = (Pkmn)fitems.get(i);
add(pkmn);
}
if (fitems.size() > 0)
notifyDataSetChanged();
else
notifyDataSetInvalidated();
}
}
private String CalcDex(int id){
String s = String.valueOf(id);
if (s.length() == 1)
s = "00"+s;
else if (s.length() == 2)
s = "0"+s;
return '#'+s;
}
}
Uwaga: listview pokazuje poprawnie elementy, ale gdy dla exaple usuwam litera w polu edycji (która uruchamia filtrowanie) tutaj zaczynają się problemy.
- - - EDIT - - -
@Janusz: Wielkie dzięki za odpowiedź . To rozwiązało mój problem .Oto kod źródłowy, który działa dla mnie, więc jeśli ktoś ma ten sam problem, może spróbować tego:
private ArrayList<Pkmn> original;
private ArrayList<Pkmn> fitems;
private Filter filter;
public PkmnAdapter(Context context, int textViewResourceId, ArrayList<Pkmn> items) {
super(context, textViewResourceId, items);
this.original = new ArrayList<Pkmn>(items);
this.fitems = new ArrayList<Pkmn>(items);
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View v = convertView;
if (v == null) {
LayoutInflater vi = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
v = vi.inflate(R.layout.row, null);
}
Pkmn pkmn = fitems.get(position);
if (pkmn != null) {
TextView tt = (TextView) v.findViewById(R.id.RlabPName);
TextView dex = (TextView)v.findViewById(R.id.RlabDex);
ImageView img = (ImageView)v.findViewById(R.id.RimgPkmn);
if (tt != null) { tt.setText(pkmn.getName()); }
if (dex != null){ dex.setText(CalcDex(pkmn.getId())); }
if (img != null){
int resId = getContext().getResources().getIdentifier("dex" + pkmn.getId(), "drawable", "com.compileguy.pokebwteam");
img.setImageResource(resId);
}
}
return v;
}
@Override
public Filter getFilter()
{
if (filter == null)
filter = new PkmnNameFilter();
return filter;
}
private class PkmnNameFilter extends Filter
{
@Override
protected FilterResults performFiltering(CharSequence constraint)
{
FilterResults results = new FilterResults();
String prefix = constraint.toString().toLowerCase();
if (prefix == null || prefix.length() == 0)
{
ArrayList<Pkmn> list = new ArrayList<Pkmn>(original);
results.values = list;
results.count = list.size();
}
else
{
final ArrayList<Pkmn> list = new ArrayList<Pkmn>(original);
final ArrayList<Pkmn> nlist = new ArrayList<Pkmn>();
int count = list.size();
for (int i=0; i<count; i++)
{
final Pkmn pkmn = list.get(i);
final String value = pkmn.getName().toLowerCase();
if (value.startsWith(prefix))
{
nlist.add(pkmn);
}
}
results.values = nlist;
results.count = nlist.size();
}
return results;
}
@SuppressWarnings("unchecked")
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
fitems = (ArrayList<Pkmn>)results.values;
clear();
int count = fitems.size();
for (int i=0; i<count; i++)
{
Pkmn pkmn = (Pkmn)fitems.get(i);
add(pkmn);
}
}
}
}
3 answers
Twoim problemem są linie:
this.original = items;
this.fitems = items;
Items jest listą, której używasz do wyświetlania listy i umieszczenie jej w dwóch różnych zmiennych nie powoduje powstania dwóch różnych list. Nadajesz tylko dwie różne nazwy elementom listy.
Możesz użyć:
this.fitems = new ArrayList(items);
To powinno wygenerować nową listę, a zmiany na tej liście zmienią tylko listę fitems.
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2011-06-27 11:40:31
Możesz osiągnąć ten sam efekt, po prostu tworząc metodę toString()
na swojej klasie Pkmn, która zwraca wartość, według której chcesz filtrować.
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-06-08 10:44:24
Najlepszym sposobem na filtrowanie ArrayAdapter jest stworzenie własnej klasy filtrów:
private class MyFilter extends Filter
Następnie w tej funkcji Utwórz nową tablicę obiektów do wyświetlenia za filtrem (dobrą implementację znajdziesz w kodzie źródłowym class ArrayAdapter
)
@Override
protected FilterResults performFiltering(CharSequence prefix)
Teraz sztuczka jest w tej metodzie
@Override
protected void publishResults(CharSequence constraint, FilterResults results)
Kiedy używasz adaptera tablicy, nie możesz zrobić tego:
myAdapterData = results.values
Od tego czasu odłączasz swoje dane od super danych, musisz to zrobić, aby zachować odniesienie do super oryginalna tablica danych:
data.clear();
data.addAll((List<YourType>) results.values);
A następnie nadpisać
GetFilter ()
W adapterze, na przykład:
@Override
public Filter getFilter() {
if (filter == null) {
filter = new MyFilter();
}
return filter;
}
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2014-05-28 14:02:36