有没有可能用角度来做一个树图?

我希望在一个网络应用程序中以树形结构显示数据。我希望用 Angular 来完成这个任务。

看起来 ng-repeat 允许我迭代一个节点列表,但是当给定节点的深度增加时,我怎样才能进行嵌套呢?

我尝试了 以下代码,但 HTML 的自动转义阻止了它的工作。另外,结束 ul 标签放错了地方。

我很确定我对这个问题的处理方式完全错了。

有什么想法吗?

213173 次浏览

看看这把小提琴

原创: http://jsfiddle.net/brendanowen/uXbn6/8/

更新: http://jsfiddle.net/animaxf/uXbn6/4779/

这应该给你一个很好的想法,如何显示一个 tree like structure使用角度。这是一种在 html 中使用递归!

这个看起来更完整一些: https://github.com/dump247/angular.tree

另一个基于 原始来源的例子,有一个已经到位的样本树结构(更容易看到它是如何工作的 IMO)和一个搜索树的过滤器:

JSFiddle

下面是一个使用递归指令的示例: < a href = “ http://jsfiddle.net/n8dPm/”rel = “ norefrer”> http://jsfiddle.net/n8dpm/ 摘自 http://groups.google.com/forum/# ! subject/angle/vswXTes _ ftM rel = “ noReferrer”> https://groups.google.com/forum/#!topic/angular/vswxtes_ftm

module.directive("tree", function($compile) {
return {
restrict: "E",
scope: {family: '='},
template:
'<p>\{\{ family.name }}</p>'+
'<ul>' +
'<li ng-repeat="child in family.children">' +
'<tree family="child"></tree>' +
'</li>' +
'</ul>',
compile: function(tElement, tAttr) {
var contents = tElement.contents().remove();
var compiledContents;
return function(scope, iElement, iAttr) {
if(!compiledContents) {
compiledContents = $compile(contents);
}
compiledContents(scope, function(clone, scope) {
iElement.append(clone);
});
};
}
};
});

如果您正在使用 Bootstrap CSS..。

我已经为 AngularJS 创建了一个简单的基于 Bootstrap“ nav”列表的可重用树控件(指令)。我添加了额外的缩进、图标和动画。HTML 属性用于配置。

它不使用递归。

我叫它 角引导导航树(朗朗上口的名字,你不觉得吗?)

这里有一个例子 给你,源代码是 给你

当制作这样的东西时,最好的解决方案是使用递归指令。然而,当你做出这样的指令时,你会发现 AngularJS 陷入了一个无止境的循环。

解决方案是让指令在编译事件期间删除元素,然后手动编译并在链接事件中添加它们。

我在 这根线中发现了这一点,并抽象出了这个功能 变成了一种服务

module.factory('RecursionHelper', ['$compile', function($compile){
return {
/**
* Manually compiles the element, fixing the recursion loop.
* @param element
* @param [link] A post-link function, or an object with function(s) registered via pre and post properties.
* @returns An object containing the linking functions.
*/
compile: function(element, link){
// Normalize the link parameter
if(angular.isFunction(link)){
link = { post: link };
}


// Break the recursion loop by removing the contents
var contents = element.contents().remove();
var compiledContents;
return {
pre: (link && link.pre) ? link.pre : null,
/**
* Compiles and re-adds the contents
*/
post: function(scope, element){
// Compile the contents
if(!compiledContents){
compiledContents = $compile(contents);
}
// Re-add the compiled contents to the element
compiledContents(scope, function(clone){
element.append(clone);
});


// Call the post-linking function, if any
if(link && link.post){
link.post.apply(null, arguments);
}
}
};
}
};
}]);

使用这个服务,您可以很容易地创建一个树指令(或其他递归指令)。下面是一个树指令的例子:

module.directive("tree", function(RecursionHelper) {
return {
restrict: "E",
scope: {family: '='},
template:
'<p>\{\{ family.name }}</p>'+
'<ul>' +
'<li ng-repeat="child in family.children">' +
'<tree family="child"></tree>' +
'</li>' +
'</ul>',
compile: function(element) {
return RecursionHelper.compile(element);
}
};
});

请参见此 笨蛋演示。 我最喜欢这个解决方案,因为:

  1. 您不需要一个特殊的指令来使您的 html 不那么干净。
  2. 递归逻辑被抽象到 RecursionHelper 服务中,因此您可以保持指令的整洁。

更新: 增加了对自定义链接函数的支持。

这么多伟大的解决方案,但我觉得他们都在某种程度上过于复杂的东西有点。

我想创建一些东西,重新创建@Mark Lagendjk 的 awnser 的简单性,但没有在指令中定义模板,而是让“用户”用 HTML 创建模板..。

随着想法从 https://github.com/stackfull/angular-tree-repeat等。我最终创建的项目: https://github.com/dotJEM/angular-tree

这样你就可以建立你的树了,比如:

<ul dx-start-with="rootNode">
<li ng-repeat="node in $dxPrior.nodes">
\{\{ node.name }}
<ul dx-connect="node"/>
</li>
</ul>

