文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Android使用ExpandableListView实现三层嵌套折叠菜单

2024-04-02 19:55

关注

前段时间项目的新功能里有些页面需要三层嵌套列表实现,虽然在移动端这种很丑,但是需求就是需求。
本来想用各种View嵌套,然后发现系统有个ExpandableListView。就直接拿来用了。

理论上来说,ExpandableListView的二级嵌套和三级嵌套没有本质区别,如果把二级嵌套的子级换成一个新的ExpandableListView,就可以实现三级嵌套。

有了思路,关于ExpandableListView的三层嵌套就直接上手实现

这里说下我的需求是有些数据是只有二级,有些数据是三级的。如果你的需求是只有三级,不需要考虑三级二级混合的情况,下面有说明怎么处理。

效果图

ExpandableListView

ExpandableListView是官方提供的一个可展示折叠列表的控件。

它的基本用法如下

基本用法

ExpandableListView的基本用法很简单,它本质上就是ListView,所以用法也差不多,这里就不介绍了。

下面开始进入正题。

布局文件

先说下,因为是三级嵌套,所以需要四个布局文件,Activity页面本身需要一个布局文件,然后就是三级嵌套的三个布局文件。

Activity布局文件


<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:layout_height="match_parent">
    <ExpandableListView
            android:id="@+id/expand_view"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:cacheColorHint="#00000000"
            android:childIndicator="@color/white"
            android:divider="@null"
            android:fadeScrollbars="false"
            android:groupIndicator="@null"
            android:listSelector="#00000000"
            android:scrollbars="none" />
</LinearLayout>

我们可以通过ExpandableListView的默认属性来控制部分样式,这里贴上菜鸟教程的属性图片

一级菜单布局文件


<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="44dp"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:background="@drawable/chapter_gradient_group">

    <TextView
        android:id="@+id/adapter_title"
        android:layout_width="0dp"
        android:layout_height="match_parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        android:layout_marginHorizontal="10dp"
        android:paddingStart="20dp"
        android:singleLine="true"
        android:ellipsize="end"
        android:text="@string/groupName"
        android:textColor="@color/white"
        android:textSize="16sp"
        android:gravity="start|center_vertical" />

</androidx.constraintlayout.widget.ConstraintLayout>

二级菜单布局文件


<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@drawable/chapter_gradient_child">

    <TextView
        android:id="@+id/adapter_child_title"
        android:layout_width="match_parent"
        android:layout_height="40dp"
        android:ellipsize="end"
        android:gravity="start|center_vertical"
        android:paddingStart="30dp"
        android:paddingEnd="10dp"
        android:singleLine="true"
        android:text="@string/childName"
        android:textColor="@color/white"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</androidx.constraintlayout.widget.ConstraintLayout>

三级菜单布局文件


<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@drawable/chapter_gradient_grandson">

    <TextView
        android:id="@+id/adapter_grandson_title"
        android:layout_width="match_parent"
        android:layout_height="40dp"
        android:ellipsize="end"
        android:gravity="start|center_vertical"
        android:paddingStart="40dp"
        android:paddingEnd="10dp"
        android:singleLine="true"
        android:text="@string/grandsonName"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</androidx.constraintlayout.widget.ConstraintLayout>

Adapter

上面说过ExpandableListView继承自ListView,所以我们需要Adapter,三级嵌套,我们需要两个Adapter。

这里有必要说一下,为什么是两个Adapter,ExpandableListView的Adapter继承自BaseExpandableListAdapter。需要重写getGroupView和getChildView。这两个方法中的view分别inflate父级菜单的布局和子级菜单的布局文件。

所以我们上面的三个级别的菜单布局文件通过两个Adapter来连接。分别是一级菜单的Adapter和三级菜单的Adapter。

下面给出这两个Adapter的详细说明,需要注意的地方已经进行备注,请仔细看备注

一级菜单Adapter
最值得注意的是该Adapter的getChildView方法和getChildrenCount。因为有些数据不包含三级菜单,有些包含了三级菜单。另外,这个地方需要对下级嵌套的ExpandableListView进行处理。



public class ChapterExpandableAdapter extends BaseExpandableListAdapter {
    
    ...

    @Override
    public int getGroupCount() {
        // 父菜单长度
        return fatherChapterList.size();
    }

    @Override
    public int getChildrenCount(int groupPosition) {
        // 子菜单长度,嵌套所以返回只能1
        return 1;
    }

    @Override
    public View getGroupView(int groupPosition, boolean isExpanded, View convertView, ViewGroup parent) {
        GroupViewHolder groupHolder;
        // 尽可能重用旧view处理
        if (convertView == null) {
            convertView = LayoutInflater.from(parent.getContext()).inflate(R.layout.adapter_expandable_group_view, parent, false);
            groupHolder = new GroupViewHolder();
            groupHolder.groupTitle = convertView.findViewById(R.id.adapter_title);
            convertView.setTag(groupHolder);
        } else {
            groupHolder = (GroupViewHolder) convertView.getTag();
        }
        // 设置title
        groupHolder.groupTitle.setText(fatherChapterList.get(groupPosition).getName());
        return convertView;
    }

