Android 多级树形菜单
在Android里要实现树形菜单,都是用ExpandableList(也有高手自己继承ListView或者LinearLayout来做),但是ExpandableList一般只能实现2级树形菜单......本文也依然使用ExpandableList,但是要实现的是3级树形菜单。本文程序运行效果图:
当用BaseExpandableListAdapter来实现二级树形菜单时,父项(getGroupView())和子项(getChildView())都是使用TextView。当要实现三级树形菜单时,子项(getChildView())就必须使用ExpandableList了.......另外还要定义结构体来方便调用三级树形的数据,二级树形菜单可以用如下:
- static public class TreeNode{
- Object parent;
- List<Object> childs=new ArrayList<Object>();
- }
三级树形菜单可以用如下,子项是二级树形菜单的结构体:
- static public class SuperTreeNode {
- Object parent;
- //二级树形菜单的结构体
- List<TreeViewAdapter.TreeNode> childs = new ArrayList<TreeViewAdapter.TreeNode>();
- }
实现三级树形菜单有两点要注意的:
1、第二级也是个树形菜单,因此必须在第二级项目展开/回收时设置足够的空间来完全显示二级树形菜单;
2、在实现三级树形菜单时,发现菜单的方法都是用不了(如OnChildClickListener、OnGroupClickListener等),因此要获得选中的数据就必须在外部定义好回调函数,然后在第二级生成二级树形菜单时回调这个外部函数。
PS:本文在解决No.2关键点的时候,只能取得第三级选中的序号.....而第一,第二级依然无法获取其序号。
main.xml源码如下:
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:orientation="vertical" android:layout_width="fill_parent"
- android:layout_height="fill_parent">
- <LinearLayout android:id="@+id/LinearLayout01"
- android:layout_width="wrap_content" android:layout_height="wrap_content">
- <Button android:layout_height="wrap_content" android:text="两层结构"
- android:layout_width="160dip" android:id="@+id/btnNormal"></Button>
- <Button android:layout_height="wrap_content" android:text="三层结构"
- android:layout_width="160dip" android:id="@+id/btnSuper"></Button>
- </LinearLayout>
- <ExpandableListView android:id="@+id/ExpandableListView01"
- android:layout_width="fill_parent" android:layout_height="fill_parent"></ExpandableListView>
- </LinearLayout>
testExpandableList.java是主类,调用其他工具类,源码如下:
- package com.testExpandableList;
- import java.util.List;
- import android.app.Activity;
- import android.os.Bundle;
- import android.util.Log;
- import android.view.View;
- import android.widget.Button;
- import android.widget.ExpandableListView;
- import android.widget.ExpandableListView.OnChildClickListener;
- import android.widget.Toast;
- public class testExpandableList extends Activity {
- /** Called when the activity is first created. */
- ExpandableListView expandableList;
- TreeViewAdapter adapter;
- SuperTreeViewAdapter superAdapter;
- Button btnNormal,btnSuper;
- // Sample data set. children[i] contains the children (String[]) for groups[i].
- public String[] groups = { "xxxx好友", "xxxx同学", "xxxxx女人"};
- public String[][] child= {
- { "A君", "B君", "C君", "D君" },
- { "同学甲", "同学乙", "同学丙"},
- { "御姐", "萝莉" }
- };
- public String[] parent = { "xxxx好友", "xxxx同学"};
- public String[][][] child_grandson= {
- {{"A君"},
- {"AA","AAA"}},
- {{"B君"},
- {"BBB","BBBB","BBBBB"}},
- {{"C君"},
- {"CCC","CCCC"}},
- {{"D君"},
- {"DDD","DDDD","DDDDD"}},
- };
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- this.setTitle("ExpandableListView练习----hellogv");
- btnNormal=(Button)this.findViewById(R.id.btnNormal);
- btnNormal.setOnClickListener(new ClickEvent());
- btnSuper=(Button)this.findViewById(R.id.btnSuper);
- btnSuper.setOnClickListener(new ClickEvent());
- adapter=new TreeViewAdapter(this,TreeViewAdapter.PaddingLeft>>1);
- superAdapter=new SuperTreeViewAdapter(this,stvClickEvent);
- expandableList=(ExpandableListView) testExpandableList.this.findViewById(R.id.ExpandableListView01);
- }
- class ClickEvent implements View.OnClickListener{
- @Override
- public void onClick(View v) {
- adapter.RemoveAll();
- adapter.notifyDataSetChanged();
- superAdapter.RemoveAll();
- superAdapter.notifyDataSetChanged();
- if(v==btnNormal)
- {
- List<TreeViewAdapter.TreeNode> treeNode = adapter.GetTreeNode();
- for(int i=0;i<groups.length;i++)
- {
- TreeViewAdapter.TreeNode node=new TreeViewAdapter.TreeNode();
- node.parent=groups[i];
- for(int ii=0;ii<child[i].length;ii++)
- {
- node.childs.add(child[i][ii]);
- }
- treeNode.add(node);
- }
- adapter.UpdateTreeNode(treeNode);
- expandableList.setAdapter(adapter);
- expandableList.setOnChildClickListener(new OnChildClickListener(){
- @Override
- public boolean onChildClick(ExpandableListView arg0, View arg1,
- int parent, int children, long arg4) {
- String str="parent id:"+String.valueOf(parent)+",children id:"+String.valueOf(children);
- Toast.makeText(testExpandableList.this, str, 300).show();
- return false;
- }
- });
- }
- else if(v==btnSuper){
- List<SuperTreeViewAdapter.SuperTreeNode> superTreeNode = superAdapter.GetTreeNode();
- for(int i=0;i<parent.length;i++)//第一层
- {
- SuperTreeViewAdapter.SuperTreeNode superNode=new SuperTreeViewAdapter.SuperTreeNode();
- superNode.parent=parent[i];
- //第二层
- for(int ii=0;ii<child_grandson.length;ii++)
- {
- TreeViewAdapter.TreeNode node=new TreeViewAdapter.TreeNode();
- node.parent=child_grandson[ii][0][0];//第二级菜单的标题
- for(int iii=0;iii<child_grandson[ii][1].length;iii++)//第三级菜单
- {
- node.childs.add(child_grandson[ii][1][iii]);
- }
- superNode.childs.add(node);
- }
- superTreeNode.add(superNode);
- }
- superAdapter.UpdateTreeNode(superTreeNode);
- expandableList.setAdapter(superAdapter);
- }
- }
- }
- /**
- * 三级树形菜单的事件不再可用,本函数由三级树形菜单的子项(二级菜单)进行回调
- */
- OnChildClickListener stvClickEvent=new OnChildClickListener(){
- @Override
- public boolean onChildClick(ExpandableListView parent,
- View v, int groupPosition, int childPosition,
- long id) {
- String str="parent id:"+String.valueOf(groupPosition)+",children id:"+String.valueOf(childPosition);
- Toast.makeText(testExpandableList.this, str, 300).show();
- return false;
- }
- };
- }
TreeViewAdapter.java是实现二级树形菜单的工具类,源码如下:
- package com.testExpandableList;
- import java.util.ArrayList;
- import java.util.List;
- import android.content.Context;
- import android.util.Log;
- import android.view.Gravity;
- import android.view.View;
- import android.view.ViewGroup;
- import android.widget.AbsListView;
- import android.widget.BaseExpandableListAdapter;
- import android.widget.TextView;
- public class TreeViewAdapter extends BaseExpandableListAdapter{
- public static final int ItemHeight=48;//每项的高度
- public static final int PaddingLeft=36;//每项的高度
- private int myPaddingLeft=0;//如果是由SuperTreeView调用,则作为子项需要往右移
- static public class TreeNode{
- Object parent;
- List<Object> childs=new ArrayList<Object>();
- }
- List<TreeNode> treeNodes = new ArrayList<TreeNode>();
- Context parentContext;
- public TreeViewAdapter(Context view,int myPaddingLeft)
- {
- parentContext=view;
- this.myPaddingLeft=myPaddingLeft;
- }
- public List<TreeNode> GetTreeNode()
- {
- return treeNodes;
- }
- public void UpdateTreeNode(List<TreeNode> nodes)
- {
- treeNodes=nodes;
- }
- public void RemoveAll()
- {
- treeNodes.clear();
- }
- public Object getChild(int groupPosition, int childPosition) {
- return treeNodes.get(groupPosition).childs.get(childPosition);
- }
- public int getChildrenCount(int groupPosition) {
- return treeNodes.get(groupPosition).childs.size();
- }
- static public TextView getTextView(Context context) {
- AbsListView.LayoutParams lp = new AbsListView.LayoutParams(
- ViewGroup.LayoutParams.FILL_PARENT, ItemHeight);
- TextView textView = new TextView(context);
- textView.setLayoutParams(lp);
- textView.setGravity(Gravity.CENTER_VERTICAL | Gravity.LEFT);
- return textView;
- }
- public View getChildView(int groupPosition, int childPosition,
- boolean isLastChild, View convertView, ViewGroup parent) {
- TextView textView = getTextView(this.parentContext);
- textView.setText(getChild(groupPosition, childPosition).toString());
- textView.setPadding(myPaddingLeft+PaddingLeft, 0, 0, 0);
- return textView;
- }
- public View getGroupView(int groupPosition, boolean isExpanded,
- View convertView, ViewGroup parent) {
- TextView textView = getTextView(this.parentContext);
- textView.setText(getGroup(groupPosition).toString());
- textView.setPadding(myPaddingLeft+(PaddingLeft>>1), 0, 0, 0);
- return textView;
- }
- public long getChildId(int groupPosition, int childPosition) {
- return childPosition;
- }
- public Object getGroup(int groupPosition) {
- return treeNodes.get(groupPosition).parent;
- }
- public int getGroupCount() {
- return treeNodes.size();
- }
- public long getGroupId(int groupPosition) {
- return groupPosition;
- }
- public boolean isChildSelectable(int groupPosition, int childPosition) {
- return true;
- }
- public boolean hasStableIds() {
- return true;
- }
- }
SuperTreeViewAdapter.java是实现三级树形菜单的工具类,会用到TreeViewAdapter.java,源码如下:
- package com.testExpandableList;
- import java.util.ArrayList;
- import java.util.List;
- import com.testExpandableList.TreeViewAdapter.TreeNode;
- import android.content.Context;
- import android.view.View;
- import android.view.ViewGroup;
- import android.widget.AbsListView;
- import android.widget.BaseExpandableListAdapter;
- import android.widget.ExpandableListView;
- import android.widget.ExpandableListView.OnChildClickListener;
- import android.widget.ExpandableListView.OnGroupCollapseListener;
- import android.widget.ExpandableListView.OnGroupExpandListener;
- import android.widget.TextView;
- public class SuperTreeViewAdapter extends BaseExpandableListAdapter {
- static public class SuperTreeNode {
- Object parent;
- //二级树形菜单的结构体
- List<TreeViewAdapter.TreeNode> childs = new ArrayList<TreeViewAdapter.TreeNode>();
- }
- private List<SuperTreeNode> superTreeNodes = new ArrayList<SuperTreeNode>();
- private Context parentContext;
- private OnChildClickListener stvClickEvent;//外部回调函数
- public SuperTreeViewAdapter(Context view,OnChildClickListener stvClickEvent) {
- parentContext = view;
- this.stvClickEvent=stvClickEvent;
- }
- public List<SuperTreeNode> GetTreeNode() {
- return superTreeNodes;
- }
- public void UpdateTreeNode(List<SuperTreeNode> node) {
- superTreeNodes = node;
- }
- public void RemoveAll()
- {
- superTreeNodes.clear();
- }
- public Object getChild(int groupPosition, int childPosition) {
- return superTreeNodes.get(groupPosition).childs.get(childPosition);
- }
- public int getChildrenCount(int groupPosition) {
- return superTreeNodes.get(groupPosition).childs.size();
- }
- public ExpandableListView getExpandableListView() {
- AbsListView.LayoutParams lp = new AbsListView.LayoutParams(
- ViewGroup.LayoutParams.FILL_PARENT, TreeViewAdapter.ItemHeight);
- ExpandableListView superTreeView = new ExpandableListView(parentContext);
- superTreeView.setLayoutParams(lp);
- return superTreeView;
- }
- /**
- * 三层树结构中的第二层是一个ExpandableListView
- */
- public View getChildView(int groupPosition, int childPosition,
- boolean isLastChild, View convertView, ViewGroup parent) {
- // 是
- final ExpandableListView treeView = getExpandableListView();
- final TreeViewAdapter treeViewAdapter = new TreeViewAdapter(this.parentContext,0);
- List<TreeNode> tmp = treeViewAdapter.GetTreeNode();//临时变量取得TreeViewAdapter的TreeNode集合,可为空
- final TreeNode treeNode=(TreeNode) getChild(groupPosition, childPosition);
- tmp.add(treeNode);
- treeViewAdapter.UpdateTreeNode(tmp);
- treeView.setAdapter(treeViewAdapter);
- //关键点:取得选中的二级树形菜单的父子节点,结果返回给外部回调函数
- treeView.setOnChildClickListener(this.stvClickEvent);
- /**
- * 关键点:第二级菜单展开时通过取得节点数来设置第三级菜单的大小
- */
- treeView.setOnGroupExpandListener(new OnGroupExpandListener() {
- @Override
- public void onGroupExpand(int groupPosition) {
- AbsListView.LayoutParams lp = new AbsListView.LayoutParams(
- ViewGroup.LayoutParams.FILL_PARENT,
- (treeNode.childs.size()+1)*TreeViewAdapter.ItemHeight + 10);
- treeView.setLayoutParams(lp);
- }
- });
- /**
- * 第二级菜单回收时设置为标准Item大小
- */
- treeView.setOnGroupCollapseListener(new OnGroupCollapseListener() {
- @Override
- public void onGroupCollapse(int groupPosition) {
- AbsListView.LayoutParams lp = new AbsListView.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT,
- TreeViewAdapter.ItemHeight);
- treeView.setLayoutParams(lp);
- }
- });
- treeView.setPadding(TreeViewAdapter.PaddingLeft, 0, 0, 0);
- return treeView;
- }
- /**
- * 三级树结构中的首层是TextView,用于作为title
- */
- public View getGroupView(int groupPosition, boolean isExpanded,
- View convertView, ViewGroup parent) {
- TextView textView = TreeViewAdapter.getTextView(this.parentContext);
- textView.setText(getGroup(groupPosition).toString());
- textView.setPadding(TreeViewAdapter.PaddingLeft, 0, 0, 0);
- return textView;
- }
- public long getChildId(int groupPosition, int childPosition) {
- return childPosition;
- }
- public Object getGroup(int groupPosition) {
- return superTreeNodes.get(groupPosition).parent;
- }
- public int getGroupCount() {
- return superTreeNodes.size();
- }
- public long getGroupId(int groupPosition) {
- return groupPosition;
- }
- public boolean isChildSelectable(int groupPosition, int childPosition) {
- return true;
- }
- public boolean hasStableIds() {
- return true;
- }
- }
总结,使用ExpandableList实现三级树形菜单时有些bug不好解决,而且定义三维数组的时候也要倍加小心......所以尽量把数据化简来使用二级树形菜单。
Android 多级树形菜单的更多相关文章
- 更快实现Android多级树形选择列表
快速实现Android多级树形列表,这个库是在鸿洋多级树形列表demo中修改而来. 解决的问题: 1. 支持ID为int类型和String类型. 2. 支持多级复选框选中,使用只需一行代码. 3. 支 ...
- Android学习笔记之树形菜单的应用...
PS:终于考完试了,总算是解脱了...可以正式上手项目开发了.... 学习内容: 1.掌握如何使用树形菜单... 对知识点进行一下补充...居然忘记了去学习树形菜单...不过在这里补上... Ex ...
- Python 递归返回树形菜单JSON串 <flask>
需求:菜单管理功能(增.删.改),多级树形菜单展示 数据库表设计 create table if not exists Menu( id serial primary key , title ) no ...
- 实用js+css多级树形展开效果导航菜单
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- Vue2 实现树形菜单(多级菜单)功能模块
结构示意图 ├── index.html ├── main.js ├── router │ └── index.js # 路由配置文件 ├── components # 组件目录 │ ├── App. ...
- Android开发之多级下拉列表菜单实现(仿美团,淘宝等)
注:本文转载于:http://blog.csdn.net/minimicall/article/details/39484493 我们在常用的电商或者旅游APP中,例如美团,手机淘宝等等,都能够看的到 ...
- Android:实现仿 美团/淘宝 多级分类菜单效果
本例要实现的是诸如美团/淘宝/百度糯米 多级分类菜单效果.当分类数量许多时能够考虑採用两级分类.而诸如美团这样的表现方式是一个不错的选择. 首先上效果图: 主要代码: 1. PopupWin ...
- cookie实现刷新不变化树形菜单
通过设置cookie来保存树形菜单的状态,在页面加载时重新读取cookie来设置菜单. 菜单的HTML结构: <div class="treemenu"> <ul ...
- jquery树形菜单
转自:http://keleyi.com/dev/3068696139522ae4.htm 代码: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1 ...
随机推荐
- CAT XQX ---- 增删改查架构说明 1
View 层 -- 以国家为例 1. 显示 数据库的 table 页面效果 对应代码: <table id="dg" title="国家信息" cla ...
- bzoj1193
#include<cstdio> #include<cstdlib> #include<algorithm> #include<queue> using ...
- Chapter9:顺序容器
现代C++程序应该使用标准库容器,而不是更原始的数据结构,例如内置数组. 新标准库容器的性能几乎肯定与最精心优化过的同类数据结构一样好. 当我们用一个对象来初始化容器时,或将一个对象插入到容器中时,实 ...
- scala 连接 mysql
code: import java.sql.{ResultSet, DriverManager} import com.mysql.jdbc.Connection object hoursAvg { ...
- hadoop的ganglia数据监控
如果我们想知道当前运行的hadoop集群的状态,可以通过hadoop的客户端和web页面来获得,但是如果我们想知道当前集群的繁忙程度,如读写次数,这些工具就办不到了.幸运的是hadoop提供了一种ga ...
- 挖坟之Spring.NET IOC容器初始化
因查找ht项目中一个久未解决spring内部异常,翻了一段时间源码.以此文总结springIOC,容器初始化过程. 语言背景是C#.网上有一些基于java的spring源码分析文档,大而乱,乱而不全, ...
- 分布式文件系统--GFS
分布式文件系统 分布式文件系统:当数据集的大小超过一台独立物理计算机的存储能力时,就有必要对它进行分区(partition)并存储到若干台单独的计算机上.管理网络中夸多台计算机存储的文件系统.这种系统 ...
- something: 重构、正则、vim -- clwu
项目需要做一个db table 操作的小工具. 从phpMyAdmin上拷贝了一些代码过来修改,但我有没有足够的时间把所有拷贝过来的代码都重构修改和测试完,于是希望后面接手的同事在需要修改这些代码时能 ...
- jQuery和js如何判断checkbox是否选中
jquery: <div id="divId" class="divTable"><div class="tableBody&quo ...
- 提升c++builder 代码输入流畅度的配置
提高c++builder 代码输入流畅度 1.输入指针的函数名后,识别函数参数移动光标到括弧内,此功能太慢,有明显延迟,建议关闭.关闭以后,输入函数名不会自动添加(),需要自己手动输入括弧了,不过速度 ...