当前位置:   article > 正文

vue3递归组件---树形组件_vue3 tree组件

vue3 tree组件

第一种方式,直接自己调用自己

Tree.vue

  1. <template>
  2. <div class="tree">
  3. <div v-for="(item, index) in data" :key="item.name">
  4. 每一层 {{ item.name }}
  5. <Tree v-if="item?.children?.length" :data='item.children' />
  6. </div>
  7. </div>
  8. </template>
  9. <script setup lang="ts">
  10. //递归的第一种方式 直接引入自己
  11. import Tree from './Tree.vue'
  12. import { TreeList } from '../ts/type'
  13. type Props = {
  14. data?: TreeList[]
  15. }
  16. defineProps<Props>();
  17. </script>
  18. <style scoped>
  19. .tree {
  20. margin-left: 20px;
  21. border-left: 2px #01847f dashed;
  22. }
  23. </style>

App.vue 里模拟树形数据,使用递归组件

  1. <template>
  2. <div>
  3. <Tree :data='data'/>
  4. </div>
  5. </template>
  6. <script setup lang="ts">
  7. import Tree from './components/Tree.vue'
  8. import { reactive } from 'vue'
  9. import {TreeList } from './ts/type'
  10. const data = reactive<TreeList[]>([
  11. {
  12. name: 'no.1',
  13. children: [
  14. {
  15. name: 'no.1-1',
  16. children: [
  17. {
  18. name: 'no.1-1-1',
  19. children:[]
  20. }
  21. ]
  22. },
  23. ],
  24. }, {
  25. name:'no.2'
  26. }, {
  27. name: 'no.3',
  28. children: [{
  29. name:'no.3-1'
  30. }]
  31. }
  32. ])
  33. </script>
  34. <style scoped></style>

type.ts 属性数据的结构

  1. export type TreeList = {
  2. name: string //名称
  3. icon?: string //图标可有可无
  4. children?: TreeList[] | [] //子节点 可有可无 还可能传空数组
  5. }

第二种方式,export 一个name出去

第二种方式 就是像vue2 一样 export一个name出去

但是setup 语法糖下没办法使用 export

我们只需要再定义一个script标签就可以了

  1. <template>
  2. <div class="tree">
  3. <div v-for="(item, index) in data" :key="item.name">
  4. 每一层 {{ item.name }}
  5. <Tree v-if="item?.children?.length" :data='item.children' />
  6. </div>
  7. </div>
  8. </template>
  9. <script setup lang="ts">
  10. import { TreeList } from '../ts/type'
  11. type Props = {
  12. data?: TreeList[]
  13. }
  14. defineProps<Props>();
  15. </script>
  16. <!--
  17. 第二种方式 就是像vue2 一样 export一个name出去
  18. 但是setup 语法糖下没办法使用 export
  19. 我们只需要再定义一个script标签就可以了
  20. -->
  21. <script lang="ts">
  22. export default {
  23. name:'Tree'
  24. }
  25. </script>
  26. <style scoped>
  27. .tree {
  28. margin-left: 20px;
  29. border-left: 2px #01847f dashed;
  30. }
  31. </style>

效果图

我们还可以给树形递归的组件添加参数传递事件

要注意在树形组件的里层也得添加自定义事件

并且这个自定义事件传的函数很有讲究

Tree.vue

  1. <template>
  2. <div class="tree">
  3. <div @click.stop="clickTreeItem(item)" v-for="(item, index) in data" :key="item.name">
  4. 每一层 {{ item.name }}
  5. <Tree @get-tree-item="clickTreeItem" v-if="item?.children?.length" :data='item.children' />
  6. <!-- Tree 组件不添加这个自定义事件的话 那么就只有最外层的根节点会向外传递数据 -->
  7. <!-- 注意此处派发的函数clickTreeItem没有传item参数了如果传了就相当于给树形组件(递归组件)的上级派发信息 没办法从外部拿到子节点所传递的数据了 -->
  8. <!-- @get-tree-item="clickTreeItem(item)" 写成这种形式的话 递归组件会依次向上层传递事件 -->
  9. <!-- 不传item的执行结果如下 -->
  10. <!--子组件派发的item Proxy {name: 'no.1-1-1', children: Array(0)}
  11. 子组件派发的item Proxy {name: 'no.1-1-1', children: Array(0)}
  12. 子组件派发的item Proxy {name: 'no.1-1-1', children: Array(0)}
  13. 父组件得到的item Proxy {name: 'no.1-1-1', children: Array(0)} -->
  14. <!-- 传item的执行的结果如下 -->
  15. <!-- 子组件派发的item Proxy {name: 'no.1-1-1', children: Array(0)}
  16. 子组件派发的item Proxy {name: 'no.1-1', children: Array(1)}
  17. 子组件派发的item Proxy {name: 'no.1', children: Array(1)}
  18. 父组件得到的item Proxy {name: 'no.1', children: Array(1)} -->
  19. </div>
  20. </div>
  21. </template>
  22. <script setup lang="ts">
  23. import { TreeList } from '../ts/type'
  24. type Props = {
  25. data?: TreeList[]
  26. }
  27. defineProps<Props>();
  28. const emit = defineEmits(['getTreeItem'])
  29. const clickTreeItem=(item:TreeList)=>{
  30. console.log('子组件派发的item', item)
  31. emit('getTreeItem',item)
  32. }
  33. </script>
  34. <!--
  35. 第二种方式 就是像vue2 一样 export一个name出去
  36. 但是setup 语法糖下没办法使用 export
  37. 我们只需要再定义一个script标签就可以了
  38. -->
  39. <script lang="ts">
  40. export default {
  41. name:'Tree'
  42. }
  43. </script>
  44. <style scoped>
  45. .tree {
  46. margin-left: 20px;
  47. border-left: 2px #01847f dashed;
  48. }
  49. </style>

App.vue

  1. <template>
  2. <div>
  3. <Tree :data='data' @get-tree-item="getTreeItem"/>
  4. </div>
  5. </template>
  6. <script setup lang="ts">
  7. import Tree from './components/Tree.vue'
  8. import { reactive } from 'vue'
  9. import {TreeList } from './ts/type'
  10. const data = reactive<TreeList[]>([
  11. {
  12. name: 'no.1',
  13. children: [
  14. {
  15. name: 'no.1-1',
  16. children: [
  17. {
  18. name: 'no.1-1-1',
  19. children:[]
  20. }
  21. ]
  22. },
  23. ],
  24. }, {
  25. name:'no.2'
  26. }, {
  27. name: 'no.3',
  28. children: [{
  29. name:'no.3-1'
  30. }]
  31. }
  32. ])
  33. const getTreeItem = (item:TreeList) => {
  34. console.log('父组件得到的item',item)
  35. }
  36. </script>
  37. <style scoped></style>

声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号