Android实现多级树形选择列表

项目中有多个地方要用到多级列表的菜单,最开始我用的是ExpandableListView,但问题是ExpandableListView只支持两级列表,于是我就用ExpandableListView嵌套ExpandableListView,但非常麻烦,而且关键的是具体分几级是不确定的,也就是可能一级,可能多级,这要是五六级嵌套ListView,于是我就去学习鸿洋大神之前写的一篇关于实现Android多级树形列表的文章,实现很巧妙,使用一个ListView就可以实现多级列表效果,我做了部分修改,功能顺利实现。

1.定义节点实体类:

package com.xiaoyehai.multileveltreelist.treelist;

import java.util.ArrayList;

import java.util.List;

/**

* 节点实体类

* Created by xiaoyehai on 2018/7/11 0011.

*/

public class Node<T> {

/**

* 当前节点id

*/

private String id;

/**

* 父节点id

*/

private String pid;

/**

* 节点数据实体类

*/

private T data;

/**

* 设置开启 关闭的图片

*/

public int iconExpand = -1, iconNoExpand = -1;

/**

* 节点名称

*/

private String name;

/**

* 当前的级别

*/

private int level;

/**

* 是否展开

*/

private boolean isExpand = false;

private int icon = -1;

/**

* 下一级的子Node

*/

private List<Node> children = new ArrayList<>();

/**

* 父Node

*/

private Node parent;

/**

* 是否被checked选中

*/

private boolean isChecked;

public Node() {

}

public Node(String id, String pid, String name) {

this.id = id;

this.pid = pid;

this.name = name;

}

public Node(String id, String pid, T data, String name) {

this.id = id;

this.pid = pid;

this.data = data;

this.name = name;

}

/**

* 是否为根节点

*

* @return

*/

public boolean isRootNode() {

return parent == null;

}

/**

* 判断父节点是否展开

*

* @return

*/

public boolean isParentExpand() {

if (parent == null)

return false;

return parent.isExpand();

}

/**

* 是否是叶子节点

*

* @return

*/

public boolean isLeaf() {

return children.size() == 0;

}

/**

* 获取当前的级别level

*/

public int getLevel() {

return parent == null ? 0 : parent.getLevel() + 1;

}

/**

* 设置展开

*

* @param isExpand

*/

public void setExpand(boolean isExpand) {

this.isExpand = isExpand;

if (!isExpand) {

for (Node node : children) {

node.setExpand(isExpand);

}

}

}

public String getId() {

return id;

}

public void setId(String id) {

this.id = id;

}

public String getPid() {

return pid;

}

public void setPid(String pid) {

this.pid = pid;

}

public T getData() {

return data;

}

public void setData(T data) {

this.data = data;

}

public int getIconExpand() {

return iconExpand;

}

public void setIconExpand(int iconExpand) {

this.iconExpand = iconExpand;

}

public int getIconNoExpand() {

return iconNoExpand;

}

public void setIconNoExpand(int iconNoExpand) {

this.iconNoExpand = iconNoExpand;

}

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public void setLevel(int level) {

this.level = level;

}

public boolean isExpand() {

return isExpand;

}

public int getIcon() {

return icon;

}

public void setIcon(int icon) {

this.icon = icon;

}

public List<Node> getChildren() {

return children;

}

public void setChildren(List<Node> children) {

this.children = children;

}

public Node getParent() {

return parent;

}

public void setParent(Node parent) {

this.parent = parent;

}

public boolean isChecked() {

return isChecked;

}

public void setChecked(boolean checked) {

isChecked = checked;

}

}

2.定义每个节点数据的实体类

因为项目中多个地方用到树形菜单,而且数据都不一样,每个节点数据都比较复杂,所以我单独封装出一个类,要是数据和简单,这步可以不用,直接用Node类。

例如:

/**

* 每个节点的具体数据

* Created by xiaoyehai on 2018/7/11 0011.

*/

public class NodeData {

private String name;

public NodeData() {

}

public NodeData(String name) {

this.name = name;

}

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

}

3.TreeHelper

package com.xiaoyehai.multileveltreelist.treelist;

import java.util.ArrayList;

import java.util.List;

/**

* Created by xiaoyehai on 2018/7/11 0011.

*/