    @Override
    public View getChildView(int groupPosition, int childPosition, boolean isLastChild, View convertView, ViewGroup parent) {
        if (convertView == null) {
            convertView = new CustomExpandableListView(context);
        }
        CustomExpandableListView expandableListView = (CustomExpandableListView) convertView;
        // 加载子级Adapter
        ChapterExpandableLowAdapter lowAdapter = new ChapterExpandableLowAdapter(context);
        lowAdapter.setTotalList(fatherChapterList.get(groupPosition).getSec());
        expandableListView.setAdapter(lowAdapter);
       
        if (fatherChapterList.get(groupPosition).getSec().get(childPosition).getThird().size() == 0) {
            expandableListView.setGroupIndicator(null);
        }
        // 本身的父级,相当于三级目录的子级监听
        expandableListView.setOnGroupClickListener((parent12, v, groupPosition12, id) -> {
            // 如果第三层size为0,意味着没有三级菜单
            if (fatherChapterList != null && fatherChapterList.size() > 0 && fatherChapterList.get(groupPosition).getSec().get(groupPosition12).getThird().size() == 0) {
                // TODO 业务处理
            }
            // 存在第三级数据,事件分发机制继续想下传递
            return false;
        });
        expandableListView.setOnChildClickListener((parent1, v, groupPosition1, childPosition1, id) -> {
            // 三级菜单的业务处理
            return true;
        });
        return expandableListView;
    }

    
    @Override
    public boolean isChildSelectable(int groupPosition, int childPosition) {
        return true;
    }

    
    static class GroupViewHolder {
        TextView groupTitle;
    }
}

三级菜单Adapter
三级菜单的Adapter就和普通的二级嵌套时的Adapter相同,没什么特别注意的地方,所以只列出了getGroupView和getChildView方法代码


	@Override
    public View getGroupView(int groupPosition, boolean isExpanded, View convertView, ViewGroup parent) {
        ChapterExpandableLowAdapter.GroupViewHolder groupHolder;
        // 尽可能重用旧view处理
        if (convertView == null) {
            convertView = LayoutInflater.from(parent.getContext()).inflate(R.layout.adapter_expandable_child_view, parent, false);
            groupHolder = new ChapterExpandableLowAdapter.GroupViewHolder();
            groupHolder.groupTitle = convertView.findViewById(R.id.adapter_child_title);
            convertView.setTag(groupHolder);
        } else {
            groupHolder = (ChapterExpandableLowAdapter.GroupViewHolder) convertView.getTag();
        }
        // 设置title
        groupHolder.groupTitle.setText(childChapterList.get(groupPosition).getName());

        return convertView;
    }

    @Override
    public View getChildView(int groupPosition, int childPosition, boolean isLastChild, View convertView, ViewGroup parent) {
        ChapterExpandableLowAdapter.ChildViewHolder childHolder;
        if (convertView == null) {
            convertView = LayoutInflater.from(parent.getContext()).inflate(R.layout.adapter_expandable_grandson_view, parent, false);
            childHolder = new ChapterExpandableLowAdapter.ChildViewHolder();
            childHolder.childTitle = convertView.findViewById(R.id.adapter_grandson_title);
            convertView.setTag(childHolder);
        } else {
            childHolder = (ChapterExpandableLowAdapter.ChildViewHolder) convertView.getTag();
        }
        if (childChapterList.get(groupPosition).getThird() != null && childChapterList.get(groupPosition).getThird().size() > 0) {
            childHolder.childTitle.setText(childChapterList.get(groupPosition).getThird().get(childPosition).getName());
        }
        return convertView;
    }

使用

当我们完成了上面的步骤之后,最后就是在Activity中的使用了。使用方法超级简单

给ExpandableListView设置Adapter就可以了


	@BindView(R.id.chapter_elv)
    ExpandableListView chapterExpandable;
    private ChapterExpandableAdapter chapterExpandableAdapter;
    
    ...
    
	chapterExpandableAdapter = new ChapterExpandableAdapter(this);
    chapterExpandable.setAdapter(chapterExpandableAdapter);

写在最后

因为是三级嵌套,所以ExpandableListView需要重写一下,重新绘制高度。不然会出现页面展示不全或者不完整的问题。

以上就是Android使用ExpandableListView实现三层嵌套折叠菜单的详细内容,更多关于Android ExpandableListView三层嵌套折叠菜单的资料请关注编程网其它相关文章!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-移动开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