在 vue.js 中一个递归组件调用的是其本身,如:
- vue.component('recursive-component', {
- template: ` < !--Invoking myself ! -->
- <recursive - component > </recursive-component>`
- });/
递归组件常用于在 blog 上显示注释、嵌套的菜单,或者基本上是父和子相同的类型,尽管具体内容不同。例如:
现在给您演示一下如何有效地使用递归组件,我将通过建立一个可扩展 / 收缩的树形菜单的来一步步进行。
一个树状 UI 的递归组件将是一些递归数据结构的可视化表达。在本教程中,我们将使用树状结构,其中每个节点都是一个对象:
与所有树结构一样,它必须有一个根节点,但可以无限深。
- 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" 的错误提示。
与任何递归函数一样,你需要一个基本事件来结束递归,否则渲染将无限期地继续下去,最终会导致堆栈溢出。
在树菜单中,当我们到达一个没有子节点的节点的时候,我们希望停止递归。你能通过 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 内容哦)的小智翻译。
来源: http://www.jianshu.com/p/684b30f0b462