public class TreeHelper {

/**

* 传入node 返回排序后的Node

* 拿到用户传入的数据,转化为List<Node>以及设置Node间关系,然后根节点,从根往下遍历进行排序;

*

* @param datas

* @param defaultExpandLevel 默认显示

* @return

* @throws IllegalArgumentException

* @throws IllegalAccessException

*/

public static List<Node> getSortedNodes(List<Node> datas, int defaultExpandLevel) {

List<Node> result = new ArrayList<Node>();

// 设置Node间父子关系

List<Node> nodes = convetData2Node(datas);

// 拿到根节点

List<Node> rootNodes = getRootNodes(nodes);

// 排序以及设置Node间关系

for (Node node : rootNodes) {

addNode(result, node, defaultExpandLevel, 1);

}

return result;

}

/**

* 过滤出所有可见的Node

* 过滤Node的代码很简单,遍历所有的Node,只要是根节点或者父节点是展开状态就添加返回

*

* @param nodes

* @return

*/

public static List<Node> filterVisibleNode(List<Node> nodes) {

List<Node> result = new ArrayList<Node>();

for (Node node : nodes) {

// 如果为跟节点,或者上层目录为展开状态

if (node.isRootNode() || node.isParentExpand()) {

setNodeIcon(node);

result.add(node);

}

}

return result;

}

/**

* 将我们的数据转化为树的节点

* 设置Node间,父子关系;让每两个节点都比较一次,即可设置其中的关系

*/

private static List<Node> convetData2Node(List<Node> nodes) {

for (int i = 0; i < nodes.size(); i++) {

Node n = nodes.get(i);

for (int j = i + 1; j < nodes.size(); j++) {

Node m = nodes.get(j);

if (m.getPid() instanceof String) {

if (m.getPid().equals(n.getId())) { //n时m的父节点

n.getChildren().add(m);

m.setParent(n);

} else if (m.getId().equals(n.getPid())) { //m时n的父节点

m.getChildren().add(n);

n.setParent(m);

}

} else {

if (m.getPid() == n.getId()) {

n.getChildren().add(m);

m.setParent(n);

} else if (m.getId() == n.getPid()) {

m.getChildren().add(n);

n.setParent(m);

}

}

}

}

return nodes;

}

/**

* 获得根节点

*

* @param nodes

* @return

*/

private static List<Node> getRootNodes(List<Node> nodes) {

List<Node> root = new ArrayList<Node>();

for (Node node : nodes) {

if (node.isRootNode())

root.add(node);

}

return root;

}

/**

* 把一个节点上的所有的内容都挂上去

* 通过递归的方式,把一个节点上的所有的子节点等都按顺序放入

*/

private static <T> void addNode(List<Node> nodes, Node<T> node, int defaultExpandLeval, int currentLevel) {

nodes.add(node);

if (defaultExpandLeval >= currentLevel) {

node.setExpand(true);

}

if (node.isLeaf())

return;

for (int i = 0; i < node.getChildren().size(); i++) {

addNode(nodes, node.getChildren().get(i), defaultExpandLeval, currentLevel + 1);

}

}

/**

* 设置节点的图标

*

* @param node

*/

private static void setNodeIcon(Node node) {

if (node.getChildren().size() > 0 && node.isExpand()) {

node.setIcon(node.iconExpand);

} else if (node.getChildren().size() > 0 && !node.isExpand()) {

node.setIcon(node.iconNoExpand);

} else {

node.setIcon(-1);

}

}

}

4.TreeListViewAdapter

对于ListView的适配器,需要继承自TreeListViewAdapter,如

package com.xiaoyehai.multileveltreelist.treelist;

import android.content.Context;

import android.view.LayoutInflater;

import android.view.View;

import android.view.ViewGroup;

import android.widget.AdapterView;

import android.widget.BaseAdapter;

import android.widget.ListView;

import java.util.ArrayList;

import java.util.List;

/**

* Created by xiaoyehai on 2018/7/11 0011.

*/

