在Vue.js中一个递归组件调用的是其自己,如:vue
Vue.component('recursive-component', { template: `<!--Invoking myself!--> <recursive-component></recursive-component>` });
递归组件经常使用于在blog上显示注释、嵌套的菜单,或者基本上是父和子相同的类型,尽管具体内容不一样。例如:node
如今给您演示一下如何有效地使用递归组件,我将经过创建一个可扩展/收缩的树形菜单的来一步步进行。数组
一个树状UI的递归组件将是一些递归数据结构的可视化表达。在本教程中,咱们将使用树状结构,其中每一个节点都是一个对象:数据结构
一个 label 属性。app
若是它有子节点,一个 nodes 属性,则它是一个或多个节点的数组属性。函数
与全部树结构同样,它必须有一个根节点,但能够无限深。性能
let tree = { label: 'root', nodes: [ { label: 'item1', nodes: [ { label: 'item1.1' }, { label: 'item1.2', nodes: [ { label: 'item1.2.1' } ] } ] }, { label: 'item2' } ] }
让咱们作一个递归组件来显示咱们的称为 TreeMenu 的数据结构。它只显示当前节点的标签,并调用本身来显示任何子节点。文件名:TreeMenu.vue,内容以下:字体
<template> <div class="tree-menu"> <div>{{ label }}</div> <tree-menu v-for="node in nodes" :nodes="node.nodes" :label="node.label" > </tree-menu> </div> </template> <script> export default { props: [ 'label', 'nodes' ], name: 'tree-menu' } </script>
若是你使用一个组件递归,必须先给 Vue.component 作一个全局的定义,或者,给它一个 name 属性。不然,任何子组件将没法进一步调用它,你会获得一个不肯定的“undefined component error”的错误提示。this
与任何递归函数同样,你须要一个基本事件来结束递归,不然渲染将无限期地继续下去,最终会致使堆栈溢出。翻译
在树菜单中,当咱们到达一个没有子节点的节点的时候,咱们但愿中止递归。你能经过 v-if 作到这一功能,但咱们选择使用 v-for 将隐式地为咱们实现它;若是 nodes 数组没有任何进一步的定义 tree-menu 组件将被调用。template.vue文件以下:
<template> <div class="tree-menu"> ... <!--If `nodes` is undefined this will not render--> <tree-menu v-for="node in nodes"></tree-menu> </template>
咱们如今如何使用这个组件?首先,咱们声明一个Vue实例,具备一个数据结构包括data属性和定义过的treemenu组件。app.js文件以下:
import TreeMenu from './TreeMenu.vue' let tree = { ... } new Vue({ el: '#app', data: { tree }, components: { TreeMenu } })
请记住,咱们的数据结构有一个根节点。咱们在主模板开始递归调用 TreeMenu 组件,使用根 nodes 属性来props:
<div id="app"> <tree-menu :label="tree.label" :nodes="tree.nodes"></tree-menu> </div>
下面是它目前的样子:
在视觉上识别子组件的“深度”是很好的,这样用户就能够从UI中得到数据结构的感受。让咱们缩进每一层的子节点来实现这个目标。
这是经过增长一个depth prop定义,经过 TreeMenu 来实现。咱们将使用这个值动态地将内联样式与转换绑定在一块儿:将使用transform: translate的CSS规则为每一个节点的标签,从而建立缩进。template.vue修改以下**:**
<template> <div class="tree-menu"> <div :style="indent">{{ label }}</div> <tree-menu v-for="node in nodes" :nodes="node.nodes" :label="node.label" :depth="depth + 1" > </tree-menu> </div> </template> <script> export default { props: [ 'label', 'nodes', 'depth' ], name: 'tree-menu', computed: { indent() { return { transform: `translate(${this.depth * 50}px)` } } } } </script>
depth 属性在主模板中从零开始。在上面的组件模板中,你能够看到每次传递到任何子节点时这个值都会递增。
<div id="app"> <tree-menu :label="tree.label" :nodes="tree.nodes" :depth="0" ></tree-menu> </div>
注意:记得 v-bind depth值以确保它是一个JavaScript数字类型而不是字符串。
因为递归数据结构可能很大,因此显示它们的一个很好的UI技巧是隐藏除根节点之外的全部节点,以便用户能够根据须要展开或收起节点。
为此,咱们将增长一个局部属性showChildren 。若是他的值为False,子节点将不会被渲染。此值应经过点击节点切换,因此咱们须要使用一个单击事件的监听器方法 toggleChildren 来进行管理。template.vue文件修改以下**:**
<template> <div class="tree-menu"> <div :style="indent" @click="toggleChildren">{{ label }}</div> <tree-menu v-if="showChildren" v-for="node in nodes" :nodes="node.nodes" :label="node.label" :depth="depth + 1" > </tree-menu> </div> </template> <script> export default { props: [ 'label', 'nodes', 'depth' ], data() { return { showChildren: false } }, name: 'tree-menu', computed: { indent() { return { transform: `translate(${this.depth * 50}px)` } } }, methods: { toggleChildren() { this.showChildren = !this.showChildren; } } } </script
这样,咱们就有了一个工做树菜单。用来画龙点睛的一个方法是,你能够添加一个加号/减号图标,这样可使UI的显示更加明显。我还增长了的很好的字体和计算性能在原来 showChildren 的基础上。
去CodePen(https://codepen.io/anthonygore/pen/PJKNqa)能够看看我是如何实现它的。
来自汇智网(www.hubwiz.com,有不少性价比极高的vue.js内容哦)的小智翻译。