Vue实现封装树状图的示例代码

 更新时间:2024年03月08日 11:33:35   作者:98年超神级前端开发工程师  
这篇文章主要为大家详细介绍了如何使用Vue实现封装树状图,文中的示例代码讲解详细,具有一定的借鉴价值,感兴趣的小伙伴可以跟随小编一起学习一下

封装准备 完整代码可直接置底查看

组件的封装来源于数据

我们把树状图(下文统称为“图”)的每一个节点作为一个看作一个对象,它将需要的节点属性如下:

属性名称作用
id唯一标识
title节点名称
children子集
expend展开控制变量
//数据准备
let treeData = [
    {
        id: 'treeRoot',
        title: '目录根节点',
        expend: false,
        children: [
            {
                id: 'subNode',
                title: '目录子节点',
            }
        ],
    },
];

开始封装

我们确认了图的数据是一个可以无限递归的层级数据,所以我们首先需要封装一个方法来遍历数据。

//树状图数据遍历方法
function TreeNode(arr) {
    return arr.reduce((pre, cur) => {
        if (cur.children && Array.isArray(cur.children)) {
            cur.children = TreeNode(cur.children);
        }
        pre.push(cur);
        return pre;
    }, []);
}

其次是组件内部结构,我们需要用到递归组件,所谓的递归组件就是在vue单文件内部给予name属性,我们就可以在该文件内直接调用组件。

<template>
    <div class="tree">
        <div class="tree-item" v-for="item in treeData" :key="item.id">
            <div>{{ item.title }}</div>
            <Tree :treeData="item.children || []" />
        </div>
    </div>
</template>

<script>
export default {
    name: 'Tree',
    props: {
        treeData: {
            default: [
                {
                    id: 'treeRoot',
                    title: '目录根节点',
                    expend: false,
                    children: [
                        {
                            id: 'subNode',
                            title: '目录子节点',
                        },
                    ],
                },
            ],
        },
    },
};
</script>

视图效果

接下来我们为每一级节点添加缩进,我们定义一个初始值pad,当用户调用时未传pad,此时采用默认值,pad为0,接下来每一级逐级+1传递,即可实现节点缩进效果。

<Tree :treeData="item.children" :pad="pad+1">

pad: {
    type: Number,
    default: 0,
},

当数据多出之后,我们需要为图添加展开/收起功能,用expend控制Tree 展示隐藏即可

<Tree :treeData='item.children' v-if="item.expend" :pad="pad+1">

我们可以添加一些图标或者标签来当作操作按钮,这里不予多做解释。至于不定高度的展开收起动画,我采用grid的template-rows:1fr/0fr;来实现动画的高度变化。

.tree-sub {
    display: grid;
    transition: all 0.3s;
}
.tree-item-expend {
    grid-template-rows: 1fr;
}
.tree-item-hidden {
    grid-template-rows: 0fr;
}

此后,树状图的已具有初步效果,我们还需要再继续进行一些细微之处的修改。各位同学可以通过CV查看具体效果或继续进行封装修改。

完整代码

<template>
    <div class="tree">
        <div class="tree-item" v-for="item in treeData" :key="item.id">
            <div class="tree-item-row" :style="`padding-left:${pad}rem`">
                <div class="tree-item-row-opr" @click.stop="item.expend = !item.expend" v-if="item.children && Array.isArray(item.children)">
                    {{ item.expend ? '-' : '+' }}
                </div>
                <div class="tree-item-row-opr-none" v-else></div>
                <div class="tree-item-row-text">{{ item.title }}</div>
            </div>
            <div
                class="tree-sub"
                :class="{
                    'tree-item-expend': item.expend,
                    'tree-item-hidden': !item.expend,
                }"
            >
                <Tree :treeData="item.children || []" :pad="pad + 1" />
            </div>
        </div>
    </div>
</template>

