RecyclerView,ListView实现顶部悬浮、字母排序、过滤搜索

抓住人生中的一分一秒,胜过虚度中的一月一年!

前言

在实际开发中避免不了字母排序,过滤搜索等问题,闲暇时对此做了个demo,希望对大家有所帮助,本demo分别用ListView,RecyclerView各实现了一版本,所以大家可以因情况随便使用

首先看下效果

sidebar-master.gif

实现目标

1、汉字转拼音,按拼音排序
2、字母显示一次
3、顶部字母悬停效果,上滑动画效果实现
4、侧滑字母栏索引跳转到指定字母
5、搜索框字母、数字等多条件搜索


分步骤开始实现

1、汉字转拼音,按拼音排序

如果想按字母排序,必然会涉及到汉字转拼音,本demo是用pinyin4j来实现的,需要将pingyin4j.jar包导入项目,或者在线依赖也可以,需要工具类如下

public static String getPingYin(String chineseStr) throws BadHanyuPinyinOutputFormatCombination {
        String zhongWenPinYin = "";
        char[] chars = chineseStr.toCharArray();

        for (int i = 0; i < chars.length; i++) {
            String[] pinYin = PinyinHelper.toHanyuPinyinStringArray(chars[i], getDefaultOutputFormat());
            if (pinYin != null)
                zhongWenPinYin += pinYin[0];
            else
                zhongWenPinYin += chars[i];
        }
        return zhongWenPinYin;
    }

    private static HanyuPinyinOutputFormat getDefaultOutputFormat() {
        HanyuPinyinOutputFormat format = new HanyuPinyinOutputFormat();
        format.setCaseType(HanyuPinyinCaseType.UPPERCASE);
        format.setToneType(HanyuPinyinToneType.WITHOUT_TONE);
        format.setVCharType(HanyuPinyinVCharType.WITH_U_AND_COLON);
        return format;
    }

将集合中字母转化为拼音,将第一字母取出,保存到实体类中

for (int i = 0; i < mList.size(); i++) {
            String pinyin = PinYinKit.getPingYin(mList.get(i).getName());
            String sortString = "";
            if (!TextUtils.isEmpty(pinyin)) {
                sortString = pinyin.substring(0, 1).toUpperCase();
            }
            if (sortString.matches("[A-Z]")) {
                mList.get(i).setSortLetters(sortString.toUpperCase());
            } else {
                mList.get(i).setSortLetters("#");
            }
        }

对集合中字母进行排序

//排序
Collections.sort(mList, new PinyinComparatorAdmin());

public static class PinyinComparatorAdmin implements Comparator {
        @Override
        public int compare(CountryBean o1, CountryBean o2) {
            if (o1.getSortLetters().equals("@") || o2.getSortLetters().equals("#")) {
                return -1;
            } else if (o1.getSortLetters().equals("#") || o2.getSortLetters().equals("@")) {
                return 1;
            } else {
                return o1.getSortLetters().compareTo(o2.getSortLetters());
            }
        }
    }

2、字母显示一次

通过上述代码,拼音已进行排序,字母如果只显示一次,我们可以遍历集合,取出当前position对应的拼音,如果它是第一个出现的,代表同类字母第一次出现,让这个position字母显示,其余同类字母全部隐藏

 public static void initLetter(List mList) {
        for (int i = 0; i < mList.size(); i++) {
            if (i == getPositionForSection(mList, mList.get(i).getSortLetters().charAt(0))) {
                mList.get(i).setLetter(true);
            } else {
                mList.get(i).setLetter(false);
            }
        }
    }

/**
     * 方法含义:将当前字母传入方法体中, 来获取当前字母在集合中第一次出现的位置position  如果等于当前item的position,UI字母栏
     * 显示,如果不是,UI字母栏隐藏
     *
     * @param section
     * @return 对应集合中第一个出现的字母
     */
    public static int getPositionForSection(List mList, int section) {
        for (int i = 0; i < mList.size(); i++) {
            String sortStr = mList.get(i).getSortLetters();
            char firstChar = sortStr.toUpperCase().charAt(0);
            if (firstChar == section) {
                return i;
            }
        }
        return -1;
    }

3、顶部字母悬停效果,上滑动画效果实现

布局列表顶部放一个固定控件如A控件,每个item都需要含有个A控件(同类字母A控件只显示一次),监听列表的最顶部的item,A控件显示列表最顶部item对应的首字母,列表向上滑动,如果显示字母的item滑动到距顶部高度等于A控件高度时,让A控件跟随向上平移,当A控件平移距离等于A控件高度时,证明列表中字母控件和A控件位置重叠,所以让A控件显示在最初位置,这样便实现了完美视差体验

所以需要通过Listview或RecyclerView的OnScrollListener来实现此效果

