最近几个项目都用到了EasyUI这个Jquery框架,目前感觉起来还是很好使的,展示效果很好,帮助文档什么的资料很多,而且互联网上Easy粉很多,大多数拥护和喜爱EasyUI的粉丝们都愿意在网络平台互相分享学习成果,甚至有专门的社区来讨论使用情况,网址是http://bbs.jeasyuicn.com/,里面的资源模块里有很多都是免费的学习资料,包括视频文档项目源码等,建议初学者去看视频,然后研究一下这个网站(sypro)的实现http://sshe.jeasyuicn.com/,甚至有视频教程教大家怎么实现这个项目。互联网是一个巨人,他博学多才,期待能站在巨人的肩膀上开开眼界,学习到更多的知识技能,对于将来或现在的工作都是一个很大的收获。
本篇博客是对EasyUI中树的实现的总结。如果想要展示一棵树,有很多方式,当然要分析你的需求。如果是展示省市区、学校、部门等大数据的话,建议还是使用异步加载。当然如果只是展示几个几乎不变的菜单项,就可以扁平化的展示你的数据了。
首先介绍怎么实现一棵异步树。
项目前准备:
1、首先你要搭建一个你熟悉的框架环境,然后再前台加入EasyUI的源码包,并在页面引入js和css等文件。本文的实例主要讲解怎么实现树,以SSH框架为例。如果还有不懂怎么搭建EasyUI框架的同学,可以在EasyUI的中文社区里找EasyUI的初级视频来看看,非常简单的。
2、建立数据库,比方说我们要通过树来展示你的菜单,那么就要先看一下EasyUI中tree的Data Format,也就是说我们要了解后台传给前台什么样式的Json格式。
从上面的文档截图可以看出,他的数据格式有以下属性:id,text,state,checked,attributes,children;
我们在设计数据库的时候,可以尽量的将节点id和节点名称分别设置成id和text,这样在前台解析Json的时候就能直接认出这些属性值,并显示出数据来。当然利用扩展的方式的话,你可以不必按照这些规范来,但是需要在tree控件里传入几个参数来传入属性值。
首先,建立一个t_menu表:(注释如图)
外键:
测试数据:
准备完以上的内容之后,我们开始做demo。
1、加入EasyUI的树控件:
<ul id="menuTree" class="easyui-tree" data-options="url:'<%=basePath%>menuAction!getTreeNode.action',parentField:'pid',lines:true,onLoadSuccess:function(node, data){$(this).tree('collapseAll')}"></ul>
解析:
data-options里的URL是Action的路径,p
arentField设置成我们model里的pid,
lines:true用来显示树节点前的加减号,
onLoadSuccess:function(node, data){$(this).tree('collapseAll')}用来设置关闭所有的树节点。
加入扩展js:
1 $.fn.tree.defaults.loadFilter = function (data, parent) {
2 var opt = $(this).data().tree.options;
3 var idFiled,
4 textFiled,
5 parentField;
6 if (opt.parentField) {
7 idFiled = opt.idFiled || 'id';
8 textFiled = opt.textFiled || 'text';
9 parentField = opt.parentField;
10
11 var i,
12 l,
13 treeData = [],
14 tmpMap = [];
15
16 for (i = 0, l = data.length; i < l; i++) {
17 tmpMap[data[i][idFiled]] = data[i];
18 }
19
20 for (i = 0, l = data.length; i < l; i++) {
21 if (tmpMap[data[i][parentField]] && data[i][idFiled] != data[i][parentField]) {
22 if (!tmpMap[data[i][parentField]]['children'])
23 tmpMap[data[i][parentField]]['children'] = [];
24 data[i]['text'] = data[i][textFiled];
25 tmpMap[data[i][parentField]]['children'].push(data[i]);
26 } else {
27 data[i]['text'] = data[i][textFiled];
28 treeData.push(data[i]);
29 }
30 }
31 return treeData;
32 }
33 return data;
34 };
2、Action类实现
首先是model类TMenu.java,映射数据库的类。
1 import java.util.HashSet;
2 import java.util.Set;
3 import javax.persistence.CascadeType;
4 import javax.persistence.Column;
5 import javax.persistence.Entity;
6 import javax.persistence.FetchType;
7 import javax.persistence.Id;
8 import javax.persistence.JoinColumn;
9 import javax.persistence.ManyToOne;
10 import javax.persistence.OneToMany;
11 import javax.persistence.Table;
12
13 /**
14 * TMenu entity. @author MyEclipse Persistence Tools
15 */
16 @Entity
17 @Table(name = "t_menu", catalog = "easyui")
18 public class TMenu implements java.io.Serializable {
19
20 // Fields
21
22 private String id;
23 private TMenu TMenu;
24 private String text;
25 private String iconCls;
26 private String url;
27 private Set<TMenu> TMenus = new HashSet<TMenu>(0);
28
29 // Constructors
30
31 /** default constructor */
32 public TMenu() {
33 }
34
35 /** minimal constructor */
36 public TMenu(String id) {
37 this.id = id;
38 }
39
40 /** full constructor */
41 public TMenu(String id, TMenu TMenu, String text, String iconCls, String url, Set<TMenu> TMenus) {
42 this.id = id;
43 this.TMenu = TMenu;
44 this.text = text;
45 this.iconCls = iconCls;
46 this.url = url;
47 this.TMenus = TMenus;
48 }
49
50 // Property accessors
51 @Id
52 @Column(name = "id", unique = true, nullable = false, length = 36)
53 public String getId() {
54 return this.id;
55 }
56
57 public void setId(String id) {
58 this.id = id;
59 }
60
61 @ManyToOne(fetch = FetchType.LAZY)
62 @JoinColumn(name = "pid")
63 public TMenu getTMenu() {
64 return this.TMenu;
65 }
66
67 public void setTMenu(TMenu TMenu) {
68 this.TMenu = TMenu;
69 }
70
71 @Column(name = "text", length = 100)
72 public String getText() {
73 return this.text;
74 }
75
76 public void setText(String text) {
77 this.text = text;
78 }
79
80 @Column(name = "iconCls", length = 50)
81 public String getIconCls() {
82 return this.iconCls;
83 }
84
85 public void setIconCls(String iconCls) {
86 this.iconCls = iconCls;
87 }
88
89 @Column(name = "url", length = 200)
90 public String getUrl() {
91 return this.url;
92 }
93
94 public void setUrl(String url) {
95 this.url = url;
96 }
97
98 @OneToMany(cascade = CascadeType.ALL, fetch = FetchType.LAZY, mappedBy = "TMenu")
99 public Set<TMenu> getTMenus() {
100 return this.TMenus;
101 }
102
103 public void setTMenus(Set<TMenu> TMenus) {
104 this.TMenus = TMenus;
105 }
106
107 }
其次是model类Menu.java,此处的model是pageModel,是为了接应前台的name值的:
1 public class Menu {
2
3 private String pid;//父菜单ID
4 private String pText;//父菜单名称
5 private String id;//子菜单Id
6 private String text;//子菜单名称
7 private String iconCls;//子菜单图标
8 private String url;//子菜单路径
9 private String state;
10
11 //---------------set/get--------------
12
13 public String getPid() {
14 return pid;
15 }
16 public void setPid(String pid) {
17 this.pid = pid;
18 }
19
20 public String getpText() {
21 return pText;
22 }
23 public void setpText(String pText) {
24 this.pText = pText;
25 }
26 public String getId() {
27 return id;
28 }
29 public void setId(String id) {
30 this.id = id;
31 }
32
33 public String getText() {
34 return text;
35 }
36 public void setText(String text) {
37 this.text = text;
38 }
39 public String getIconCls() {
40 return iconCls;
41 }
42 public void setIconCls(String iconCls) {
43 this.iconCls = iconCls;
44 }
45 public String getUrl() {
46 return url;
47 }
48 public void setUrl(String url) {
49 this.url = url;
50 }
51 public String getState() {
52 return state;
53 }
54 public void setState(String state) {
55 this.state = state;
56 }
57
58
59 }
Action类:
1 package com.action;
2
3
4 import org.apache.struts2.convention.annotation.Action;
5 import org.apache.struts2.convention.annotation.Namespace;
6 import org.apache.struts2.convention.annotation.ParentPackage;
7 import org.springframework.beans.factory.annotation.Autowired;
8
9 import com.pageModel.Menu;
10 import com.service.IMenuService;
11
12 import com.opensymphony.xwork2.ModelDriven;
13
14 @ParentPackage("basePackage")
15 @Namespace("/")
16 @Action(value="menuAction")
17 public class MenuAction extends BaseAction implements ModelDriven<Menu> {
18 Menu menu=new Menu();
19 @Override
20 public Menu getModel() {
21 // TODO Auto-generated method stub
22 return menu;
23 }
24 private IMenuService menuService;
25
26 public IMenuService getMenuService() {
27 return menuService;
28 }
29 @Autowired
30 public void setMenuService(IMenuService menuService) {
31 this.menuService = menuService;
32 }
33 /**
34 * 异步获得树节点
35 */
36 public void getTreeNode(){
37 super.writeJson(menuService.getTreeNode(menu.getId()));
38 }
39 }
2、了解getTreeNode()方法的实现:
对应service的实现类:
1 package com.service.impl; 2 3 import java.util.ArrayList; 4 import java.util.HashMap; 5 import java.util.List; 6 import java.util.Map; 7 import java.util.Set; 8 9 import org.springframework.beans.BeanUtils; 10 import org.springframework.beans.factory.annotation.Autowired; 11 import org.springframework.stereotype.Service; 12 13 import com.dao.IBaseDao; 14 import com.model.TMenu; 15 import com.pageModel.Menu; 16 import com.service.IMenuService; 17 18 @Service("menuService") 19 public class MenuServiceImpl implements IMenuService { 20 private IBaseDao<TMenu> menuDao; 21 22 23 public IBaseDao<TMenu> getMenuDao() { 24 return menuDao; 25 } 26 27 @Autowired 28 public void setMenuDao(IBaseDao<TMenu> menuDao) { 29 this.menuDao = menuDao; 30 } 31 32 33 @Override 34 public List<Menu> getTreeNode(String id) { 35 List<Menu> menus=new ArrayList<Menu>(); 36 StringBuffer hql=new StringBuffer(); 37 hql=hql.append("from TMenu t where "); 38 Map<String, Object> map=new HashMap<String, Object>(); 39 if (id==null || "".equals(id)) { 40 //返回总根节点 41 hql=hql.append(" t.TMenu is null"); 42 43 } else { 44 //异步加载当前id下的子节点 45 hql=hql.append(" t.TMenu.id=:id"); 46 map.put("id", id); 47 } 48 List<TMenu> tMenus= menuDao.find(hql.toString(),map); 49 for (TMenu tMenu : tMenus) { 50 Menu menu=new Menu(); 51 BeanUtils.copyProperties(tMenu, menu); 52 Set<TMenu> set=tMenu.getTMenus(); 53 if (set!=null && !set.isEmpty()) { 54 menu.setState("closed"); //节点以根节点形式体现(文件夹) 55 } else { 56 menu.setState("open"); //节点 以叶子形式体现(文件) 57 } 58 menus.add(menu); 59 } 60 return menus; 61 } 62 63 }
最后展示实现效果:当单击加号的时候才会加载其子节点,异步实现了功能树。
下篇博客将介绍另一种加载树的方式,就是一次把所有的树节点都加载上来,显示扁平化数据。