展会信息港展会大全

Android 多级树形菜单
来源:互联网   发布日期:2015-10-03 13:18:46   浏览:2817次  

导读:在Android里要实现树形菜单,都是用ExpandableList(也有高手自己继承ListView或者LinearLayout来做),但是ExpandableList一般只能实现2级树形菜单......本文也依然使用ExpandableList,但是要实现的是3级树形菜单......

在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不好解决,而且定义三维数组的时候也要倍加小心......所以尽量把数据化简来使用二级树形菜单。

赞助本站

人工智能实验室

相关热词: 多级树形菜单

相关内容
AiLab云推荐
推荐内容
展开

热门栏目HotCates

Copyright © 2010-2024 AiLab Team. 人工智能实验室 版权所有    关于我们 | 联系我们 | 广告服务 | 公司动态 | 免责声明 | 隐私条款 | 工作机会 | 展会港