<script>
export default {
    name: 'Tree',
    props: {
        treeData: {
            Type: Array,
            default: () => [
                {
                    id: 'treeRoot',
                    title: '目录根节点',
                    expend: false,
                    children: [
                        {
                            id: 'subNode1',
                            title: '目录子节点1',
                            expend: false,
                            children: [
                                {
                                    id: 'subNode11',
                                    title: '目录子节点1-1',
                                },
                                {
                                    id: 'subNode12',
                                    title: '目录子节点1-2',
                                },
                                {
                                    id: 'subNode13',
                                    title: '目录子节点1-3',
                                },
                                {
                                    id: 'subNode14',
                                    title: '目录子节点1-4',
                                },
                            ],
                        },
                        {
                            id: 'subNode2',
                            title: '目录子节点2',
                        },
                        {
                            id: 'subNode3',
                            title: '目录子节点3',
                        },
                        {
                            id: 'subNode4',
                            title: '目录子节点4',
                        },
                        {
                            id: 'subNode5',
                            title: '目录子节点5',
                        },
                        {
                            id: 'subNode6',
                            title: '目录子节点6',
                        },
                    ],
                },
            ],
        },
        pad: {
            type: Number,
            default: 0,
        },
    },
};
</script>

<style lang="less">
.tree {
    box-sizing: border-box;
    overflow: hidden;
    &-item {
        display: flex;
        flex-direction: column;
        gap: 4px;
        &-row {
            display: flex;
            gap: 4px;
            &-opr {
                cursor: pointer;
                width: 16px;
                font-size: 16px;
                height: 20px;
                line-height: 20px;
                text-align: center;
            }
            &-opr-none {
                cursor: default;
            }
        }
        .tree-sub {
            display: grid;
            transition: all 0.3s;
        }
        .tree-item-expend {
            grid-template-rows: 1fr;
        }
        .tree-item-hidden {
            grid-template-rows: 0fr;
        }
    }
}
</style>

组件封装为系列作品,树状图组件远远不止于此,开发还有特定需求,如关键字检索,目录检索,多选,单选,懒加载,标题超长缩略提示等。

到此这篇关于Vue实现封装树状图的示例代码的文章就介绍到这了,更多相关Vue封装树状图内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解vue移动端项目的适配(以mint-ui为例)

    详解vue移动端项目的适配(以mint-ui为例)

    这篇文章主要介绍了详解vue移动端项目的适配(以mint-ui为例),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-08-08
  • vue+element-plus上传图片及回显问题及数量限制

    vue+element-plus上传图片及回显问题及数量限制

    本文主要介绍了vue+element-plus上传图片及回显问题及数量限制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • elementui之封装下载模板和导入文件组件方式

    elementui之封装下载模板和导入文件组件方式

    这篇文章主要介绍了关于elementui之封装下载模板和导入文件组件方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-12-12
  • 如何使用 Vuex的入门教程

    如何使用 Vuex的入门教程

    在vue中当我们管理数据的时候比较乱,我们要用到下面的这个库,本文主要介绍了如何使用 Vuex的入门教程,具有一定的参考价值,感兴趣的可以了解一下
    2022-02-02
  • VUE v-for中的:key详解

    VUE v-for中的:key详解

    这篇文章主要为大家介绍了v-for中的:key,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2021-11-11
  • Vue3响应式对象是如何实现的(1)

    Vue3响应式对象是如何实现的(1)

    这篇文章主要介绍了Vue3响应式对象是如何实现的,文章围绕主题展开详细的内容介绍具有一定的参考价值,需要的小伙伴可以参考一下
    2022-08-08
  • vue3.2新增指令v-memo的基本使用教程

    vue3.2新增指令v-memo的基本使用教程

    ue3.2新增了一个指令v-memo,引入这个指令的目的是帮助大家更好的为我们的应用做性能优化,下面这篇文章主要给大家介绍了关于vue3.2新增指令v-memo基本使用的相关资料,需要的朋友可以参考下
    2022-09-09
  • Vue源码学习defineProperty响应式数据原理实现

    Vue源码学习defineProperty响应式数据原理实现

    这篇文章主要为大家介绍了Vue源码学习defineProperty响应式数据原理实现,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-09-09
  • Vue核心概念Getter的使用方法

    Vue核心概念Getter的使用方法

    今天小编就为大家分享一篇关于Vue核心概念Getter的使用方法,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • Vue非单文件组件使用详解

    Vue非单文件组件使用详解

    Vue单文件组件通常使用.vue文件扩展名,下面这篇文章主要给大家介绍了关于Vue非单文件组件使用的相关资料,文中通过图文以及实例代码介绍的非常详细,需要的朋友可以参考下
    2023-06-06

最新评论