对我来说,这比为不同结构的树创建多个指令更简单... ..。从本质上说,将上面的命名为树是有点错误的,它从@ganaraj 的“递归模板”的 awnser 中挑选了更多的东西,但是允许我们在需要树的地方定义模板。

(您可以使用基于脚本标记的模板来做到这一点,但是它仍然必须位于实际的树节点之外,而且仍然感觉有点恶心...)

离开这里只是为了另一个选择。

您可以尝试与 角树 DnD样本与角-Ui-树,但我编辑,与表格,网格,列表兼容。

  • 能拖放
  • 列表 (下一个,前情提要,getChildren,...)的扩展函数指令
  • 过滤数据。
  • OrderBy (ver)

是的,绝对有可能。这里的问题可能假设角度为1.x,但是为了以后的参考,我将包括一个角度为2的例子:

从概念上讲,您所要做的就是创建一个递归模板:

<ul>
<li *for="#dir of directories">


<span><input type="checkbox" [checked]="dir.checked" (click)="dir.check()"    /></span>
<span (click)="dir.toggle()">\{\{ dir.name }}</span>


<div *if="dir.expanded">
<ul *for="#file of dir.files">
\{\{file}}
</ul>
<tree-view [directories]="dir.directories"></tree-view>
</div>
</li>
</ul>

然后将一个树对象绑定到模板上,让 Angular 发挥它的魔力。这个概念显然也适用于角度1.x。

下面是一个完整的例子: http://www.syntaxsuccess.com/viewarticle/recursive-treeview-in-angular-2.0

可以使用角递归注入器: https://github.com/knyga/angular-recursion-injector

允许您做无限深度嵌套与条件。仅在需要时进行重新编译,并仅编译正确的元素。密码里没有魔法。

<div class="node">
<span>\{\{name}}</span>


<node--recursion recursion-if="subNode" ng-model="subNode"></node--recursion>
</div>

允许它比其他解决方案工作得更快、更简单的一个因素是“——递归”后缀。

当树结构很大时,Angular (高达1.4.x)在呈现递归模板时会变得非常慢。在尝试了这些建议之后,我最终创建了一个简单的 HTML 字符串,并使用 ng-bind-html来显示它。当然,这不是使用角特征的方法

这里显示了一个基本的递归函数(使用最少的 HTML) :

function menu_tree(menu, prefix) {
var html = '<div>' + prefix + menu.menu_name + ' - ' + menu.menu_desc + '</div>\n';
if (!menu.items) return html;
prefix += menu.menu_name + '/';
for (var i=0; i<menu.items.length; ++i) {
var item = menu.items[i];
html += menu_tree(item, prefix);
}
return html;
}
// Generate the tree view and tell Angular to trust this HTML
$scope.html_menu = $sce.trustAsHtml(menu_tree(menu, ''));

在模板中,它只需要这一行:

<div ng-bind-html="html_menu"></div>

这绕过了 Angular 的所有数据绑定,仅仅用递归模板方法的一小部分时间来显示 HTML。

使用这样的菜单结构(Linux 文件系统的部分文件树) :

menu = {menu_name: '', menu_desc: 'root', items: [
{menu_name: 'bin', menu_desc: 'Essential command binaries', items: [
{menu_name: 'arch', menu_desc: 'print machine architecture'},
{menu_name: 'bash', menu_desc: 'GNU Bourne-Again SHell'},
{menu_name: 'cat', menu_desc: 'concatenate and print files'},
{menu_name: 'date', menu_desc: 'display or set date and time'},
{menu_name: '...', menu_desc: 'other files'}
]},
{menu_name: 'boot', menu_desc: 'Static files of the boot loader'},
{menu_name: 'dev', menu_desc: 'Device files'},
{menu_name: 'etc', menu_desc: 'Host-specific system configuration'},
{menu_name: 'lib', menu_desc: 'Essential shared libraries and kernel modules'},
{menu_name: 'media', menu_desc: 'Mount point for removable media'},
{menu_name: 'mnt', menu_desc: 'Mount point for mounting a filesystem temporarily'},
{menu_name: 'opt', menu_desc: 'Add-on application software packages'},
{menu_name: 'sbin', menu_desc: 'Essential system binaries'},
{menu_name: 'srv', menu_desc: 'Data for services provided by this system'},
{menu_name: 'tmp', menu_desc: 'Temporary files'},
{menu_name: 'usr', menu_desc: 'Secondary hierarchy', items: [
{menu_name: 'bin', menu_desc: 'user utilities and applications'},
{menu_name: 'include', menu_desc: ''},
{menu_name: 'local', menu_desc: '', items: [
{menu_name: 'bin', menu_desc: 'local user binaries'},
{menu_name: 'games', menu_desc: 'local user games'}
]},
{menu_name: 'sbin', menu_desc: ''},
{menu_name: 'share', menu_desc: ''},
{menu_name: '...', menu_desc: 'other files'}
]},
{menu_name: 'var', menu_desc: 'Variable data'}
]
}