public abstract class TreeListViewAdapter extends BaseAdapter {

protected Context mContext;

/**

* 默认不展开

*/

private int defaultExpandLevel = 0;

/**

* 展开与关闭的图片

*/

private int iconExpand = -1, iconNoExpand = -1;

/**

* 存储所有的Node

*/

protected List<Node> mAllNodes = new ArrayList<>();

protected LayoutInflater mInflater;

/**

* 存储所有可见的Node

*/

protected List<Node> mNodes = new ArrayList<>();

/**

* 点击的回调接口

*/

private OnTreeNodeClickListener onTreeNodeClickListener;

public void setOnTreeNodeClickListener(OnTreeNodeClickListener onTreeNodeClickListener) {

this.onTreeNodeClickListener = onTreeNodeClickListener;

}

public TreeListViewAdapter(ListView listView, Context context, List<Node> datas, int defaultExpandLevel, int iconExpand, int iconNoExpand) {

this.mContext = context;

this.defaultExpandLevel = defaultExpandLevel;

this.iconExpand = iconExpand;

this.iconNoExpand = iconNoExpand;

for (Node node : datas) {

node.getChildren().clear();

node.setIconExpand(iconExpand);

node.setIconNoExpand(iconNoExpand);

}

/**

* 对所有的Node进行排序

*/

mAllNodes = TreeHelper.getSortedNodes(datas, defaultExpandLevel);

/**

* 过滤出可见的Node

*/

mNodes = TreeHelper.filterVisibleNode(mAllNodes);

mInflater = LayoutInflater.from(context);

/**

* 设置节点点击时,可以展开以及关闭;并且将ItemClick事件继续往外公布

*/

listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {

@Override

public void onItemClick(AdapterView<?> parent, View view, int position, long id) {

expandOrCollapse(position);

if (onTreeNodeClickListener != null) {

onTreeNodeClickListener.onClick(mNodes.get(position), position);

}

}

});

}

/**

* @param listView

* @param context

* @param datas

* @param defaultExpandLevel 默认展开几级树

*/

public TreeListViewAdapter(ListView listView, Context context, List<Node> datas, int defaultExpandLevel) {

this(listView, context, datas, defaultExpandLevel, -1, -1);

}

/**

* 相应ListView的点击事件 展开或关闭某节点

*

* @param position

*/

public void expandOrCollapse(int position) {

Node n = mNodes.get(position);

if (n != null) {// 排除传入参数错误异常

if (!n.isLeaf()) {

n.setExpand(!n.isExpand());

//获取所有可见的Node

mNodes = TreeHelper.filterVisibleNode(mAllNodes);

notifyDataSetChanged();// 刷新视图

}

}

}

@Override

public int getCount() {

return mNodes.size();

}

@Override

public Object getItem(int position) {

return mNodes.get(position);

}

@Override

public long getItemId(int position) {

return position;

}

@Override

public View getView(int position, View convertView, ViewGroup parent) {

Node node = mNodes.get(position);

convertView = getConvertView(node, position, convertView, parent);

// 设置内边距

convertView.setPadding(node.getLevel() * 50, 12, 12, 12);

return convertView;

}

/**

* 获取排序后所有节点

*

* @return

*/

public List<Node> getAllNodes() {

if (mAllNodes == null)

mAllNodes = new ArrayList<Node>();

return mAllNodes;

}

/**

* 获取所有选中节点

*

* @return

*/

public List<Node> getSelectedNode() {

List<Node> checks = new ArrayList<Node>();

for (int i = 0; i < mAllNodes.size(); i++) {

Node n = mAllNodes.get(i);

if (n.isChecked()) {

checks.add(n);

}

}

return checks;

}

/**

* 设置多选

*

* @param node

* @param checked

*/

protected void setChecked(final Node node, boolean checked) {

node.setChecked(checked);

setChildChecked(node, checked);

if (node.getParent() != null)

setNodeParentChecked(node.getParent(), checked);

notifyDataSetChanged();

}

/**

* 设置是否选中

*

* @param node

* @param checked

*/

public <T> void setChildChecked(Node<T> node, boolean checked) {

if (!node.isLeaf()) {

node.setChecked(checked);

for (Node childrenNode : node.getChildren()) {

setChildChecked(childrenNode, checked);

}

} else {

node.setChecked(checked);

}

}

private void setNodeParentChecked(Node node, boolean checked) {

if (checked) {

node.setChecked(checked);

if (node.getParent() != null)

setNodeParentChecked(node.getParent(), checked);

} else {

List<Node> childrens = node.getChildren();

boolean isChecked = false;

for (Node children : childrens) {

if (children.isChecked()) {

isChecked = true;

}

}

//如果所有自节点都没有被选中 父节点也不选中

if (!isChecked) {

node.setChecked(checked);

}

if (node.getParent() != null)

setNodeParentChecked(node.getParent(), checked);

}

}

/**

* 清除掉之前数据并刷新 重新添加

*

* @param mlists

* @param defaultExpandLevel 默认展开几级列表

*/

public void addDataAll(List<Node> mlists, int defaultExpandLevel) {

mAllNodes.clear();

addData(-1, mlists, defaultExpandLevel);

}

