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

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

前言

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

首先看下效果

1688293310204.png



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();
    }

最后,祝大家创作愉快

相关文章
|
Java 开发工具 Android开发
Android Studio OpenCV 4.5.2环境搭建
Android Studio OpenCV 4.5.2环境搭建
964 0
|
11月前
|
Java 关系型数据库 MySQL
2025 年互联网公司校招 Java 面试题总结及答案实操示例解析
本项目基于Spring Boot 3与Java 17技术栈,围绕校园招聘常见面试题,提供核心知识点的实操示例。涵盖多线程、RESTful API设计、数据库操作(Spring Data JPA)、事务管理及异常处理等。通过完整代码实现与运行步骤,帮助理解用户管理、线程池配置等实际应用场景。资源包含项目结构、关键代码示例(如User实体类、UserService服务层、ThreadService多线程实现)及数据库迁移脚本,适合深入学习与实践。环境要求:JDK 17+、Maven 3.8+、MySQL 8.0+。
420 3
|
数据库 Android开发 开发者
Android常用的room增删改查语句(外部数据库)
本文分享了将一个原生数据库驱动的单词APP重构为使用Room库的过程及遇到的问题,重点解决了Room中增删改查的常用语句实现。文章通过具体示例(以“forget”表为例),详细展示了如何定义实体类、Dao接口、Database类以及Repository和ViewModel的设计与实现。同时,提供了插入、删除、更新和查询数据的代码示例,包括模糊查询、分页加载等功能。此外,针对外部数据库导入问题,作者建议可通过公众号“计蒙不吃鱼”获取更多支持。此内容适合有一定Room基础的开发者深入学习。
344 0
Android常用的room增删改查语句(外部数据库)
|
XML Android开发 UED
|
编解码 人工智能 文字识别
用PDF转换图片的方式弥补通义千问在扫描版PDF支持方面的缺失
当前通义千问Web版和本地版qwen-VL在处理扫描版PDF时均无法直接识别,导致实际应用中处理大量扫描PDF的需求难以满足。为此,通过使用Python的pdf2image库,可将PDF文件转换为图片,再进行OCR处理,实现解决方案。文中提供了具体的代码示例,展示了如何将PDF文件的每一页转换成图片,并保存至指定文件夹,为后续的OCR处理做好准备。
1135 14
|
安全 Java 编译器
Kotlin学习教程(一)
在5月18日谷歌在I/O开发者大会上宣布,将Kotlin语言作为安卓开发的一级编程语言。并且会在Android Studio 3.0版本全面支持Kotlin。 Kotlin是一个基于JVM的新的编程语言,由JetBrains开发。JetBrains作为目前广受欢迎的 Java IDE IntelliJ的提供商,在Apache许可下已经开源其Kotlin编程语言。 Kotlin可以编译成Java字节码,也可以编译成JavaScript,方便在没有JVM的设备上运行。 Kotlin已正式成为Android官方开发语言。
839 4
Kotlin学习教程(一)
|
XML Java API
23. 【Android教程】轮播滚动视图:ViewFlipper
23. 【Android教程】轮播滚动视图:ViewFlipper
909 2
|
自然语言处理 程序员 开发工具
iOS生成Bundle 资源文件包与使用
iOS生成Bundle 资源文件包与使用
1061 0
|
XML 数据库 Android开发
[Android]搜索框SearchView
[Android]搜索框SearchView
417 0
|
Web App开发 Java Android开发
android 7.1分屏(应用可能无法在分屏模式下正常运行)
android 7.1分屏(应用可能无法在分屏模式下正常运行)
951 0
android 7.1分屏(应用可能无法在分屏模式下正常运行)