Android UI 之实现多级列表TreeView
所谓TreeView就是在Windows中常见的多级列表树,在Android中系统只默认提供了ListView和ExpandableListView两种列表,最多只支持到二级列表的实现,所以如果想要实现三级和更多层次的列表,就需要我们自己来做一些处理了。
其实这个效果很久以前就有人想办法实现了,但是实现的效果有一些问题,我的实现思路主要也是来自于网络,但是在其基础上修正了逻辑上的一些错误,做了一些优化。
先来看一下效果:
然后大体说一下思路:
其实这里实现的多级列表只是一个视觉效果,我们看到的分级效果是由于每行的缩进不同造成的。比如在上面的效果中,山东省和广东省是级别最高的层次,山东省下的青岛市作为山东省的子项,我们增加他的左缩进,这样看起来就有了层次感了。其他的层次也是同理。
也就是说,我们只用了一个ListView,工作的重点就在于不断变化ListView显示的数据,根据用户的操作,将数据修改为用户想要看到的数据内容,并根据每个数据项的不同,在显示效果上做不同的缩进处理,最终呈现出一个TreeView的效果。
具体的实现思路参考下面的项目结构和具体代码:
Element.Java:
- package com.example.androidtreeviewdemo.treeview;
- /**
- * Element类
- * @author carrey
- *
- */
- public class Element {
- /** 文字内容 */
- private String contentText;
- /** 在tree中的层级 */
- private int level;
- /** 元素的id */
- private int id;
- /** 父元素的id */
- private int parendId;
- /** 是否有子元素 */
- private boolean hasChildren;
- /** item是否展开 */
- private boolean isExpanded;
- /** 表示该节点没有父元素,也就是level为0的节点 */
- public static final int NO_PARENT = -1;
- /** 表示该元素位于最顶层的层级 */
- public static final int TOP_LEVEL = 0;
- public Element(String contentText, int level, int id, int parendId,
- boolean hasChildren, boolean isExpanded) {
- super();
- this.contentText = contentText;
- this.level = level;
- this.id = id;
- this.parendId = parendId;
- this.hasChildren = hasChildren;
- this.isExpanded = isExpanded;
- }
- public boolean isExpanded() {
- return isExpanded;
- }
- public void setExpanded(boolean isExpanded) {
- this.isExpanded = isExpanded;
- }
- public String getContentText() {
- return contentText;
- }
- public void setContentText(String contentText) {
- this.contentText = contentText;
- }
- public int getLevel() {
- return level;
- }
- public void setLevel(int level) {
- this.level = level;
- }
- public int getId() {
- return id;
- }
- public void setId(int id) {
- this.id = id;
- }
- public int getParendId() {
- return parendId;
- }
- public void setParendId(int parendId) {
- this.parendId = parendId;
- }
- public boolean isHasChildren() {
- return hasChildren;
- }
- public void setHasChildren(boolean hasChildren) {
- this.hasChildren = hasChildren;
- }
- }
TreeViewAdapter.java:
- package com.example.androidtreeviewdemo.treeview;
- import java.util.ArrayList;
- import com.example.androidtreeviewdemo.R;
- import android.view.LayoutInflater;
- import android.view.View;
- import android.view.ViewGroup;
- import android.widget.BaseAdapter;
- import android.widget.ImageView;
- import android.widget.TextView;
- /**
- * TreeViewAdapter
- * @author carrey
- *
- */
- public class TreeViewAdapter extends BaseAdapter {
- /** 元素数据源 */
- private ArrayList<Element> elementsData;
- /** 树中元素 */
- private ArrayList<Element> elements;
- /** LayoutInflater */
- private LayoutInflater inflater;
- /** item的行首缩进基数 */
- private int indentionBase;
- public TreeViewAdapter(ArrayList<Element> elements, ArrayList<Element> elementsData, LayoutInflater inflater) {
- this.elements = elements;
- this.elementsData = elementsData;
- this.inflater = inflater;
- indentionBase = 50;
- }
- public ArrayList<Element> getElements() {
- return elements;
- }
- public ArrayList<Element> getElementsData() {
- return elementsData;
- }
- @Override
- public int getCount() {
- return elements.size();
- }
- @Override
- public Object getItem(int position) {
- return elements.get(position);
- }
- @Override
- public long getItemId(int position) {
- return position;
- }
- @Override
- public View getView(int position, View convertView, ViewGroup parent) {
- ViewHolder holder = null;
- if (convertView == null) {
- holder = new ViewHolder();
- convertView = inflater.inflate(R.layout.treeview_item, null);
- holder.disclosureImg = (ImageView) convertView.findViewById(R.id.disclosureImg);
- holder.contentText = (TextView) convertView.findViewById(R.id.contentText);
- convertView.setTag(holder);
- } else {
- holder = (ViewHolder) convertView.getTag();
- }
- Element element = elements.get(position);
- int level = element.getLevel();
- holder.disclosureImg.setPadding(
- indentionBase * (level + 1),
- holder.disclosureImg.getPaddingTop(),
- holder.disclosureImg.getPaddingRight(),
- holder.disclosureImg.getPaddingBottom());
- holder.contentText.setText(element.getContentText());
- if (element.isHasChildren() && !element.isExpanded()) {
- holder.disclosureImg.setImageResource(R.drawable.close);
- //这里要主动设置一下icon可见,因为convertView有可能是重用了"设置了不可见"的view,下同。
- holder.disclosureImg.setVisibility(View.VISIBLE);
- } else if (element.isHasChildren() && element.isExpanded()) {
- holder.disclosureImg.setImageResource(R.drawable.open);
- holder.disclosureImg.setVisibility(View.VISIBLE);
- } else if (!element.isHasChildren()) {
- holder.disclosureImg.setImageResource(R.drawable.close);
- holder.disclosureImg.setVisibility(View.INVISIBLE);
- }
- return convertView;
- }
- /**
- * 优化Holder
- * @author carrey
- *
- */
- static class ViewHolder{
- ImageView disclosureImg;
- TextView contentText;
- }
- }
TreeViewItemClickListener.java:
- package com.example.androidtreeviewdemo.treeview;
- import java.util.ArrayList;
- import android.view.View;
- import android.widget.AdapterView;
- import android.widget.AdapterView.OnItemClickListener;
- /**
- * TreeView item点击事件
- * @author carrey
- *
- */
- public class TreeViewItemClickListener implements OnItemClickListener {
- /** adapter */
- private TreeViewAdapter treeViewAdapter;
- public TreeViewItemClickListener(TreeViewAdapter treeViewAdapter) {
- this.treeViewAdapter = treeViewAdapter;
- }
- @Override
- public void onItemClick(AdapterView<?> parent, View view, int position,
- long id) {
- //点击的item代表的元素
- Element element = (Element) treeViewAdapter.getItem(position);
- //树中的元素
- ArrayList<Element> elements = treeViewAdapter.getElements();
- //元素的数据源
- ArrayList<Element> elementsData = treeViewAdapter.getElementsData();
- //点击没有子项的item直接返回
- if (!element.isHasChildren()) {
- return;
- }
- if (element.isExpanded()) {
- element.setExpanded(false);
- //删除节点内部对应子节点数据,包括子节点的子节点...
- ArrayList<Element> elementsToDel = new ArrayList<Element>();
- for (int i = position + 1; i < elements.size(); i++) {
- if (element.getLevel() >= elements.get(i).getLevel())
- break;
- elementsToDel.add(elements.get(i));
- }
- elements.removeAll(elementsToDel);
- treeViewAdapter.notifyDataSetChanged();
- } else {
- element.setExpanded(true);
- //从数据源中提取子节点数据添加进树,注意这里只是添加了下一级子节点,为了简化逻辑
- int i = 1;//注意这里的计数器放在for外面才能保证计数有效
- for (Element e : elementsData) {
- if (e.getParendId() == element.getId()) {
- e.setExpanded(false);
- elements.add(position + i, e);
- i ++;
- }
- }
- treeViewAdapter.notifyDataSetChanged();
- }
- }
- }
MainActivity.java:
- package com.example.androidtreeviewdemo;
- import java.util.ArrayList;
- import com.example.androidtreeviewdemo.treeview.Element;
- import com.example.androidtreeviewdemo.treeview.TreeViewAdapter;
- import com.example.androidtreeviewdemo.treeview.TreeViewItemClickListener;
- import android.os.Bundle;
- import android.app.Activity;
- import android.content.Context;
- import android.view.LayoutInflater;
- import android.view.Menu;
- import android.widget.ListView;
- public class MainActivity extends Activity {
- /** 树中的元素集合 */
- private ArrayList<Element> elements;
- /** 数据源元素集合 */
- private ArrayList<Element> elementsData;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- LayoutInflater inflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
- init();
- ListView treeview = (ListView) findViewById(R.id.treeview);
- TreeViewAdapter treeViewAdapter = new TreeViewAdapter(
- elements, elementsData, inflater);
- TreeViewItemClickListener treeViewItemClickListener = new TreeViewItemClickListener(treeViewAdapter);
- treeview.setAdapter(treeViewAdapter);
- treeview.setOnItemClickListener(treeViewItemClickListener);
- }
- private void init() {
- elements = new ArrayList<Element>();
- elementsData = new ArrayList<Element>();
- //添加节点 -- 节点名称,节点level,节点id,父节点id,是否有子节点,是否展开
- //添加最外层节点
- Element e1 = new Element("山东省", Element.TOP_LEVEL, 0, Element.NO_PARENT, true, false);
- //添加第一层节点
- Element e2 = new Element("青岛市", Element.TOP_LEVEL + 1, 1, e1.getId(), true, false);
- //添加第二层节点
- Element e3 = new Element("市南区", Element.TOP_LEVEL + 2, 2, e2.getId(), true, false);
- //添加第三层节点
- Element e4 = new Element("香港中路", Element.TOP_LEVEL + 3, 3, e3.getId(), false, false);
- //添加第一层节点
- Element e5 = new Element("烟台市", Element.TOP_LEVEL + 1, 4, e1.getId(), true, false);
- //添加第二层节点
- Element e6 = new Element("芝罘区", Element.TOP_LEVEL + 2, 5, e5.getId(), true, false);
- //添加第三层节点
- Element e7 = new Element("凤凰台街道", Element.TOP_LEVEL + 3, 6, e6.getId(), false, false);
- //添加第一层节点
- Element e8 = new Element("威海市", Element.TOP_LEVEL + 1, 7, e1.getId(), false, false);
- //添加最外层节点
- Element e9 = new Element("广东省", Element.TOP_LEVEL, 8, Element.NO_PARENT, true, false);
- //添加第一层节点
- Element e10 = new Element("深圳市", Element.TOP_LEVEL + 1, 9, e9.getId(), true, false);
- //添加第二层节点
- Element e11 = new Element("南山区", Element.TOP_LEVEL + 2, 10, e10.getId(), true, false);
- //添加第三层节点
- Element e12 = new Element("深南大道", Element.TOP_LEVEL + 3, 11, e11.getId(), true, false);
- //添加第四层节点
- Element e13 = new Element("10000号", Element.TOP_LEVEL + 4, 12, e12.getId(), false, false);
- //添加初始树元素
- elements.add(e1);
- elements.add(e9);
- //创建数据源
- elementsData.add(e1);
- elementsData.add(e2);
- elementsData.add(e3);
- elementsData.add(e4);
- elementsData.add(e5);
- elementsData.add(e6);
- elementsData.add(e7);
- elementsData.add(e8);
- elementsData.add(e9);
- elementsData.add(e10);
- elementsData.add(e11);
- elementsData.add(e12);
- elementsData.add(e13);
- }
- @Override
- public boolean onCreateOptionsMenu(Menu menu) {
- // Inflate the menu; this adds items to the action bar if it is present.
- getMenuInflater().inflate(R.menu.activity_main, menu);
- return true;
- }
- }
treeview_item.xml:
- <?xml version="1.0" encoding="utf-8"?>
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="match_parent"
- android:layout_height="match_parent" >
- <ImageView
- android:id="@+id/disclosureImg"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_centerVertical="true"
- android:layout_alignParentLeft="true"/>
- <TextView
- android:id="@+id/contentText"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_centerVertical="true"
- android:layout_toRightOf="@id/disclosureImg"/>
- </RelativeLayout>
activity_main.xml:
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- tools:context=".MainActivity" >
- <ListView
- android:id="@+id/treeview"
- android:layout_width="match_parent"
- android:layout_height="match_parent"/>
- </RelativeLayout>
Android UI 之实现多级列表TreeView相关推荐
- android 仿qq好友动态,Android UI仿QQ好友列表分组悬浮效果
本文实例为大家分享了Android UI仿QQ好友列表分组悬浮效果的具体代码,供大家参考,具体内容如下 楼主是在平板上測试的.图片略微有点大,大家看看效果就好 接下来贴源代码: PinnedHeade ...
- android二级菜单ui,Android UI 之实现多级树形列表TreeView示例
所谓TreeView就是在Windows中常见的多级列表树,在Android中系统只默认提供了ListView和ExpandableListView两种列表,最多只支持到二级列表的实现,所以如果想要实 ...
- Android UI开发第二篇——多级列表(ExpandableListView)
开发中很多地方使用到了多级列表,android可以使用ExpandableListView很好的实现,下面模仿了手机qq的实现,见下图. 多级列表使用了ExpandableListView,自定义了A ...
- Android UI(五)云通讯录项目之联系人列表,带侧滑选择,带搜索框
作者:泥沙砖瓦浆木匠 网站:http://blog.csdn.net/jeffli1993 个人签名:打算起手不凡写出鸿篇巨作的人,往往坚持不了完成第一章节. 交流QQ群:[编程之美 36523458 ...
- android多级列表
我们开发app过程中,经常会碰到需要 多级列表展示的效果.而android原生sdk中根本没有3级 4级甚至更多级别的列表控件. 所以我们就要自己去实现一个类似treeListView 的控件,下面这 ...
- Android多级列表的实现
多级列表是ui入门,没什么特别需要注意的难点,直接上代码: 第一级列表窗口 /*** 第一级列表* 1.编辑状态下选中一个条目,会弹出底部操作栏* 2.非编辑状态下选中条目会跳转到下一级* 3.复选框 ...
- Android UI 小结
UI开发第1篇--自定义列表 编辑器加载中... MainActivity.classpublic class MainActivity extends Activity { @Override pu ...
- 安卓listView实现多级列表
安卓listView实现多级列表 实现两级列表我们可以使用ExpandableListView,但对于实现多级列表,使用ExpandableListView嵌套实现起来就比较费劲,可以使用listVi ...
- android教程 - android ui 介绍,多图详解 “Android UI”设计官方教程
我们曾经给大家一个<MeeGo移动终端设备开发UI设计基础教程>,同时很多朋友都在寻找Android UI开发的教程,我们从Android的官方开发者博客找了一份幻灯片,介绍了一些Andr ...
最新文章
- 基于增强现实和脑机接口的机械臂控制系统
- 020_html格式化
- Project Tango 的一些应用
- 使用WordPress的Kyma plugin同Kyma断开连接的实现
- 使用springfox 集成swagger 与spring mvc
- 带有谓词的Java中的功能样式-第1部分
- java hibernate dto_java – 正确使用Entity和DTO在Restful Web服务中...
- iptables 指定网卡_LINUX系统下的IPTABLES防火墙系统讲解(二)实战操作
- Hibernate学习笔记--映射配置文件详解
- hibernate 的三种状态 如何转化的。
- Excel VBA简单使用——数据缺失处理
- 新手学编程必会的100个代码
- Java课程设计-学生成绩管理系统
- CSDN账号找回密码的解决方法(原手机号不能使用)
- android9.0+wifi叹号,手机wifi连上有个感叹号怎么解决_wifi已连接但有感叹号的处理方法-系统城...
- 安全健康的使用计算机就要注意,长时间使用电脑应该注意健康
- 2015年最新苹果开发者账号注册流程详解
- 浙江省计算机程序比赛员,计算机学子在“图森未来杯”第十七届浙江省大学生程序设计竞赛中斩获金奖...
- PPT如何直接转换为word
- Maple学习(一)Maple的安装
热门文章
- Typora自定义主题分享 (Mac风、图片立体感...)
- PHP微信防止token过期,微信调用接口,防止Access_token过期的方法
- 没有json数据,自己造!mockjs的使用-模拟数据其实超级简单
- 语音邮件 voice mail 概述
- asp.net 是什么?
- java synchronized_Java中synchronized关键字理解
- python基础教程第三版豆瓣-数据结构与算法必读书单吐血整理推荐【附网盘链接】...
- 关于桌面程序被安全软件误判为HEUR:Trojan.Win32.Generic的解决方案
- Hamilton Jacobi
- c语言用分支结构判断最大字符,第3章C语言 分支结构PPT课件.ppt