这一章咱们要实现是一个网格组件,该组件除了最基本的数据展现功能外,还提供排序以及数据过滤功能。css
为了测试咱们即将编写好网格组件,咱们采用以下格式的数据源。此数据源包含两部分的内容,分别是表头数据集和表体数据集。网格组件实例最终的列数由表头数据集的长度决定。app
// 10-01 var data = { gridColumns: ['name', 'power'], gridData: [ { name: 'Chuck Norris', power: Infinity }, { name: 'Bruce Lee', power: 9000 }, { name: 'Jackie Chan', power: 7000 }, { name: 'Jet Li', power: 8000 } ] };
从视觉上,咱们很天然地把网格组件划分为表头与表体。此网格组件有三个功能,因此应该提供三个动态接口。但咱们注意到排序功能是经过点击表头进行的,而表头属于网格组件的一部分,因此该功能应该内置。从而,实际上咱们的网格组件对外只暴露两个动态接口:一个用于过滤,另外一个用于接收数据源。因而咱们能够获得以下的一个顶层设计。函数
// 10-01 DataGrid: { xml: `<table id='datagrid'> <Thead id='thead'/> <Tbody id='tbody'/> </table>`, fun: function (sys, items, opts) { function setValue(data) { items.thead.val(data.gridColumns); items.tbody.val(data.gridColumns, data.gridData); } function filter(filterKey) { // 过滤函数 } return { val: setValue, filter: filter }; } }
表头只有一行,因此能够直接给它提供一个 tr 元素。tr 元素的子级项 th 元素的个数取决于表头数据集的长度,因此须要动态建立。因为 th 元素包含了排序功能,因此须要另行封装。下面是咱们给出的表头的设计。测试
// 10-01 Thead: { xml: `<thead id='thead'> <tr id='tr'/> </thead>`, fun: function (sys, items, opts) { return function (value) { sys.tr.children().call("remove"); data.forEach(item => sys.tr.append("Th").value().val(item)); }; } }
表头数据项组件提供一个文本设置接口。该组件自己并不负责排序,它只完成自身视图状态的变动以及排序命令的派发。排序命令的派发须要携带两个数据:一个是排序关键字,也就是表头文本;另外一个排序方向:升序或者降序。this
// 10-01 Th: { css: "#active { color: #fff; } #active #arrow { opacity: 1; } #active #key { color: #fff; }\ #arrow { display: inline-block; vertical-align: middle; width: 0; height: 0; margin-left: 5px; opacity: 0.66; }\ #asc { border-left: 4px solid transparent; border-right: 4px solid transparent; border-bottom: 4px solid #fff;}\ #dsc { border-left: 4px solid transparent; border-right: 4px solid transparent; border-top: 4px solid #fff; }", xml: "<th id='th'>\ <span id='key'/><span id='arrow'/>\ </th>", fun: function (sys, items, opts) { var order = "#asc"; this.watch("sort", function (e, key, order) { sys.key.text().toLowerCase() == key || sys.th.removeClass("#active"); }); this.on("click", function (e) { sys.th.addClass("#active"); sys.arrow.removeClass(order); order = order == "#asc" ? "#dsc" : "#asc"; sys.arrow.addClass(order).notify("sort", [sys.key.text().toLowerCase(), order]); }); sys.arrow.addClass("#asc"); return sys.key.text; } }
表体能够有多行,但表体只负责展现数据,因此实现起来比表头要简单的多。spa
// 10-01 Tbody: { xml: `<tbody id='tbody'/>`, fun: function (sys, items, opts) { return function (gridColumns, gridData) { sys.tbody.children().call("remove"); gridData.forEach(data => tr = sys.tbody.append("tr"); gridColumns.forEach(key => tr.append("td").text(data[key])); )); }; } }
此组件提供了一个接收数据源的动态接口,数据源须要包含两个部分:表头数据集与表体数据集。该动态接口根据这两个数据集完成数据的展现。设计
为了便于管理,咱们把排序功能单独封装成一个组件,该组件提供一个排序接口,同时侦听一个排序消息。一旦接收到排序消息,则记录下关键字与排序方向,并派发一个表体刷新命令。code
// 10-01 Sort: { fun: function (sys, items, opts) { var sortKey, sortOrder; this.watch("sort", function (e, key, order) { sortKey = key, sortOrder = order; this.trigger("update"); }); return function (data) { return sortKey ? data.slice().sort(function (a, b) { a = a[sortKey], b = b[sortKey]; return (a === b ? 0 : a > b ? 1 : -1) * (sortOrder == "#asc" ? 1 : -1); }) : data; }; } }
要完整地实现排序功能,对组件 DataGrid 做一些修正,主要是内置上述的排序功能组件并侦听表体刷新指令。一旦接收到刷新指令,则对表体数据完成排序并刷新表体。xml
// 10-01 DataGrid: { xml: `<table id='table'> <Thead id='thead'/> <Tbody id='tbody'/> <Sort id='sort'/> </table>`, fun: function (sys, items, opts) { var data = {gridColumns: [], gridData: []}; function setValue(value) { data = value; items.thead(data.gridColumns); items.tbody(data.gridColumns, data.gridData); } function filter(filterKey) { // 过滤函数 } this.on("update", function() { items.tbody(items.sort(data.gridData)); }); return { val: setValue, filter: filter }; } }
与排序功能的加入流程相似,咱们把过滤功能单独封装成一个组件,该组件提供一个过滤接口,同时侦听一个过滤消息。一旦接收到消息,则记录下过滤关键字,并派发一个表体刷新命令。blog
// 10-01 Filter: { fun: function (sys, items, opts) { var filterKey = ""; this.watch("filter", function (e, key) { filterKey = key.toLowerCase(); this.trigger("update"); }); return function (data) { return data.filter(function (row) { return Object.keys(row).some(function (key) { return String(row[key]).toLowerCase().indexOf(filterKey) > -1; }); }); }; } }
另外须要对组件 DataGrid 做一些修正,修正内容与上述的排序功能的加入相似,区别在于额外完善了 filter 接口以及对消息做用域进行了限制。下面是咱们最终的网格组件。
// 10-01 DataGrid: { css: `#table { border: 2px solid #42b983; border-radius: 3px; background-color: #fff; } #table th { background-color: #42b983; color: rgba(255,255,255,0.66); cursor: pointer; ... } #table td { background-color: #f9f9f9; } #table th, #table td { min-width: 120px; padding: 10px 20px; }`, xml: `<table id='table'> <Thead id='thead'/> <Tbody id='tbody'/> <Sort id='sort'/> <Filter id='filter'/> </table>`, map: { msgscope: true }, fun: function (sys, items, opts) { var data = {gridColumns: [], gridData: []}; function setValue(value) { data = value; items.thead(data.gridColumns); items.tbody(data.gridColumns, data.gridData); } function filter(filterKey) { sys.table.notify("filter", filterKey); } this.on("update", function() { items.tbody(data.gridColumns, items.filter(items.sort(data.gridData))); }); return { val: setValue, filter: filter }; } }
值得注意的是这里必定要在映射项中配置限制消息做用域的选项。不然,当在一个应用中实例化多个网格组件时,消息就会互相干扰。
最后咱们来测试下咱们完成的组件,测试的功能主要就是刚开始提到的三个:数据展现、排序以及过滤。
// 10-01 Index: { css: `#index { font-family: Helvetica Neue, Arial, sans-serif; font-size: 14px; color: #444; }, #search { margin: 8px 0; }` xml: `<div id='index' xmlns:i='datagrid'> Search <input id='search'/> <i:DataGrid id='datagrid'/> </div>`, fun: function (sys, items, opts) { items.datagrid.val({ gridColumns: ['name', 'power'], gridData: [ { name: 'Chuck Norris', power: Infinity }, { name: 'Bruce Lee', power: 9000 }, { name: 'Jackie Chan', power: 7000 }, { name: 'Jet Li', power: 8000 } ] }); sys.search.on("input", e => items.datagrid.filter(sys.search.prop("value"))); } }