RecyclerView 监听
private class mScrollListener extends RecyclerView.OnScrollListener {

        private int mFlowHeight = 0;
        private int mCurrentPosition = -1;

        @Override
        public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
            if (mLlIndex != null || mFlowHeight < 1) {
                mFlowHeight = mLlIndex.getMeasuredHeight();
            }
        }

        @Override
        public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
            int firstVisibleItemPosition = layoutManager.findFirstVisibleItemPosition();
            View view = layoutManager.findViewByPosition(firstVisibleItemPosition + 1);

            if (view != null) {
                //当item距顶部距离小于等于A控件原始高度,并且当前item是第一个字母  便让A控件平移,反之固定到起始位置
                if (view.getTop() <= mFlowHeight && mCountryList.get(firstVisibleItemPosition + 1).getLetter()) {
                    mLlIndex.setY(view.getTop() - mFlowHeight);
                } else {
                    mLlIndex.setY(0);
                }
            }

            mCurrentPosition = firstVisibleItemPosition;
            if (mCountryList.size() > 0) {
                mTvIndex.setText(mCountryList.get(mCurrentPosition).getSortLetters());
                mLlIndex.setVisibility(View.VISIBLE);
            } else {
                mLlIndex.setVisibility(View.GONE);
            }
        }
    }
ListView 监听
private class mScrollListener implements AbsListView.OnScrollListener {

        private int mCurrentPosition = -1;

        @Override
        public void onScrollStateChanged(AbsListView absListView, int i) {
            if (mLlIndex != null || mFlowHeight < 1) {
                mFlowHeight = mLlIndex.getMeasuredHeight();
            }
        }

        @Override
        public void onScroll(AbsListView absListView, int position, int i1, int i2) {
            int firstVisibleItemPosition = absListView.getFirstVisiblePosition();
            View view = absListView.getChildAt(position + 1 - absListView.getFirstVisiblePosition());

            if (view != null) {
                //当item距顶部距离小于等于A控件原始高度,并且当前item是第一个字母  便让A控件平移,反之固定到起始位置
                if (view.getTop() <= mFlowHeight && mCountryList.get(firstVisibleItemPosition + 1).getLetter()) {
                    mLlIndex.setY(view.getTop() - mFlowHeight);
                } else {
                    mLlIndex.setY(0);
                }
            }

            if (mCurrentPosition != firstVisibleItemPosition) {
                mCurrentPosition = firstVisibleItemPosition;
                if (mCountryList.size() > 0) {
                    mTvIndex.setText(mCountryList.get(mCurrentPosition).getSortLetters());
                }
            }
        }
    }

4、侧滑字母栏索引跳转到指定字母

侧滑字母栏索就不必说了,有前人造好的轮子,介绍下ListView和RecyclerView各自的跳到指定position方法

LIstView跳转方法
 mListView.setSelection(position);
RecyclerView跳转方法
 /**
                 * 直接到指定位置
                 */
                layoutManager.scrollToPositionWithOffset(position, 0);
//                layoutManager.setStackFromEnd(true);
                /**
                 * 滚动到指定位置(有滚动效果)
                 */
//                LinearSmoothScroller s1 = new TopSmoothScroller(this);
//                s1.setTargetPosition(position);
//                layoutManager.startSmoothScroll(s1);
5、搜索框字母、数字等多条件搜索

此步注意一步,根据拼音搜索或数字搜索出来的集合字母出现次数已发生改变,所有必须再次对集合进行排序,UI才能正常显示

private void filerData(String str) throws BadHanyuPinyinOutputFormatCombination {
        if (TextUtils.isEmpty(str)) {
            mCountryList.clear();
            mCountryList.addAll(mCountryListAll);
        } else {
            mCountryList.clear();
            for (CountryBean ms : mCountryListAll) {
                String name = ms.getName();
                String code = ms.getCode();
                if (name.indexOf(str.toString()) != -1
                        || PinYinKit.getPingYin(name).startsWith(str.toString())
                        || PinYinKit.getPingYin(name).startsWith(str.toUpperCase().toString())
                        || name.contains(str)

                        || PinYinKit.getPingYin(code).startsWith(str.toString())
                        || PinYinKit.getPingYin(code).startsWith(str.toUpperCase().toString())
                        || code.contains(str)
                        ) {
                    mCountryList.add(ms);
                }
            }
        }
        PinYinKit.initLetter(mCountryList);
        layoutManager.scrollToPositionWithOffset(0, 0);
        mAdapter.notifyDataSetChanged();
    }

最后,祝大家创作愉快

github地址:https://github.com/LPTim/SideBar-master

你可能感兴趣的:(RecyclerView,ListView实现顶部悬浮、字母排序、过滤搜索)