2012-08-07 20 views
5

Quindi ho personalizzato ArrayAdapter in modo da poter utilizzare la vista Titolo/sottotitolo disponibile su ListView. Ho un EditText che accetta una stringa e filtra l'adattatore.ArrayAdapter personalizzato di Android non si aggiorna dopo il filtro

Il filtro lavora nel senso che è filtrando gli oggetti corretti (posso dire facendo clic su di esso e inizia l'intento con gli "extra" correct.)

Tuttavia, anche se i lavori di filtraggio, il gli elementi nell'adattatore non vengono aggiornati per visualizzare le informazioni corrette ... Il titolo e il sottotitolo non sono corretti.

Diciamo che abbiamo articoli da 0 a 9 su ListView, filtro a 3 elementi con una ricerca e diciamo che gli elementi filtrati sono 5, 6, 9 ... 3 elementi sono visualizzati, ma sono i primi 3 articoli della ricerca preliminare originale ListView (0-2). Se clicco sull'elemento 2 (il terzo elemento), il contenuto di 9 è contenuto nel nuovo intento. Questo è corretto per i criteri di ricerca, tuttavia il titolo riflette le informazioni corrette.

Non sono sicuro di cosa devo comunicare allo ListView per l'aggiornamento. Non penso che sia notifyDataSetChanged();

Qualsiasi aiuto è apprezzato. Grazie!

public class myListAdapter extends ArrayAdapter<Pizza>{ 
    private ArrayList<Pizza> items; 
    private PizzaViewHolder myListHolder; 

    private class PizzaViewHolder{ 
     TextView title; 
     TextView subtitle; 
    } 

    public myListAdapter(Context context, int textViewResourceId, ArrayList<Pizza> items) { 
     super(context, textViewResourceId, items); 
     this.items = items; 
     // TODO Auto-generated constructor stub 
    } 

    @Override 
    public View getView(int pos, View convertView, ViewGroup parent){ 
     View v = convertView; 
     if(v == null){ 
      LayoutInflater vi = (LayoutInflater)getSystemService(LAYOUT_INFLATER_SERVICE); 
      v = vi.inflate(R.layout.myList_item, null); 
      myListHolder = new PizzaViewHolder(); 
      myListHolder.title = (TextView)v.findViewById(R.id.title); 
      myListHolder.subtitle = (TextView)v.findViewById(R.id.subtitle); 
      v.setTag(myListHolder); 
     }else myListHolder = (PizzaViewHolder)v.getTag(); 

     Pizza myList = items.get(pos); 

     if (myList != null){ 
      myListHolder.title.setText(myList.getTitle()); 
      myListHolder.subtitle.setText(myList.getSubTitle()); 
     }  
     return v;   
    } 
} 

Questa è la ricerca

private TextWatcher filterTextWatcher = new TextWatcher(){ 

    public void afterTextChanged(Editable s) { 
     // TODO Auto-generated method stub 
     if(!s.equals("")){ 
      ((Filterable) this.listView1.getAdapter()).getFilter().filter(s); 
      } 
    } 

    public void beforeTextChanged(CharSequence s, int start, int count, 
      int after) { 
     // TODO Auto-generated method stub 

    } 

    public void onTextChanged(CharSequence s, int start, int before, 
      int count) { 
     // TODO Auto-generated method stub 

    } 

}; 
+0

Utilizzando un adattatore standard funziona bene, non è necessario utilizzare notifyDataSetChanged, la mia scheda personalizzata non aggiorna –

risposta

13

Da quello che ho deciso, sembra che avevo bisogno di un filtro personalizzato per il mio ArrayAdapter personalizzato. L'usanza ArrayAdapter ha un'implementazione override di filtro, ecco il codice:

import java.util.ArrayList; 
import android.content.Context; 
import android.util.Log; 
import android.view.LayoutInflater; 
import android.view.View; 
import android.view.ViewGroup; 
import android.widget.ArrayAdapter; 
import android.widget.Filter; 
import android.widget.TextView; 

public class PizzaAdapter extends ArrayAdapter<Pizza>{ 

private ArrayList<Pizza> original; 
private ArrayList<Pizza> fitems; 
private Filter filter; 


public PizzaAdapter(Context context, int textViewResourceId, ArrayList<Pizza> items) { 
    super(context, textViewResourceId, items); 

    this.original = new ArrayList<Pizza>(items); 
    this.fitems = new ArrayList<Pizza>(items); 
    this.filter = new PizzaFilter(); 
} 

@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.pizza_list_item, null); 
    } 

    Pizza pizza = fitems.get(position); 

    if(pizza != null){ 
     String subtitleString = new String("[" + pizza.getPizzaType() + "] " + pizza.getPizzaCategory() + ": " + pizza.getPizzaCode()); 

     TextView title = (TextView)v.findViewById(R.id.title); 
     TextView subtitle = (TextView)v.findViewById(R.id.subtitle); 

     if(title != null){ 
      title.setText(pizza.getPizzaName()); 
     } 
     if(subtitle != null){ 
      subtitle.setText(subtitleString); 
     } 
    } 
    return v; 
} 

@Override 
public Filter getFilter(){ 

    if(filter == null){ 
     filter = new PizzaFilter(); 
    } 
    return filter; 
} 

private class PizzaFilter extends Filter{ 
    @Override 
    protected FilterResults performFiltering(CharSequence constraint){ 
     FilterResults results = new FilterResults(); 
     String prefix = constraint.toString().toLowerCase(); 

     if (prefix == null || prefix.length() == 0){ 
      ArrayList<Pizza> list = new ArrayList<Pizza>(original); 
      results.values = list; 
      results.count = list.size(); 
     }else{ 
      final ArrayList<Pizza> list = new ArrayList<Pizza>(original); 
      final ArrayList<Pizza> nlist = new ArrayList<Pizza>(); 
      int count = list.size(); 

      for (int i = 0; i<count; i++){ 
       final Pizza pizza = list.get(i); 
       final String value = Pizza.getPizzaName().toLowerCase(); 

       if(value.contains(prefix)){ 
        nlist.add(pizza); 
       } 
       results.values = nlist; 
       results.count = nlist.size(); 
      } 
     } 
     return results; 
    } 

    @SuppressWarnings("unchecked") 
    @Override 
    protected void publishResults(CharSequence constraint, FilterResults results) { 
     fitems = (ArrayList<Pizza>)results.values; 
     notifyDataSetChanged(); 
     clear(); 
     int count = fitems.size(); 
     for(int i = 0; i<count; i++){ 
      add(fitems.get(i)); 
      notifyDataSetInvalidated(); 
     } 
    } 
} 
} 

Si scopre che un implmentation personalizzato del filtro aggiornato display quando si cercato. Speriamo che questo possa aiutare alcune persone.

+3

uno dei migliori e più semplici esempi su Internet. + – AlexAndro

+0

Qualche idea su come posso usare il tuo codice per risolvere questo problema: http://stackoverflow.com/questions/20524417/listview-is-blank-while-using-getfilter-function – Si8

+0

'String prefix = constraint.toString(). toLowerCase(); 'può lanciare' NullPointerException' – mallaudin

Problemi correlati