输出结果是:

- root
/bin - Essential command binaries
/bin/arch - print machine architecture
/bin/bash - GNU Bourne-Again SHell
/bin/cat - concatenate and print files
/bin/date - display or set date and time
/bin/... - other files
/boot - Static files of the boot loader
/dev - Device files
/etc - Host-specific system configuration
/lib - Essential shared libraries and kernel modules
/media - Mount point for removable media
/mnt - Mount point for mounting a filesystem temporarily
/opt - Add-on application software packages
/sbin - Essential system binaries
/srv - Data for services provided by this system
/tmp - Temporary files
/usr - Secondary hierarchy
/usr/bin - user utilities and applications
/usr/include -
/usr/local -
/usr/local/bin - local user binaries
/usr/local/games - local user games
/usr/sbin -
/usr/share -
/usr/... - other files
/var - Variable data

一点都不复杂。

<div ng-app="Application" ng-controller="TreeController">
<table>
<thead>
<tr>
<th>col 1</th>
<th>col 2</th>
<th>col 3</th>
</tr>
</thead>
<tbody ng-repeat="item in tree">
<tr>
<td>\{\{item.id}}</td>
<td>\{\{item.fname}}</td>
<td>\{\{item.lname}}</td>
</tr>
<tr ng-repeat="children in item.child">
<td style="padding-left:15px;">\{\{children.id}}</td>
<td>\{\{children.fname}}</td>
</tr>
</tbody>
</table>
</div>

控制器代码:

angular.module("myApp", []).
controller("TreeController", ['$scope', function ($scope) {
$scope.tree = [{
id: 1,
fname: "tree",
child: [{
id: 1,
fname: "example"
}],
lname: "grid"
}];




}]);

基于@ganaraj 的 回答和@dnc253的 回答,我为具有选择、添加、删除和编辑功能的树结构制作了一个简单的“指令”。

http://jsfiddle.net/yoshiokatsuneo/9dzsms7y/

HTML:

<script type="text/ng-template" id="tree_item_renderer.html">
<div class="node"  ng-class="{selected: data.selected}" ng-click="select(data)">
<span ng-click="data.hide=!data.hide" style="display:inline-block; width:10px;">
<span ng-show="data.hide && data.nodes.length > 0" class="fa fa-caret-right">+</span>
<span ng-show="!data.hide && data.nodes.length > 0" class="fa fa-caret-down">-</span>
</span>
<span ng-show="!data.editting" ng-dblclick="edit($event)" >\{\{data.name}}</span>
<span ng-show="data.editting"><input ng-model="data.name" ng-blur="unedit()" ng-focus="f()"></input></span>
<button ng-click="add(data)">Add node</button>
<button ng-click="delete(data)" ng-show="data.parent">Delete node</button>
</div>
<ul ng-show="!data.hide" style="list-style-type: none; padding-left: 15px">
<li ng-repeat="data in data.nodes">
<recursive><sub-tree data="data"></sub-tree></recursive>
</li>
</ul>
</script>
<ul ng-app="Application" style="list-style-type: none; padding-left: 0">
<tree data='{name: "Node", nodes: [],show:true}'></tree>
</ul>

JavaScript:

angular.module("myApp",[]);


/* https://stackoverflow.com/a/14657310/1309218 */
angular.module("myApp").
directive("recursive", function($compile) {
return {
restrict: "EACM",
require: '^tree',
priority: 100000,


compile: function(tElement, tAttr) {
var contents = tElement.contents().remove();
var compiledContents;
return function(scope, iElement, iAttr) {
if(!compiledContents) {
compiledContents = $compile(contents);
}
compiledContents(scope,
function(clone) {
iElement.append(clone);
});
};
}
};
});


angular.module("myApp").
directive("subTree", function($timeout) {
return {
restrict: 'EA',
require: '^tree',
templateUrl: 'tree_item_renderer.html',
scope: {
data: '=',
},
link: function(scope, element, attrs, treeCtrl) {
scope.select = function(){
treeCtrl.select(scope.data);
};
scope.delete = function() {
scope.data.parent.nodes.splice(scope.data.parent.nodes.indexOf(scope.data), 1);
};
scope.add = function() {
var post = scope.data.nodes.length + 1;
var newName = scope.data.name + '-' + post;
scope.data.nodes.push({name: newName,nodes: [],show:true, parent: scope.data});
};
scope.edit = function(event){
scope.data.editting = true;
$timeout(function(){event.target.parentNode.querySelector('input').focus();});
};
scope.unedit = function(){
scope.data.editting = false;
};


}
};
});




angular.module("myApp").
directive("tree", function(){
return {
restrict: 'EA',
template: '<sub-tree data="data" root="data"></sub-tree>',
controller: function($scope){
this.select = function(data){
if($scope.selected){
$scope.selected.selected = false;
}
data.selected = true;
$scope.selected = data;
};
},
scope: {
data: '=',
}
}
});