《Extjs教程__Tree(树)..doc》由会员分享,可在线阅读,更多相关《Extjs教程__Tree(树)..doc(18页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、第八章 Tree(树)层级数据是很多开发者所熟知的。“根-枝-叶的结构是很多用户界面的最根本的特征。windows的资源管理器中就展示了一个包含子节点、父节点和更深层次节点的树,以此来展示文件夹和文件的层级关系。Ext.tree允许开发者只通过使用几行代码就展示出这样的层级数据,并且提供了大量的简单的配置来适应更广泛的需求。虽然ExtJS默认的tree节点是file和folder图标样式,但是它不会仅仅将树局限于文件系统这一概念里。每一项的图标和文字,或者树中的节点,都可以根据动态或者静态的数据来改变不需要自己写代码。想想,我们如果希望建立一个用户组,为每个用户展示它们自己的图标;又或者希望展
2、示一画廊的图片并且在图标中预览这些图片。ExtJS可以帮助我们实现这些愿望,而且十分简单。种植未来:作者使用培育植物的过程形象化地比喻建立tree的过程ExtJS的tree不会关心你显示什么样的数据,因为它可以随心所欲地处理任何你碰到情况。数据可以实现就加载好,又或者从逻辑上进行分割。你可以直接在tree中编辑数据,改变标签和位置,或者你可以修改整棵树的样子以及每个节点的外观,一切都为了用户体验。ExtJS的tree是从Component模型上建立起来的。Component是真个ExtJS框架的根底。也就是说,开发者从熟悉的Component的系统中获得了便利,因为用户能偶得到一种统一的和集成
3、的体验效果。你同样可以保证tree和应用中的其他组件天衣无缝地工作在一起。从小种子开始:在这章里,你可以看到如何使用很少的代码建立一棵树。我们还将讨论利用唯一的数据结构来产生一个tree,以及如何使用数据可以让你操控重要的配置项。ExtJS树提供了很多高级支持,例如排序、拖拽等。但是,如果你想要真的定制一棵,我们还需要探索如何重写或者扩展configure options(配置工程)、methods(方法)、events(事件)的方法。tree是通过实例化Ext.tree.TreePanel类来建立的,它包含了很多Ext.tree.TreeNodes的节点类。这两个类是ExtJS树的核心,也是
4、我们这章讨论的主题。但是,还有很多其他先关的类需要介绍,我们现在列出Ext.tree包中的全部条目:AsyncTreeNode允许子节点异步加载的节点DefaultSelectionModel标准的TreePanel的单项选择模式MultiSelectionModel允许多项选择节点的选择模式RootTreeNodeUI作为TreePanel根的特殊的TreeNodeTreeDragZone为TreeNode的抓起提供支持TreeDropZone为TreeNode的放下提供支持TreeEditor允许节点标签被编辑TreeFilter对TreePanel中子节点的过滤进行支持TreeLoade
5、r从指定的URL生成TreePanelTreeNode在TreePanel中显示节点的最主要的类TreeNodeUI为TreeNode提供最根本的界面TreePanel树状结构显示数据最主要的树类TreeSorter支持TreePanel中节点的排序天啊!幸运的是,你不用同时全部使用它们。TreeNode和TreePanel提供了最根底的东西,其它的类是用来提供额外功能的。我们将一个一个对其介绍,讨论如何使用它们并且展示几个练习例如。我们的第一个幼苗:现在,你可能还在思考ExtJS树带来的各种可能性,想亲手去干。尽管Ext.tree类包含了丰富的功能,但是你只需要几行代码就能让一切跑起来。在接
6、下来的例子中,我们假设你有一个准备好的空白HTML页面,包含了所有ExtJS所需要的引用。大局部的代码都基于以前的章节,好让我们抓住重点,你要孤立地看待它们。最好的方式是吧JS分别放在各个文件中并把代码放在Ext.onReady函数中。但是,你依然可以根据你自己的编码风格来处理。准备好土地:首先我们需要建立元素,用来向其渲染TreePanel。因此我们需要把它设置为我们想要的tree的大小:tree的JS代码可以分为三局部。首先,我们需要确定tree展现的方式。Ext.tree.TreeLoader类提供了这样的功能,现在我们采用最简单的方法来使用它:var treeLoader = new
7、Ext.tree.TreeLoader(dataUrl: :/localhost/samplejson.php);dataUrl配置项说明了提供产生树所需要的JSON数据的脚本存在的位置。我现在不想讨论JSON的具体结构,让我们先保存这个问题。每个tree都需要一个根节点,它扮演了所有后裔的终极祖先的角色。为了建立root node根节点,我们使用Ext.tree.AsyncTreeNode类:var rootNode = new Ext.tree.AsyncTreeNode(text: Root);之所以使用AsyncTreeNode而不是TreeNode,是因为我们需要从效劳器端获得节点,
8、并且一层一层地去加载而不是一次性加载。N:AsyncTreeNode使用AJAX来保证用户不用花太多时间等待数据加载以及首次节点的渲染。最后,我们建立tree本身,利用Ext.tree.TreePanel类:var tree = new Ext.tree.TreePanel(renderTo:treecontainer,loader: treeLoader,root: rootNode);只需要把root node和TreeLoader放在配置中,再加上决定TreePanel渲染位置的renderTo配置,就可以显示tree了。再次提醒,你一定要记住把这些代码放在Ext.onReady中,以此
9、确定DOM在代码执行前已经准备好了。tree(树)离开data(数据)生长不了:我们看到,只需要十一行数据就可以显示如下的用户界面:我猜这还不够,但是这十一行代码提供了大量的功能。利用异步远程加载子节点,我们获得了统一的交互界面和体验。并不只是这么简单,因为我们将要介绍Ext树的最重要的局部data数据。JSON:标准的TreeLoader支持以一种特定的格式支持JSON数据包含node定义的数组,如下所示: id: 1, text: No Children, leaf: true , id: 2, text: Has Children,children: id: 3,text: Youngs
10、ter,leaf: truetext属性代表了tree里node的标签文本。id属性用来唯一标识一个node,并且将被用来决定选中和展开哪个节点。利用id属性我们可以使得TreePanel中高级功能的实现变的简单,这在之后会介绍到。children属性是可选的。leaf属性是被用来判断是否为叶子节点。在tree中leaf节点不能被展开,也不会有节点前卖弄的加号图标。ID简介:默认来说,TreeNode会被分配一个自动产生的ID,说明id配置可有可无。这个自动产生的id是一个字符串,形式如:ynode-xx,xx代表数字。id可以被用来获得一个你之前引用的节点。但是,你很可能自己分配id。当你异
11、步加载节点的时候,效劳器脚本需要准确知道那个节点被点击从而传回其子节点数据。通过设置id,你可以发现在效劳器端匹配节点变得很简单。额外的数据:虽然id、text和leaf属性十分常用,但是JSON的用途不只局限于此。事实上,任何TreeNode的配置都可以被JSON初始化,这是我们探索tree的高级功能的一个小技巧。你可以添加应用需要的特定的数据例如也许你的节点代表了产品并且你希望它们包含价格信息。任何属性都可以以TreeNode配置项的方式组织起来,并且包含在TreeNode的attributes属性之中。也就是说,TreeNode本身没有的属性会被组织在attributes属性之中。XML
12、:XML不是直接被tree支持的。但是你可以利用ExtJS的数据支持来让XML也能被读取。通常,使用JSON会让一切变得简单,虽然一些程序会使用XML传送数据。所以它值得我们讨论一下。我们可以使用Ext.data. Proxy来获得数据,但是我们需要在读取数据的时候把XML转换一下:var xmltree = new Ext.tree.TreePanel(el: treeContainer);var proxy = new Ext.data. Proxy(url: :/localhost:81/ext/treexml.php);proxy.load(null, read: function(x
13、mlDocument) parseXmlAndCreateNodes(xmlDocument);, function() xmltree.render(); );我们建立了一个新的TreePanel和 Proxy,并且指定在proxy加载时使用Ext.data.Reader来处理进入的XML数据。我们接下来会告诉reader把XML传递到parseXmlAdnCreateNodes。在这个函数中,你可以根据XML数据来建立根节点和子节点,我们向它直接传递了一个XML文档。JS完全能够处理XML数据,虽然你可能更习惯于转换XHTML文档中DOM的方式。通过读取XML文档你可以建立和使用textn
14、odes文本节点。因为在这种方式里,你需要访问原始的XML节点,你可以完全地控制由此产生的tree和相应的树节点。照料你的树:我们将讨论使你的tree更加实用的功能。拖拽、排序和编辑节点。拖拽:当你使用TreePanel的时候,ExtJS管理着由拖拽产生的用户界面。只要添加enableDD: true配置到tree里,这时你可以通过拖拽重新组织树的节点,当显示绿色加号的时候,代表你可以向目标防止该节点。N:TreePanel不仅仅只会“照顾它们自己的节点,当你的界面中有两棵树的时候,他们的节点可以在树之间相互拖拽。但这还没有完。当你刷新你的页面,所有的节点又回到原位。因为TreePanel不会
15、自动知道你是否需要保存改动,为了实现保存,我们需要借助一些事件event。TreePanel的beforemovenode事件在拖放的时候松开鼠标,但是在TreePanel反映变化前这一时间点触发。我们可以添加如下代码来告诉效劳器移动节点这一事件:tree.on(beforemovenode, function(tree, node, oldParent, newParent, index) Ext.Ajax.request(url: :/localhost/node-move.php,params: nodeid: node.id,newparentid: newParent.id,oldp
16、arentid: oldParent.id,dropindex: index););我们为beforemovenode事件添加了事件处理函数。这个函数在调用的时候向其传递了几个有用的参数:1, tree:该事件所发生的TreePanel;2, node:被移动的节点;3, oldParent:被移动节点之前的父节点;4, newParent:被移动节点的新的父节点;5, index:移动目的的序号。我们利用以上这些来形成AJAX对效劳器发请求时传递的参数。这样,你可以获得tree现在的任何信息和状态,你的效劳端脚本可据此以完成任何你需要的动作。在某些桩抗下,你可能需要取消拖放节点。当你在bef
17、oremovenode函数中出现了逻辑上的错误,你需要复原改动。如果你不发送AJAX请求,可以直接在函数的最后返回false即可,相应的动作将被取消。但是如果你采用了AJAX,就困难了许多,因为XML Request是异步发生的,而且这个事件函数会执行一些默认的动作,其中就包括允许节点移动。即然这样,你需要确定你为AJAX提供了failure函数的配置,向这个函数传递足够的参数,好让树复原到以前的状态。因为beforemovenode通过默认传递的参数提供了大量的信息,所以你可以传递必须的数据来对产生的错误进行管理。排序:我们可以通过一种灵活的方法为TreePanel排序TreeSorter。
18、在先前代码的根底上,我们可以建立一个如下的TreeSorter:new Ext.tree.TreeSorter(tree, folderSort: true,dir: “asc);因为TreeSorter采用了几个约定leaf节点由leaf属性标明并且标签文本在text属性里说明我们可以很容易地按字母进行排序。dir属性g奥素我们TreeSorter是按照升序还是降序进行排序的。folderSort为true默认为false说明叶子节点需要在非叶子节点下进行排序,也就是说,整个树每层都要进行排序。如果你的数据不仅仅是简单的文本,我们可以通过sortType配置项自定义排序。sortType的值
19、是一个函数,而且只向其传递一个参数:a TreeNode。sortType允许你为TreeNode新加自定义属性这个属性可能是效劳器传来的和业务相关的信息并且把它转化成Ext可以排序的格式,换句话说,也就是转化为标准的JS类型,如整型、字符串、日期。sortType一般用在原有的数据格式不能或者不方便做搜索以及排序的时候从效劳器返回的数据可能会被用于不同的目的,我们可以把日期转化为标准的格式从US样式的MM/DD/YY转化为YYYYMMDD样式以便排序或者我们要去掉货币中无用的符号从而转化为数字。sortType: function(node) 在上面的这个例子中,我们返回了node里自定义的
20、属性,因为这个属性的值是有效的JS日期,Ext可以对其进行排序。这就是一个关于如何通过sortType选项让TreeSorter对任何效劳器端数据进行排序的演示。编辑:很多时候,如果能编辑节点的值将是很有用的一个功能。在观察分类产品的层级结构时,你可能希望对产品的种类或者产品的名字做直接的修改,而不希望再去访问别的页面。那么你就可以通过Ext.tree.TreeEditor来实现这个功能。TreeEditor默认在你双击节点标签的时候会为你的节点提供一个用于编辑的TextField。但是,和drag-and-drop拖拽一样,开启这个功能并不会自动向效劳器端保存改动。你需要在编辑完节点后触发某
21、个事件然后调用事件提供的函数来进行相应的操作:editor.on(beforecomplete, function(editor, newValue, originalValue)/ Possible Ajax call?);beforecomplete事件函数有如下三个参数:1. editor:用来编辑这个节点的编辑字段;2. newValue:输入的值;3. originalValue:改动前的值。然而,你需要注意的是editor这个参数不是一个普通的Ext.form.Field。它有一些额外的属性,其中最有用的就是editNode对编辑节点的引用。你可以通过这个属性来获得节点的id,这个
22、属性在向效劳器发请求来和数据库同步编辑的数据时很重要。与TreePanel的beforemovenode事件相比,beforecomplete可以让用户通过在处理函数的结尾返回false来取消编辑改动。AJAX请求需要提供failure处理函数来手动地复原先前的数值。现在我们介绍了如何建立一个简单的节点编辑器,但这还意味着我们还需要完成一些更复杂的功能。TreeEditor的构造函数里有两个可选的参数一共三个,后两个可选。一个是field的配置对象,一个是TreeEditor的配置对象。field的配置可以是一下两者之一: 标准TextField编辑器的配置对象; 已经建立的表单字段的实例。如
23、果是后者,你可以在此采用NumberField,DateField或者其他的Ext.form.Field。第二个可选参数可以让你配置TreeEditor,只要一点小小的改动就能实现令人冲动的功能。例如,我们可以使用cancelOnEsc来允许用户通过按下Esc键取消任何编辑;或者使用ignoreNoChange来避开编辑完成事件如果值在编辑后并没有改变。修剪树枝:关于TreePanel,还支持一些其它的技巧改变selection model选择模型,过滤节点,以及显示背景菜单等。所以,让我们现在就学习它们。Selection models选择模型:在我们之前的例如代码里,我们可以通过拖拽和编辑
24、TreeNodes来即时改变tree。TreePanel默认使用single-selection model单项选择模型。我们之前的代码都需要去选择节点,但是对于tree来讲,简单地选择一个节点并不能满足所有的需求,所以我们需要知道控制选择方式的更多的方法和特性。一个很好的例子就是,我们在选择一个节点以后,会自动产生一个信息面板来展示节点的细节。也许你有一个tree用来展示产品,点击一个节点需要显示该产品的价格和库存量。我们可以使用selectionchange事件来让这一切得以实现。我们还是使用先前的代码做开始,我们添加一下的代码:tree.selModel.on(selectionchan
25、ge, function(selModel, node) var price = node.attributes.price;);selectionchange事件的第二个参数node使得我们很容易获得自定义的节点属性值。如果我们允许多节点选择呢?我们怎么去实现,我们怎么去处理selectionchange事件呢?我们可以使用Ext.tree.MultiSelectionModel来建立我们的TreePanel:var tree = new Ext.tree.TreePanel(renderTo:treeContainer,loader: treeLoader,root: rootNode,s
26、elModel: new Ext.tree.MultiSelectionModel();配置就是这么简单。虽然selectionchange的处理方法和默认的selection model很相似,但是还是有重要的区别。第二个参数是节点的数组,而不是单一的节点。selection model并不只局限于获得与选择相关的信息。它还允许控制当前的选择。例如:MultiSelectionModel.clearSelections()方法在你处理有关多节点事件之后可以帮助你去除所有的选中状态。DefaltSelectionModel有方法selectNext和selectPrevious可以让你在tre
27、e中定位,在节点的层级中根据需求向上或者向下移动。背景菜单context menu:我们现在已经介绍了很多TreePanel提供的特性,所以让我们用一些练习来稳固一下。当你右击节点时弹出一个context menu,这是一种界面上的“捷径。我们使用的代码在之前的段落中已经介绍过,首先,让我们建立menu,然后把它加到TreePanel中:var contextMenu = new Ext.menu.Menu(items: text: Delete, handler: deleteHandler , text: Sort, handler: sortHandler );tree.on(conte
28、xtmenu, treeContextHandler);TreePanel提供了一个contextmenu事件来监听右击节点。注意,我们的监听器不像之前的例子中一样采用匿名函数为了让代码更易读。首先,treeContextHandler用来处理contextmenu事件:function treeContextHandler(node) node.select();contextMenu.show(node.ui.getAnchor();这个函数在调用时有一个node参数,我们需要在这个函数中先选择这个node。然后我们用show方法弹出context menu,这个方法有一个参数,用来说明弹
29、出的位置。在这个例子里,我们的位置为TreeNode的文本处。菜单处理:这个菜单有两个入口Delete和Sort,让我们看下Delete的处理函数:function deleteHandler() tree.getSelectionModel().getSelectedNode().remove();利用我们之前的有关selection model的知识,我们可以获得在treeContextHandler中选择的节点,然后调用它的remove方法。这将从TreePanel中删除这个节点和它的子节点。注意,我们不把改变传递给效劳器,但是如果你需要这么做,TreePanel有remove事件,你可
30、以用它的处理函数来提供这个功能。Sort处理函数如下:function sortHandler() tree.getSelectionModel().getSelectedNode().sort(function (leftNode, rightNode) return (leftNode.text.toUpperCase() rightNode.text.toUpperCase() ? 1 : -1););再一次地,我们使用selection model来获得选中的节点。ExtJS提供了sort方法,这个方法的第一个参数是一个函数,拥有两个参数:一对节点。在这个例子里,我们通过节点的text
31、属性来进行降序排序,但是你可以利用别的自定义节点属性。N:使用这个方法和TreeSorter的排序不冲突,因为TreeSorter的排序只监听beforechildrendered、append、insert和textchange事件。其他的改动并不受影响。Delete动作将彻底地删除选择的节点,Sort动作将依照text标签来对子节点进行排序。过滤:“试验的标记,所以我只简要介绍。它用在用户想要根据节点某个特定的属性去检索节点时。这个属性可以是text、id或者其他在创立节点时自定义的信息。让我们利用之前的context menu来演示过滤功能,首先,我们需要建立TreeFilter:var
32、 filter = new Ext.tree.TreeFilter(tree);你需要回到context menu的配置里,在items里为其添加一个新的入口: text: Filter, handler: filterHandler 我们现在需要建立一个filterHandler函数来实现filter动作:function filterHandler() var node = tree.getSelectionModel().getSelectedNode();filter.filter(Bee, text, node);像别的处理函数一样,我们先获得当前选中的节点,然后调用filter函数
33、。这个函数有三个参数:1. 被过滤的值2. 过滤的属性可选,默认为text3. 开始过滤的节点我们让被选中的节点作为开始过滤的节点,这意味着我们将对右击的产生菜单的节点的子节点通过特定的值做过滤。我们的上面的例子包含aardvark、bee、cockroach这些动物的例子不需要做过滤,但是在某些状况中需要。线上的软件文档多级且详细,采用过滤可以对标题迅速检索。你可以使用checkbox或者弹出对话框来让用户输入过滤的数据,这样会带来灵活的用户体验。根:虽然我们演示了很多Ext中树的功能,但是tree真正的强大之处在于它的设置、方法、各种类提供的加载点。我们已经了解了很多配置TreePanel
34、和TreeNode的方法,让我们得以实现强大的功能。然而,还有很多的配置项可以加强我们的tree,我们将会对一些有趣的配置进行说明。调整TreePanel:默认的,TreePanel有很多图形界面上的加强配置,可以满足一些需求。例如,把animate设置为false可以阻止节点的展开和收拢的平滑的动画显示效果。这点在用户频繁展开关闭节点时很有帮助,可以放置动画效果带来的显示不畅。因为TreePanel是扩展自Ext.Panel,所以它支持所有标准的Panel的功能。它可以支持tbar和bbar(顶部和底部工具栏),header和footer元素,收起/展开的功能。TreePanel同样可以包含
35、于Ext.ViewPort和Ext.layout之中。装饰:对于纯粹的装饰属性,TreePanel提供了lines选项,当它被设置为false,将把TreeNodes之间的层级导航线禁止掉。这点在形成简单的tree时很有用,可以放置界面的混乱。hlColor属性适用于drag-drop属性开启的tree。它控制节点的颜色高亮属性值为十六位数字串,例如990000,当节点放下时被触发可以通过设置dlDrop属性为flase来禁止。设置trackMouseOver属性为false可以禁止悬停在某个节点上时产生的高亮。调整TreeNode:在很多情况下,你不能人工地建立TreeNode除了root节
36、点,所以你可能想,那些配置选项对你来说没什么用。其实不是这样的,因为不只是id和text属性可以拿来创立节点所有JSON中的TreeNode属性满足配置属性要求的都可以被用来建立节点。如果你有如下样子的JSON: text: My Node, disabled: true, href: :/extjs 你将获得一个开始为disabled的节点,但是一旦它设置为enabled,它将成为到extjs 的链接。这个功能在传递应用的特殊信息的时候很有用。例如,你的效劳逻辑要求你某些节点不能拥有子节点。这是你设置allowChildren: false就可以实现这样的功能该节点不能作为drop的目标。你
37、还可以设置draggable: false,这样你就可以组织某些节点的的拖放。我们可以通过设置checked: true来使得某个节点拥有checkbox事实上,checked不管设置为true或者false都起作用。这些配置选项允许你规定节点的行为。还有许多其它的和TreeNode相关的配置。你可以使用icon配置来提供自定义的图标,或者通过cls属性来提供CSS样式。qtip选项允许你弹出tooltip可以是对节点的说明。控制操作:当TreePanel被配置后,我们可以操控它的节点。TreePanel允许你在不同的层级之间“航行,从选择的节点转移到父节点或者子节点,或者顺着当前的层级向上或
38、者向下移动。我们也可以根据节点的路径选择或者展开节点,也可以以此找到某个节点。expandAll和collapseAll方法可以使我们展开合起所有节点,可以被用来还复原默认的状态。每个方法都有一个布尔值的参数,用来说明改变是否需要animate有动画效果。expandPath方法的第一个参数是节点的path(路径)。这个path唯一的利用层级确定了某个节点例如:/n-15/n-56/n-101。我们从这个路径可以知道,目标节点的id为n-101,n-15是root节点,root节点有一个id为n-56的子节点。如果你了解XPath,这个就很好理解。如果你不了解的话,你可以把她和IP地址做比照提
39、供了为一的路径引用方式。通过向expandPath传递这个参数,tree可以定位到指定的节点,并且展开之。想象一下代码:Ext.Msg.prompt(Node, Please enter a product name,function(btn, text)if (btn = ok)var path = GetNodePathFromName(text);tree.expandPath(path););GetNodePathFromName函数可以查询效劳器并且返回节点的id,可以通过用户的输入快速定位节点。TreePanel.getNodeById也可以通过类似的方式使用。不仅是展开节点,还可
40、以进行进一步的控制。在某些环境下,你可能需要做一些撤销动作,这是你需要获得节点的路径。TreeNode.getPath提供了这样的功能,你可以用这个方法来存储节点的位置。更多的方法:TreeNode还有很多其他的方法。我们已经介绍了sort和remove,但是现在我们可以添加一些根本应用程序的方法,如collapse和expand、enabled和disable,expandChildNodes和collapseChildNodes可以用来展开合起所有的子节点。findChild和findChildBy方法可以允许简单或者自定义的对节点的搜索,以下这个例子我们用来查找第一个price属性值为3
41、00的节点:var node = root.findChild(price, 300);在某些情况下,你可能需要对大量的节点属性进行操控,你可以使用TreeNode.eachChild方法:root.eachChild(function(currentNode) currentNode.attributes.price += 30;);因为第一个参数是一个函数,我们可以展示不同的需求带来的逻辑。事件捕获:我们已经演示了很多用户和tree交互的方法,但是还有很多有用的事件。先前,我们讨论了TreeNode的checked配置项的用法,当checkbox被勾选或者取消,checkchange事件被
42、激发。以下代码可以使得选中状态高亮显示:tree.on(checkchange, function(node, checked) node.eachChild(function(currentNode) currentNode.ui.toggleCheck(););我们让选中对TreeNode的子节点都生效,我们可以高亮显示节点来清楚地显示选中状态,或者实现其他的逻辑,例如在页面的任何位置展示新被选中的节点的信息。TreePanel事件的一个更通常的应用就是检查改动和与效劳端同步改动。例如,一棵分类产品的树有着逻辑约束某些特价商品需要说明最高价格。我们可以使用beforeappend事件来检测
43、:tree.on(beforeappend, function(tree, parent, node) return node.attributes.price parent.attributes.maximumPrice;);这个例如演示了ExtJS提供的一种模式返回false来取消动作的执行。在这里,如果价格超过了最高价,就会返回false,节点就不会被参加。记录状态:在很多应用中,TreePanel是用来做导航的,展示层级结构,没个节点是一个HTML的链接。在这种场景里,如果用户希望观看多个页面,一个接着一个,默认的TreePanel的行为会导致混乱。因为tree在页面刷新时不会保持状态
44、,当用户返回时,所有的展开的节点将被收起。所以,当用户需要经常定位他们感兴趣的页面时,往往会因此失去耐性。StateManager状态管理器:现在我们有一个管理TreePanel的好方法,可以方便地存储和复原状态。我们需要存储每个TreeNode的展开的状态,当页面重新加载的时候,我们可以重现展开的状态。我们可以使用Ext.state.Manager和它的CookieProvider来存储展开状态。我们可以这样初始化:Ext.state.Manager.setProvider(new Ext.state.CookieProvider();这是一个标准地设定state provider的方式,我
45、们现在需要建立我们所需要存储的东西,我们选择存储节点的路径这可以使得用户方便地访问他/她所感兴趣的最末端的节点:tree.on(expandnode, function (node) Ext.state.Manager.set(“treestate, node.getPath(););在这段代码里,我们使用expandnode事件来记录路径使用TreeNode.getPath方法。这样treestate就包含了最后一次展开的节点。我们可以在页面加载的时候检查这个值:var treeState = Ext.state.Manager.get(“treestate);if (treeState)t
46、ree.expandPath(treeState);如果treestate之前被记录过,我们使用它来展开最后一次展开的节点。警告:需要强调的是,这个只是单纯地执行。它不会去管你在展开节点后是否合起来,因为合起节点的状态不会被保存。所以当你复原状态的时候,合起的节点又将展开。通过对collapsenode事件进行处理,我们可以解决这个问题。另外,对于展开多节点也存在着问题。如果很多节点同时展开,只会展开最近展开的一个。存储一个展开节点的数组是一个好的解决方法。总结:用十一行代码展示一个丰富的Ext.tree.TreePanel留给了我们很深的印象。本章还进一步演示了TreePanel的强大功能它为我们提供了丰富的配置选项。异步加载是一个很重要的功能,因为它提供了展示大量动态信息的一种方式。在Ext.tree中这个功能得到了很好的处理,这意味着无论对于开发者还是最终用户都能得到好处。尽管很强大,但是Ext.tree类仍然让人感觉是一种很轻量级的应用。使用配