用Vue.js递归组件构建一个可折叠的树形菜单

简介:

在Vue.js中一个递归组件调用的是其本身,如:


 Vue.component('recursive-component', {
 
   template: `<!--Invoking myself!-->
 
              <recursive-component></recursive-component>`
 
 });

递归组件常用于在blog上显示注释、嵌套的菜单,或者基本上是父和子相同的类型,尽管具体内容不同。例如:

现在给您演示一下如何有效地使用递归组件,我将通过建立一个可扩展/收缩的树形菜单的来一步步进行。

数据结构

一个树状UI的递归组件将是一些递归数据结构的可视化表达。在本教程中,我们将使用树状结构,其中每个节点都是一个对象:

  1. 一个 label 属性。

  2. 如果它有子节点,一个 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”的错误提示。

基本事件

与任何递归函数一样,你需要一个基本事件来结束递归,否则渲染将无限期地继续下去,最终会导致堆栈溢出。

在树菜单中,当我们到达一个没有子节点的节点的时候,我们希望停止递归。你能通过 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)可以看看我是如何实现它的。

50a40001b9f98a9a2810

原文发布时间为:2017年012月24日
原文作者:笔阁 

本文来源:开源中国 如需转载请联系原作者




目录
相关文章
|
7月前
|
JavaScript 前端开发 物联网
JavaScript:构建动态世界的引擎
JavaScript:构建动态世界的引擎
|
7月前
|
前端开发 JavaScript 开发者
JavaScript:构建动态网络的引擎
JavaScript:构建动态网络的引擎
|
7月前
|
JavaScript
Vue中如何实现兄弟组件之间的通信
在Vue中,兄弟组件可通过父组件中转、事件总线、Vuex/Pinia或provide/inject实现通信。小型项目推荐父组件中转或事件总线,大型项目建议使用Pinia等状态管理工具,确保数据流清晰可控,避免内存泄漏。
623 2
|
7月前
|
前端开发 JavaScript 开发者
JavaScript:构建动态Web的核心力量
JavaScript:构建动态Web的核心力量
|
11月前
|
前端开发 算法 API
构建高性能图像处理Web应用:Next.js与TailwindCSS实践
本文分享了构建在线图像黑白转换工具的技术实践,涵盖技术栈选择、架构设计与性能优化。项目采用Next.js提供优秀的SSR性能和SEO支持,TailwindCSS加速UI开发,WebAssembly实现高性能图像处理算法。通过渐进式处理、WebWorker隔离及内存管理等策略,解决大图像处理性能瓶颈,并确保跨浏览器兼容性和移动设备优化。实际应用案例展示了其即时处理、高质量输出和客户端隐私保护等特点。未来计划引入WebGPU加速、AI增强等功能,进一步提升用户体验。此技术栈为Web图像处理应用提供了高效可行的解决方案。
|
10月前
|
人工智能 JavaScript 算法
Vue 中 key 属性的深入解析:改变 key 导致组件销毁与重建
Vue 中 key 属性的深入解析:改变 key 导致组件销毁与重建
1060 0
|
10月前
|
JavaScript UED
用组件懒加载优化Vue应用性能
用组件懒加载优化Vue应用性能
|
12月前
|
前端开发 搜索推荐 JavaScript
如何通过DIY.JS快速构建出一个DIY手机壳、T恤的应用?
DIY.JS 是一款基于原生 Canvas 的业务级图形库,专注于商品定制的图形交互功能,帮助开发者轻松实现个性化设计。适用于 T 恤、手机壳等多种商品场景。它自带丰富功能,无需从零构建,快速集成到项目中。通过创建舞台、添加模型、定义 DIY 区域和添加素材四个步骤即可完成基础用法。支持在线演示体验,文档详细,易上手。
549 57
|
10月前
|
机器学习/深度学习 JavaScript 前端开发
JS进阶教程:递归函数原理与篇例解析
通过对这些代码示例的学习,我们已经了解了递归的原理以及递归在JS中的应用方法。递归虽然有着理论升华,但弄清它的核心思想并不难。举个随手可见的例子,火影鸣人做的影分身,你看到的都是同一个鸣人,但他们的行为却能在全局产生影响,这不就是递归吗?雾里看花,透过其间你或许已经深入了递归的魅力之中。
377 19
|
10月前
|
JavaScript 前端开发 UED
Vue 表情包输入组件的实现代码:支持自定义表情库、快捷键发送和输入框联动的聊天表情解决方案
本文详细介绍了在 Vue 项目中实现一个功能完善、交互友好的表情包输入组件的方法,并提供了具体的应用实例。组件设计包含表情分类展示、响应式布局、与输入框的交互及样式定制等功能。通过核心技术实现,如将表情插入输入框光标位置和点击外部关闭选择器,确保用户体验流畅。同时探讨了性能优化策略,如懒加载和虚拟滚动,以及扩展性方案,如自定义主题和国际化支持。最终,展示了如何在聊天界面中集成该组件,为用户提供丰富的表情输入体验。
742 8