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<CountryBean> {
        @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<CountryBean> 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<CountryBean> 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

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351