2013-02-02 56 views
78

我已经创造了Android的列表视图,我想添加列表上方编辑文本,并且当用户输入文本列表将被过滤根据用户输入列表视图筛选的Android

谁能告诉我请,如果有一种方法来过滤Android中的列表适配器?

+1

嗨尝试这个例子[示例一](https://rakhi577.wordpress.com/2012/06/26/buttons-on-list-view-with-easy-searching-in-android/)和第二个一个[示例2](http://stackoverflow.com/questions/2463777/autocompletetextview-with-custom-list-how-to-set-up-onitemclicklistener)我已经实现了基于这个教程的相同..我希望这将帮助您 – Pragnani

+0

+1升masr ..... –

+4

最佳答案只是没有提供足够的信息给我。 [这个答案](http://stackoverflow.com/a/24771174/11912)到一个类似的问题有更多的上下文,是完全足够的信息,让我整理出来。 –

回答

122

是的。 在其aml布局文件中的列表视图顶部添加一个EditText。 而在你的活动/片段..

lv = (ListView) findViewById(R.id.list_view); 
    inputSearch = (EditText) findViewById(R.id.inputSearch); 

// Adding items to listview 
adapter = new ArrayAdapter<String>(this, R.layout.list_item, R.id.product_name, products); 
lv.setAdapter(adapter);  
inputSearch.addTextChangedListener(new TextWatcher() { 

    @Override 
    public void onTextChanged(CharSequence cs, int arg1, int arg2, int arg3) { 
     // When user changed the Text 
     MainActivity.this.adapter.getFilter().filter(cs); 
    } 

    @Override 
    public void beforeTextChanged(CharSequence arg0, int arg1, int arg2, int arg3) { } 

    @Override 
    public void afterTextChanged(Editable arg0) {} 
}); 

这里的基本是一个OnTextChangeListener添加到您编辑文本和其回调方法里面将过滤器应用于您的ListView的适配器。

编辑

为了让过滤器,以自定义BaseAdapter你“会需要实现可筛选接口。

class CustomAdapter extends BaseAdapter implements Filterable { 

    public View getView(){ 
    ... 
    } 
    public Integer getCount() 
    { 
    ... 
    } 

    @Override 
    public Filter getFilter() { 

     Filter filter = new Filter() { 

      @SuppressWarnings("unchecked") 
      @Override 
      protected void publishResults(CharSequence constraint, FilterResults results) { 

       arrayListNames = (List<String>) results.values; 
       notifyDataSetChanged(); 
      } 

      @Override 
      protected FilterResults performFiltering(CharSequence constraint) { 

       FilterResults results = new FilterResults(); 
       ArrayList<String> FilteredArrayNames = new ArrayList<String>(); 

       // perform your search here using the searchConstraint String. 

       constraint = constraint.toString().toLowerCase(); 
       for (int i = 0; i < mDatabaseOfNames.size(); i++) { 
        String dataNames = mDatabaseOfNames.get(i); 
        if (dataNames.toLowerCase().startsWith(constraint.toString())) { 
         FilteredArrayNames.add(dataNames); 
        } 
       } 

       results.count = FilteredArrayNames.size(); 
       results.values = FilteredArrayNames; 
       Log.e("VALUES", results.values.toString()); 

       return results; 
      } 
     }; 

     return filter; 
    } 
} 

perfromFiltering()你需要做的实际比较将搜索查询转换为数据库中的值,并将结果传递给publishResults()方法

+0

我已经创建了一个自定义适配器,它扩展了BaseAdapter并在其中定义了我的对象的向量,该对象将显示在列表中,当我尝试使用上述代码时,我无法在适配器中找到getFilter方法,你请告诉我,如果我必须实现任何接口? –

+0

在BaseAdapter的情况下过滤数据有点棘手。您将不得不实施可执行界面来实现BaseAdapter。您将有getFilter()方法,并且在其中您必须实现两个回调方法才能使用; void publishResults()和FilterResults performFiltering(CharSequence约束)。 –

+0

你能用一个简单的例子来支持吗? –

9

实现您的适配器可筛选:

public class vJournalAdapter extends ArrayAdapter<JournalModel> implements Filterable{ 
private ArrayList<JournalModel> items; 
private Context mContext; 
.... 

然后创建过滤器类:

private class JournalFilter extends Filter{ 

    @Override 
    protected FilterResults performFiltering(CharSequence constraint) { 
     FilterResults result = new FilterResults(); 
     List<JournalModel> allJournals = getAllJournals(); 
     if(constraint == null || constraint.length() == 0){ 

      result.values = allJournals; 
      result.count = allJournals.size(); 
     }else{ 
      ArrayList<JournalModel> filteredList = new ArrayList<JournalModel>(); 
      for(JournalModel j: allJournals){ 
       if(j.source.title.contains(constraint)) 
        filteredList.add(j); 
      } 
      result.values = filteredList; 
      result.count = filteredList.size(); 
     } 

     return result; 
    } 
    @SuppressWarnings("unchecked") 
    @Override 
    protected void publishResults(CharSequence constraint, FilterResults results) { 
     if (results.count == 0) { 
      notifyDataSetInvalidated(); 
     } else { 
      items = (ArrayList<JournalModel>) results.values; 
      notifyDataSetChanged(); 
     } 
    } 

} 

这样,你的适配器是可筛选,你可以通过过滤器项目适配器的过滤器,做的工作。 我希望这会有所帮助。

1

如果有人对此主题感兴趣,我发现过滤列表的最佳方法是创建一个通用的Filter类,并将其与Java老派SDK包中包含的一些基本反射/泛型技术结合使用。下面是我所做的:

public class GenericListFilter<T> extends Filter { 

    /** 
    * Copycat constructor 
    * @param list the original list to be used 
    */ 
    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); 
      } 

     } 
    } 

    /** 
    * Let's filter the data with the given constraint 
    * @param constraint 
    * @return 
    */ 
    @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; 
    } 

    /** 
    * Publish the filtering adapter list 
    * @param constraint 
    * @param 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(); 
     } 
    } 

    // class properties 
    private ArrayAdapter<T> mAdapterUsed; 
    private List<T> mInternalList; 
    private Method mCompairMethod; 
} 

而且事后,你需要做的唯一的事情就是创建过滤器作为成员类(有可能在视图的“的onCreate”)通过你的适配器参考,您的列表,以及被调用的方法过滤:

this.mFilter = new GenericFilter<MyObjectBean> (list, "getName", adapter); 

现在唯一缺少的,就是要覆盖“用getFilter”方法的适配器类:

@Override public Filter getFilter() { 
    return MyViewClass.this.mFilter; 
} 

全部完成!你应该成功过滤你的列表 - 当然,你也应该实现你的过滤器算法描述你的需要的最佳方式,下面的代码只是一个例子。。希望它有帮助,保重。

+0

我不知道有关android,但我记得被告知尽量避免反射,如果可能在C#因为它是相当资源密集型(我通常在Windows移动应用程序所以这可能是一个问题),这是否适用于Android?或者反射与构建没有泛型的实际类相同的效果?我正在考虑创建一个可过滤的模板,并添加用作参数的类和方法 – Cruces

+0

是的,你是对的。这同样适用于这里,反射给程序处理一个权重,但在这种情况下,它是一个非常简单的使用它,并且因为我们使用通用/模板表示法,它也有助于编译器。祝你好运! – jbrios777