xmlplus組件設(shè)計(jì)系列之樹(Tree)(9)
樹形組件是一種具有層級(jí)結(jié)構(gòu)的組件,廣泛應(yīng)用于各種場景。本章會(huì)實(shí)現(xiàn)一個(gè)簡單的樹形組件,盡管功能有限,但你可以通過擴(kuò)展它來實(shí)現(xiàn)自己所需要的樹形組件。
數(shù)據(jù)源
樹形組件的數(shù)據(jù)源可以是 JSON 格式的數(shù)據(jù)對(duì)象,也可以是具有 XML 結(jié)構(gòu)的數(shù)據(jù)或者是其它的具有層級(jí)結(jié)構(gòu)的數(shù)據(jù)。本章將采用具有如下 JSON 格式的數(shù)據(jù)對(duì)象。
var data = { name: 'My Tree', children: [ { name: 'hello' }, { name: 'world' }, { name: 'child folder', children: [ { name: 'alice' } ] } ] };
上述數(shù)據(jù)源中,name 值會(huì)作為樹結(jié)點(diǎn)的名稱顯示,含 children 的數(shù)組代表節(jié)點(diǎn)的子級(jí)。
遞歸結(jié)構(gòu)的設(shè)計(jì)
由 HTML 中的列表元素 ul 以及 li 組合而成對(duì)象具有天然的樹形結(jié)構(gòu),我們不妨采用它們作為構(gòu)建樹形組件的基本元素。樹形組件的最外層必然是一個(gè) ul 元素,所以我們可以暫時(shí)定義樹形組件如下:
Tree: { xml: `<ul id='tree'> <Item id='item'/> </ul>` }
這里的未定義的組件 Item 是一個(gè)需要遞歸定義的子項(xiàng)組件,它會(huì)根據(jù)提供的數(shù)據(jù)遞歸地生成子孫對(duì)象。下面是一種可能的設(shè)計(jì):
Item: { xml: `<li id='item'> <div id='content'> <span id='neme'/><span id='expand'/> </div> <ul id='entries'/> </li>`, map: { defer: "entries" } }
注意,上面的 neme 對(duì)象是用于顯示 name 屬性的。expand 對(duì)象用于展開或者關(guān)閉子級(jí)對(duì)象 entries。子級(jí)對(duì)象 entries 被設(shè)置為需要延遲實(shí)例化,只有當(dāng)用戶點(diǎn)擊 expand 對(duì)象展開子級(jí)時(shí),該對(duì)象才會(huì)實(shí)例化。
數(shù)據(jù)的加載與渲染
如上一節(jié)所述,我們?cè)O(shè)定了子級(jí)對(duì)象 entries 需要延遲實(shí)例化。所以,在給子項(xiàng) Item 提供的數(shù)據(jù)設(shè)置接口不應(yīng)該立馬對(duì) entries 實(shí)例化。下面我們先給出數(shù)據(jù)接口函數(shù)。
Item: { // css, xml, map 項(xiàng)同上 fun: function (sys, items, opts) { var data; function val(value) { data = value; sys.neme.text(data.name); data.children && data.children.length && sys.expand.show().text(" [+]"); } return { val: val }; } }
該接口函數(shù) val 只是設(shè)置了當(dāng)前節(jié)點(diǎn)相關(guān)的內(nèi)容。下面我們來偵聽 expand 對(duì)象的點(diǎn)擊事件,并適時(shí)地完成組件對(duì)象 entries 的實(shí)例化。
Item: { // css, xml, map 項(xiàng)同上 fun: function (sys, items, opts) { var data, open; sys.expand.on("click", function () { open = !open; sys.expand.text(open ? " [-]" : " [+]"); open ? (sys.entries.show() && load()) : sys.entries.hide(); }); function load() { if ( sys.entries.children().length == 0 ) for ( var item of data.children ) sys.add.before("Item").value().val(item); } function val(value) { data = value; sys.neme.text(data.name); data.children && data.children.length && sys.expand.show().text(" [+]"); } return { val: val }; } }
上述代碼中包含一個(gè) open 參數(shù),該參數(shù)記錄了當(dāng)前節(jié)點(diǎn)的是否處于展開狀態(tài)以供相關(guān)的偵聽器使用。
動(dòng)態(tài)添加節(jié)點(diǎn)
現(xiàn)在我們對(duì)上述組件進(jìn)行一個(gè)小的擴(kuò)展,使得它支持動(dòng)態(tài)添加樹節(jié)點(diǎn)的功能。首先,我們?cè)趯?duì)象 entries 的子級(jí)添加一個(gè)觸發(fā)按鈕,并命名為 add。
Item: { xml: "<li id='item'> <div id='content'> <span id='neme'/><span id='expand'/> </div> <ul id='entries'> <li id='add'>+</li> </ul> </li>", map: { defer: "entries" } }
其次,需要偵聽 add 對(duì)象的點(diǎn)擊事件,在偵聽器中完成對(duì)象的添加。
Item: { // css, xml, map 項(xiàng)同前 fun: function (sys, items, opts) { var data, open; sys.item.on("click", "http://*[@id='add']", function () { var stuff = {name: 'new stuff'}; data.children.push(stuff); sys.add.before("Item").value().val(stuff); }); // 其余代碼同前 } }
這里需要注意,對(duì) add 對(duì)象的偵聽不能采取直接式的偵聽:sys.add.on("click",...),而應(yīng)該使用代理的方式,否則會(huì)報(bào)錯(cuò)。因?yàn)槠涓讣?jí)屬于延遲實(shí)例化的組件,在 entries 對(duì)象未實(shí)例化之間,add 對(duì)象并不可見。
完整的樹形組件
綜合以上的內(nèi)容,現(xiàn)在給出一個(gè)完整版本的樹形組件:
Tree: { css: `#tree { font-family: Menlo, Consolas, monospace; color: #444; } #tree, #tree ul { padding-left: 1em; line-height: 1.5em; list-style-type: dot; }`, xml: `<ul id='tree'> <Item id='item'/> </ul>`, fun: function (sys, items, opts) { return items.item; } }, Item: { css: "#item { cursor: pointer; }", xml: `<li id='item'> <div id='content'> <span id='neme'/><span id='expand'/> </div> <ul id='entries'> <li id='add'>+</li> </ul> </li>`, map: { defer: "entries" }, fun: function (sys, items, opts) { var data, open; sys.item.on("click", "http://*[@id='add']", function () { var stuff = {name: 'new stuff'}; data.children.push(stuff); sys.add.before("Item").value().val(stuff); }); sys.expand.on("click", function () { open = !open; sys.expand.text(open ? " [-]" : " [+]"); open ? (sys.entries.show() && load()) : sys.entries.hide(); }); function load() { if ( sys.entries.children().length == 1 ) for ( var item of data.children ) sys.add.before("Item").value().val(item); } function val(value) { data = value; sys.neme.text(data.name); data.children && data.children.length && sys.expand.show().text(" [+]"); } return { val: val }; } }
在實(shí)際應(yīng)用中的樹形組件會(huì)比這里的功能更豐富些,你可以在上述代碼的基礎(chǔ)上進(jìn)一步的改進(jìn),比如添加些 ICON 圖標(biāo)、讓子項(xiàng)成為可拖動(dòng)的等等。但在改進(jìn)過程中盡量避免代碼的復(fù)雜化,適當(dāng)?shù)貏冸x些代碼出來封裝成組件是非常有必要的。
本系列文章基于 xmlplus 框架。如果你對(duì) xmlplus 沒有多少了解,可以訪問 www.xmlplus.cn。這里有詳盡的入門文檔可供參考。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
iframe自適應(yīng)寬度、高度 ie6 7 8,firefox 3.86下測試通過
近期需要一個(gè)iframe自適應(yīng)高度的東西,在網(wǎng)上找了很多,都不能用……一看大體的日期都是大概 2008年前后的其他近期的基本都是以前的轉(zhuǎn)載,所以只好自己動(dòng)手了。2010-07-07JavaScript面試出現(xiàn)頻繁的一些易錯(cuò)點(diǎn)整理
通過幾個(gè)常見面試開始,討論針對(duì)一個(gè)題目的分析思路,就有了下面這篇文章,本文主要給大家整理總結(jié)介紹了關(guān)于JavaScript面試中會(huì)頻繁出現(xiàn)的一些易錯(cuò)點(diǎn),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起看看吧。2018-03-03javascript控制Div層透明屬性由淺變深由深變淺逐漸顯示
控制Div層透明屬性,由淺變深逐漸顯示,由深變淺逐漸消失,具體實(shí)現(xiàn)代碼如下,喜歡的朋友可以感受下2013-11-11JavaScript實(shí)現(xiàn)頁面高亮操作提示和蒙板
這篇文章主要介紹了JavaScript實(shí)現(xiàn)頁面高亮操作提示和蒙板,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-01-01javascript 判斷數(shù)組是否已包含了某個(gè)元素的函數(shù)
javascript判斷數(shù)組是否已包含了某個(gè)元素的js函數(shù),方便數(shù)組的判斷。2010-05-05