带有动画的Android Recycler视图适配器过滤器

时间:2016-12-02 09:14:23

标签: android android-recyclerview recycler-adapter android-filterable

我正在尝试为Android中的RecyclerView Adapter优化过滤方法。该列表用作ArrayList。我见过这个post但他们每次都会从原始列表中过滤掉。

示例:如果String' a'有10个结果。然后用户输入' m' ,' am'结果是' a'结果(results.size()< = 10)。

我在这个问题上有三点要问,

  1. 我可以使用ArrayMap优化HashMap内存吗?我应该在String中使用逗号分隔位置而不是Integer对象数组或使用int原始数组的任何方式吗?
  2. 我在这个结果中没有得到任何动画,如何获得? (我正在使用notifyItemInserted仍然没有动画)
  3. 在Hashmap中应该保留多少数据,直到2个字符或者应该根据结果列表大小?
我很高兴知道除了这个代码以外什么都可以做得更好这些要点。

在下面的代码中,mList用于onBindViewHolder方法。 copyList始终包含所有数据(不对其进行插入或删除)。

class MyFilter extends Filter {


        /**
         * 1. check do we have search results available (check map has this key)
         * 2. if available, remove all rows and add only those which are value for that key (string)
         * 3. else check do we have any key available starting like this, s=har, already available -ha then it can be reused
         *
         * @param constraint
         */
        @Override
        protected FilterResults performFiltering(CharSequence constraint) {
            //Here you have to implement filtering way
            final FilterResults results = new FilterResults();

            if (!mSearchMap.containsKey(constraint.toString())) {
                String supersetKey = getSupersetIfAvailable(mSearchMap, constraint.toString());
                if (supersetKey == null) {
                    List<Integer> foundPositions = doFullSearch(copyList, constraint.toString());
                    mSearchMap.put(constraint.toString(), foundPositions);
                } else {
                    List<Integer> foundPositions = filterFromSuperset(copyList, mSearchMap.get(supersetKey), constraint.toString());
                    mSearchMap.put(constraint.toString(), foundPositions);
                }
            }


            return results;
        }

        private String getSupersetIfAvailable(Map<String, List<Integer>> mSearchMap, String s) {
            Set<String> set = mSearchMap.keySet();
            List<String> list = new ArrayList<>(set);
            Collections.sort(list);
            Collections.reverse(list);
            for (String c : list) {
                if (s.startsWith(c)) {
                    return c;
                }
            }
            return null;
        }
        private List<Integer> filterFromSuperset(List<WeekWorkBean> list, List<Integer> supersetResults, String s) {
            List<Integer> results = new ArrayList<>();
            String lowerS = s.toLowerCase();
            for (int i = 0; i < supersetResults.size(); i++) {
                if (list.get(supersetResults.get(i)).getEmpName().toLowerCase().startsWith(lowerS)) {
                    results.add(supersetResults.get(i));
                }
            }
            return results;
        }

        private List<Integer> doFullSearch(List<WeekWorkBean> list, String s) {
            List<Integer> results = new ArrayList<>();
            for (int i = 0; i < list.size(); i++) {
                if (list.get(i).getEmpName().toLowerCase().startsWith(s.toLowerCase())) {
                    results.add(i);
                }
            }
            return results;
        }

        @Override
        protected void publishResults(CharSequence constraint, FilterResults results) {
            // here you can use result - (f.e. set in in adapter list)
            mList.clear();
            notifyDataSetChanged();
            List<Integer> res = mSearchMap.get(constraint.toString());
            int j = 0;
            for (Integer i : res) {
                mList.add(copyList.get(i));
                notifyItemInserted(j++);
            }
        }
    }

3 个答案:

答案 0 :(得分:3)

检查一下 https://medium.com/@iammert/using-diffutil-in-android-recyclerview-bdca8e4fbb00#.ehc0gaijt

您正在寻找DiffUtils。您可以在Rx链中使用它将其移出mainThread以获取大数据。 这是一个例子 https://medium.com/@nullthemall/diffutil-is-a-must-797502bc1149#.yg35y9q9b

答案 1 :(得分:1)

要给出第二点的答案,你可以试试这个:

 notifyItemRangeChanged(pos, ItemList.size());

答案 2 :(得分:1)

  1. HashMap是一个Map,还有TreeMap,LinkedHashMap和Hashtable。他们每个人都有自己的功能,界面是Map,而不是Collection。您还可以使用其他数据结构,如Treeset,HashSet,ArrayList,LinkedList等。这些结构来自Set和List接口,它扩展到Collection接口。你可以使用它们中的每一个。

  2. 如果您向集合中插入任何对象,请使用notifyItemInserted(int position),如果删除任何对象,请使用notifyItemRemoved(int position),如果您使用notifyDataSetChanged()更新任何对象。注意收集长度和适配器视图计数的相等性。

  3. 您可以在地图中存储您想要的参数。没有任何限制。但是你应该为你选择最好的收藏,设置,列表或地图。