/**

* 在指定位置添加数据并刷新 可指定刷新后显示层级

*

* @param index

* @param mlists

* @param defaultExpandLevel 默认展开几级列表

*/

public void addData(int index, List<Node> mlists, int defaultExpandLevel) {

this.defaultExpandLevel = defaultExpandLevel;

notifyData(index, mlists);

}

/**

* 在指定位置添加数据并刷新

*

* @param index

* @param mlists

*/

public void addData(int index, List<Node> mlists) {

notifyData(index, mlists);

}

/**

* 添加数据并刷新

*

* @param mlists

*/

public void addData(List<Node> mlists) {

addData(mlists, defaultExpandLevel);

}

/**

* 添加数据并刷新 可指定刷新后显示层级

*

* @param mlists

* @param defaultExpandLevel

*/

public void addData(List<Node> mlists, int defaultExpandLevel) {

this.defaultExpandLevel = defaultExpandLevel;

notifyData(-1, mlists);

}

/**

* 添加数据并刷新

*

* @param node

*/

public void addData(Node node) {

addData(node, defaultExpandLevel);

}

/**

* 添加数据并刷新 可指定刷新后显示层级

*

* @param node

* @param defaultExpandLevel

*/

public void addData(Node node, int defaultExpandLevel) {

List<Node> nodes = new ArrayList<>();

nodes.add(node);

this.defaultExpandLevel = defaultExpandLevel;

notifyData(-1, nodes);

}

/**

* 刷新数据

*

* @param index

* @param mListNodes

*/

public void notifyData(int index, List<Node> mListNodes) {

for (int i = 0; i < mListNodes.size(); i++) {

Node node = mListNodes.get(i);

node.getChildren().clear();

node.iconExpand = iconExpand;

node.iconNoExpand = iconNoExpand;

}

for (int i = 0; i < mAllNodes.size(); i++) {

Node node = mAllNodes.get(i);

node.getChildren().clear();

//node.isNewAdd = false;

}

if (index != -1) {

mAllNodes.addAll(index, mListNodes);

} else {

mAllNodes.addAll(mListNodes);

}

/**

* 对所有的Node进行排序

*/

mAllNodes = TreeHelper.getSortedNodes(mAllNodes, defaultExpandLevel);

/**

* 过滤出可见的Node

*/

mNodes = TreeHelper.filterVisibleNode(mAllNodes);

//刷新数据

notifyDataSetChanged();

}

public abstract View getConvertView(Node node, int position, View convertView, ViewGroup parent);

}

5.接口回调:

选中状态改变的回调:

package com.xiaoyehai.multileveltreelist.treelist;

/**

* Created by xiaoyehai on 2018/7/12 0012.

*/

public interface OnTreeNodeCheckedChangeListener {

void onCheckChange(Node node, int position, boolean isChecked);

}

条目点击的回调:

package com.xiaoyehai.multileveltreelist.treelist;

/**

* Created by xiaoyehai on 2018-07-12.

*/

public interface OnTreeNodeClickListener {

void onClick(Node node, int position);

}

6.使用:

布局文件:

<ListView

android:id="@+id/listview"

android:layout_width="match_parent"

android:layout_height="match_parent"></ListView>

Activity:

public class ListViewActivity extends AppCompatActivity {

private ListView mListView;

private List<Node> dataList = new ArrayList<>();

private ListViewAdapter mAdapter;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_list_view);

mListView = (ListView) findViewById(R.id.listview);

initData();

//第一个参数 ListView

//第二个参数 上下文

//第三个参数 数据集

//第四个参数 默认展开层级数 0为不展开

//第五个参数 展开的图标

//第六个参数 闭合的图标

mAdapter = new ListViewAdapter(mListView, this, dataList,

0, R.drawable.zoomout_yzs, R.drawable.zoomin_yzs);

mListView.setAdapter(mAdapter);

//获取所有节点

final List<Node> allNodes = mAdapter.getAllNodes();

for (Node allNode : allNodes) {

//Log.e("xyh", "onCreate: " + allNode.getName());

}

//选中状态监听

mAdapter.setCheckedChangeListener(new OnTreeNodeCheckedChangeListener() {

@Override

public void onCheckChange(Node node, int position, boolean isChecked) {

//获取所有选中节点

List<Node> selectedNode = mAdapter.getSelectedNode();

for (Node n : selectedNode) {

Log.e("xyh", "onCheckChange: " + n.getName());

}

}

});

}

/**

* 模拟数据,实际开发中对返回的json数据进行封装

*/

private void initData() {

//根节点

Node<NodeData> node = new Node<>("0", "-1", "根节点1");

dataList.add(node);

dataList.add(new Node<>("1", "-1", "根节点2"));

dataList.add(new Node<>("2", "-1", "根节点3"));

//根节点1的二级节点

dataList.add(new Node<>("3", "0", "二级节点"));

dataList.add(new Node<>("4", "0", "二级节点"));

dataList.add(new Node<>("5", "0", "二级节点"));

//根节点2的二级节点

dataList.add(new Node<>("6", "1", "二级节点"));

dataList.add(new Node<>("7", "1", "二级节点"));

dataList.add(new Node<>("8", "1", "二级节点"));

//根节点3的二级节点

dataList.add(new Node<>("9", "2", "二级节点"));

dataList.add(new Node<>("10", "2", "二级节点"));

dataList.add(new Node<>("11", "2", "二级节点"));

//三级节点

dataList.add(new Node<>("12", "3", "三级节点"));

dataList.add(new Node<>("13", "3", "三级节点"));

dataList.add(new Node<>("14", "3", "三级节点"));

dataList.add(new Node<>("15", "4", "三级节点"));

dataList.add(new Node<>("16", "4", "三级节点"));

dataList.add(new Node<>("17", "4", "三级节点"));

dataList.add(new Node<>("18", "5", "三级节点"));

dataList.add(new Node<>("19", "5", "三级节点"));

dataList.add(new Node<>("20", "5", "三级节点"));

//四级节点

dataList.add(new Node<>("21", "12", "四级节点"));

//...

//可以有无线多层级

}

}

adapter:

package com.xiaoyehai.multileveltreelist.adapter;

import android.content.Context;

import android.view.View;

import android.view.ViewGroup;

import android.widget.CheckBox;

import android.widget.ImageView;

import android.widget.ListView;

import android.widget.TextView;

import com.xiaoyehai.multileveltreelist.R;

import com.xiaoyehai.multileveltreelist.treelist.OnTreeNodeCheckedChangeListener;

import com.xiaoyehai.multileveltreelist.treelist.TreeListViewAdapter;

import com.xiaoyehai.multileveltreelist.treelist.Node;

import java.util.List;

/**

* Created by xiaoyehai on 2018/7/12 0012.

*/

public class ListViewAdapter extends TreeListViewAdapter {

private OnTreeNodeCheckedChangeListener checkedChangeListener;

public void setCheckedChangeListener(OnTreeNodeCheckedChangeListener checkedChangeListener) {

this.checkedChangeListener = checkedChangeListener;

}

public ListViewAdapter(ListView listView, Context context, List<Node> datas, int defaultExpandLevel, int iconExpand, int iconNoExpand) {

super(listView, context, datas, defaultExpandLevel, iconExpand, iconNoExpand);

}

@Override

public View getConvertView(final Node node, final int position, View convertView, ViewGroup parent) {

final ViewHolder holder;

if (convertView == null) {

convertView = View.inflate(mContext, R.layout.item, null);

holder = new ViewHolder(convertView);

convertView.setTag(holder);

} else {

holder = (ViewHolder) convertView.getTag();

}

holder.tvName.setText(node.getName());

if (node.getIcon() == -1) {

holder.ivExpand.setVisibility(View.INVISIBLE);

} else {

holder.ivExpand.setVisibility(View.VISIBLE);

holder.ivExpand.setImageResource(node.getIcon());

}

holder.checkBox.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

setChecked(node, holder.checkBox.isChecked());

if (checkedChangeListener != null) {

checkedChangeListener.onCheckChange(node, position,holder.checkBox.isChecked());

}

}

});

if (node.isChecked()) {

holder.checkBox.setChecked(true);

} else {

holder.checkBox.setChecked(false);

}

return convertView;

}

static class ViewHolder {

private CheckBox checkBox;

private TextView tvName;

private ImageView ivExpand;

public ViewHolder(View convertView) {

checkBox = convertView.findViewById(R.id.cb);

tvName = convertView.findViewById(R.id.tv_name);

ivExpand = convertView.findViewById(R.id.iv_expand);

}

}

}

也可以用RecycleView实现,在我的项目里面都有。

[项目地址]:MultilevelTreeList

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

以上是 Android实现多级树形选择列表 的全部内容, 来源链接: utcz.com/p/241821.html

回到顶部