vue递归组件:树形控件小白知识_树教程视频

1、递归组件-简单树形控件预览及问题在编写树形组件时遇到的问题:组件如何才能递归调用?递归组件点击事件如何传递?2、树形控件基本结构及样式

  • <div class="expand-ar

vue递归组件:树形控件小白知识

1、递归组件-简单树形控件预览及问题

vue递归组件:树形控件小白知识_树教程视频

vue递归组件:树形控件小白知识_树教程视频

在编写树形组件时遇到的问题:

  • 组件如何才能递归调用?
  • 递归组件点击事件如何传递?

2、树形控件基本结构及样式

<template>
  <ul class="vue-tree">
    <li class="tree-item">
      <div class="tree-content"><!--节点内容-->
        <div class="expand-arrow"></div><!--展开或收缩节点按钮-->
        <div class="tree-label">小学</div><!--节点文本内容-->
      </div>
      <ul class="sub-tree"><!--子节点-->
        <li class="tree-item expand">
          <div class="tree-content">
            <div class="expand-arrow"></div>
            <div class="tree-label">语文</div>
          </div>
        </li>
        <li class="tree-item">
          <div class="tree-content">
            <div class="expand-arrow"></div>
            <div class="tree-label">数学</div>
          </div>
        </li>
      </ul>
    </li>
  </ul>
</template>

<style lang="stylus">
.vue-tree{
  list-style: none;
  padding: 0;
  margin: 0;
  .tree-item{
    cursor: pointer;
    transition: background-color .2s;
    .tree-content{
      position: relative;
      padding-left: 28px;
      &:hover{
        background-color: #f0f7ff;
      }
    }
    .expand-arrow{
      position: absolute;
      top: 0;
      left: 0;
      width: 28px;
      height: 28px;
      cursor: pointer;
      &::after{
        position: absolute;
        top: 50%;
        left: 50%;
        display: block;
        content: ' ';
        border-width: 5px;
        border-style: solid;
        border-color: transparent;
        border-left-color: #ccc;
        margin: -5px 0 0 -2.5px;
        transition: all .2s;
      }
    }
    &.expand{
      &>.tree-content{
        background-color: #f0f7ff;
        &>.expand-arrow{
          &::after{
            transform: rotate(90deg);
            margin: -2.5px 0 0 -5px;
          }
        }
      }
    }
    .tree-label{
      height: 28px;
      line-height: 28px;
      font-size: 14px;
    }
    .sub-tree{
      display: none;
      list-style: none;
      padding: 0 0 0 28px;
      margin: 0;
    }
    &.expand>.sub-tree{
      display: block;
    }
    &.no-child{
      &>.tree-content{
        &>.expand-arrow{
          display: none;
        }
      }
    }
  }
}
</style>

3、组件目录及数据结构

目录结构
vue-tree

  • VueTree.vue 树形控件父组件
  • TreeItem.vue 树形控件递归组件

树形控件数据结构

let treeData = [
  {
    text: "一级", // 显示的文字
    expand: false, // 默认是否展开
    children: [ // 子节点
      {
        text: "一级-1",
        expand: false,
      },
      {
        text: "一级-2",
        expand: false,
        children: [
          {
            text: "一级-2-1",
            expand: false,
          },
          {
            text: "一级-2-2",
            expand: false,
          }
        ]
      }
    ]
  }
];

3.1、TreeItem.vue 代码

<template>
  <li class="tree-item" :class="{expand: isExpand, 'no-child': !treeItemData.children || treeItemData.children.length  === 0}">
    <div class="tree-content" @click="_clickEvent">
      <div class="expand-arrow" @click.stop="expandTree()"></div>
      <div class="tree-label">{{treeItemData.text}}</div>
    </div>
    <ul class="sub-tree" v-if="treeItemData.children && treeItemData.children.length > 0">
      <!--TreeItem组件中调用TreeItem组件-->
      <TreeItem
        v-for="item in treeItemData.children"
        :tree-item-data="item"
        :key="uuid()"
        :tree-click-event="treeClickEvent"></TreeItem>
    </ul>
  </li>
</template>

<script>
  export default {
    name: "TreeItem",
    props: {
      treeItemData: {
        type: Object,
        default(){
          return {};
        }
      },
      // 节点点击事件
      treeClickEvent: {
        type: Function,
        default() {
          return function () {};
        }
      }
    },
    data(){
      return {
        // 节点是否展开
        isExpand: this.treeItemData.expand || false
      }
    },
    methods: {
      // 展开/收缩
      expandTree(flag){
        if(!this.treeItemData.children || this.treeItemData.children.length === 0){
          return;
        }
        if(typeof flag === 'undefined'){
          flag = !this.isExpand;
        }else{

          flag = !!flag;
        }
        this.isExpand = flag;
      },
      // 创建一个唯一id
      uuid(){
        let str = Math.random().toString(32);
        str = str.substr(2);
        return str;
      },
      // 节点点击事件
      _clickEvent(){
        // 如果有传递事件函数,则调用事件函数并传递当前节点数据及组件
        if(this.treeClickEvent && typeof this.treeClickEvent === 'function'){
          this.treeClickEvent(this.treeItemData, this);
        }
      }
    }
  }
</script>

3.1.1、解决 组件如何才能递归调用? 问题

在组件模板内调用自身必须明确定义组件的name属性,并且递归调用时组件名称就是name属性。如在TreeItem.vue组件中组件的name名称为’TreeItem’,那么在template中调用时组件名称就必须是<TreeItem>。
当然也可以全局注册组件,具体可以查看vue官方文档 递归组件

3.1.2、解决 递归组件点击事件如何传递? 问题

我这里的解决方案是使用props将事件函数传递进来,在点击节点的时候调用事件函数,并把相应的数据传递进去。
之前也尝试过使用$emit的形式并把数据传递过去,由于是递归组件,这样一直$emit,到最外层时传递的数据就变了,比如传递是第3层节点的数据,到最后执行时数据就变成第1层节点的数据了

4、VueTree.vue 组件

<template>
  <ul class="vue-tree">
    <TreeItem
        v-for="(item, index) in treeData"
        :key="index"
        :treeItemData="item"
        :tree-click-event="treeClickEvent"></TreeItem>
  </ul>
</template>

<script>
  import TreeItem from "./TreeItem";
  export default {
    name: "VueTreeMenu",
    components: {
      TreeItem
    },
    props: {
     // 树形控件数据
      treeData: {
        type: Array,
        default(){
          return [];
        }
      },
      // 节点点击事件
      treeClickEvent: {
        type: Function,
        default() {
          return function () {};
        }
      }
    }
  }
</script>

<style lang="stylus">
.vue-tree{
  list-style: none;
  padding: 0;
  margin: 0;
  .tree-item{
    cursor: pointer;
    transition: background-color .2s;
    .tree-content{
      position: relative;
      padding-left: 28px;
      &:hover{
        background-color: #f0f7ff;
      }
    }
    .expand-arrow{
      position: absolute;
      top: 0;
      left: 0;
      width: 28px;
      height: 28px;
      cursor: pointer;
      &::after{
        position: absolute;
        top: 50%;
        left: 50%;
        display: block;
        content: ' ';
        border-width: 5px;
        border-style: solid;
        border-color: transparent;
        border-left-color: #ccc;
        margin: -5px 0 0 -2.5px;
        transition: all .2s;
      }
    }
    &.expand{
      &>.tree-content{
        background-color: #f0f7ff;
        &>.expand-arrow{
          &::after{
            transform: rotate(90deg);
            margin: -2.5px 0 0 -5px;
          }
        }
      }
    }
    .tree-label{
      height: 28px;
      line-height: 28px;
      font-size: 14px;
    }
    .sub-tree{
      display: none;
      list-style: none;
      padding: 0 0 0 28px;
      margin: 0;
    }
    &.expand>.sub-tree{
      display: block;
    }
    &.no-child{
      &>.tree-content{
        /*padding-left: 0;*/
        &>.expand-arrow{
          display: none;
        }
      }
    }
  }
}
</style>

5、使用树形组件

<span type="button" data-toggle="tooltip" data-placement="top" data-clipboard-text="

import VueTree from "./components/vue-tree/VueTree";

export default {
name: ‘app’,
data(){
return {
treeData2: [
{
text: "一级", // 显示的文字
expand: false, // 默认是否展开
children: [
{
text: "二级-1",
expand: false,
},
{
text: "二级-2",
expand: false,
children: [
{
text: "三级-1",
expand: false,
},
{
text: "三级-2",
expand: false,
children: [
{
text: "四级-1",
expand: false,
}
]
}
]
}
]
},
{
text: "一级-2",
expand: false
}
]
}
},
methods: {
treeClickEvent(item, treeItem){
console.log(item);
}
},
components: {
VueTree
}
}

” title=”” data-original-title=”复制”>

<template>
  <div class="app" id="app">
    <VueTree :tree-data="treeData2" :tree-click-event="treeClickEvent"></VueTree>
  </div>
</template>

<script>
import VueTree from "./components/vue-tree/VueTree";

export default {
  name: 'app',
  data(){
    return {
      treeData2: [
        {
          text: "一级", // 显示的文字
          expand: false, // 默认是否展开
          children: [
            {
              text: "二级-1",
              expand: false,
            },
            {
              text: "二级-2",
              expand: false,
              children: [
                {
                  text: "三级-1",
                  expand: false,
                },
                {
                  text: "三级-2",
                  expand: false,
                  children: [
                    {
                      text: "四级-1",
                      expand: false,
                    }
                  ]
                }
              ]
            }
          ]
        },
        {
          text: "一级-2",
          expand: false
        }
      ]
    }
  },
  methods: {
    treeClickEvent(item, treeItem){
      console.log(item);
    }
  },
  components: {
    VueTree
  }
}
</script>
海计划公众号
(0)
上一篇 2020/03/26 23:34
下一篇 2020/03/26 23:34

您可能感兴趣的内容

  • Nginx的DNS解析过程分析小白知识_dns小白知识

    Nginx怎么做域名解析?怎么在你自己开发的模块里面使用Nginx提供的方法解析域名?它内部实现是什么样的?本文以Nginx 1.5.1为例,从nginx_mail_smtp模块如何进行域名解析出发,分析Nginx进行域名解析的过程。为了简化流程,突出重点,在示例代码中省掉了一些异常部分的处理,比如内存分配失败等。DNS查询分为两种:根据域名查询地址和根据地

    2020/03/26
  • Angular 中的 asyncPipe 源码探究菜鸟攻略_源码零基础入门

    Angular 中的 asyncPipe 源码探究菜鸟攻略 async 管道用于解包异步原始数据。说到异步数据,就自然而然的会想起 observable 和 promise, as…

    2020/03/20
  • realworld小白基础_由React,Angular,Node,Django等提供支持的示例

    realworld小白基础 官方网址:https://realworld.io/ GitHub:https://github.com/gothinkster/realworld 简…

    2020/03/10
  • html页面锚点链接对SEO的作用,以及实现描点的三种方式新手入门_网页小白基础

    HTML中的链接,正确的说法应该称作”锚点“,它命名锚点链接(也叫书签链接)常常用于那些内容庞大繁琐的网页,
    通过点击命名锚点,不仅让我们能指向文档,还能指向页面里的特定段落,更能当作”精准链接”的便利工具,让链接对象接近焦点。便于浏览者查看网页内容。
    类似于我们阅读书籍时的目录页码或章回提示。在需要指定到页面的特定部分时,标记锚点是最佳的方法。锚点链接对S

    2020/04/03
  • React 菜鸟教程下载 Webpack 构建打包优化小白攻略_React

    React 相关的优化使用 babel-react-optimize 对 React 代码进行优化检查没有使用的库,去除 import 引用按需打包所用的类库,比如 lodash 、echarts 等lodash 可以采用 babel-plugin-lodash 进行优化。需要注意的是:在 babel-react-optimize 中使用了 babel-pl

    2020/04/05
  • AlloyCrop小白知识_做最好且最小的移动裁剪组件

    AlloyCrop小白知识 官方网址:https://alloyteam.github.io/AlloyCrop/ GitHub:https://github.com/AlloyT…

    2020/03/10
  • 将async/await编译到 ES3/ES5 (外部帮助库)小白知识_async基础指南

    自2015年11 发布1.7版以来,TypeScript 已支持 async/await 关键字。编译器使用 yield 将异步函数转换为生成器函数。这意味着咱们无法针对 ES3 或 ES5,因为生成器仅在 ES6 中引入的。TypeScript 2.1 现在支持将异步函数编译为 ES3 和 ES5。与生成的其余代码一样,它们在所有 JS 环境中运行。(这甚

    2020/03/23
  • Vue中你可能认为是bug的情况原来是这样的入门百科_bug使用说明

    Vue中你可能认为是bug的情况原来是这样的入门百科 前言 我们知道Vue框架剧本双向数据绑定功能,在我们使用方便的同时,还有一些细节问题我们并不知道,接下来一起探讨一些吧 双向数…

    2020/03/19
  • CssStats基础入门在线CSS代码分析工具

    CssStats入门基础教程 官方网址:https://cssstats.com/ 简介描述:在线CSS代码分析工具 是一个输入网站CSS网址即可进行CSS代码分析的在线工具,是前…

    2020/03/06
  • Typescript 和 Javascript之间的区别入门基础知识_TypeScript使用教程

    TypeScript 和 JavaScript 是目前项目开发中较为流行的两种脚本语言,我们已经熟知 TypeScript 是 JavaScript 的一个超集,但是 TypeScript 与 JavaScript 之间又有什么样的区别呢?在选择开发语言时,又该如何抉择呢?本文将会深入对比这两种语言,讨论两种语言之间的关联和差异,并概述两种语言各自的优势。J

    2020/04/05
  • Js ajax状态码菜鸟攻略_码教程视频

    hr.readyState==4(ajax运行正常)xhr.status==200(后台运行正常)xhr.status==5xx(后台运行异常 联系后台代码人员) HTTP 状态码 ajax.status 1**:请求收到,继续处理 2**:操作成功收到,分析、接受 3**:完成此请求必须进一步处理 4**:请求包含一个错误语法或不能

    Web前端 2020/03/23
  • bejson入门教程_在线JSON校验格式化工具

    bejson入门教程 官方网址:http://www.bejson.com/ 简介描述:在线JSON校验格式化工具 在线,JSON,JSON 校验,格式化,xml转json 工具,…

    2020/03/12
  • 郑州网站建设告诉你移动端网站关键词怎么优化新手入门_网站入门知识

    郑州网站建设告诉你移动端网站关键词怎么优化?现在人人都要用手机,手机已经成为我们生活中不可或缺的一部分,随着互联网的发展,移动端也越来越重要了移动端网站关键词怎么优化?随着现在互联网的快速发展,据目前的用户搜索习惯来看,移动端网站的索引已经开始优先于PC端网站的索引了,移动端的优质流量开始远远大于PC端了,很多企业都开始关注移动端关键词排名,移动端和PC端排

    2020/04/03
  • 多阶Hash算法入门教程_Hash小白攻略

    在分布式系统中,会经常用到K-V存储,一般实现的方式有红黑树或者哈希表,在Redis中还用到了跳表。都是通过一个确定的Key值,来查找Key附带的Value属性。本文会介绍一种高效的算法——多阶Hash。1原理多阶Hash的实现原理很简单,每个Hash桶数组算作一阶,如果有20阶的多阶Hash,那么就是一个二维数组,第一维是Hash桶的编号,第二维是每个Ha

    2020/03/23
  • vue 基础知识入门 webpack 前端性能优化入门知识_性能

    背景对于程序开发者而言,开发一个项目不仅仅注重效率和功能,前端的性能问题也是非常重要的。这直接影响用户的体验,从而间接的也反应该项目质量的好坏。 影响项目性能的原因有很多,如:资源文件的大小,业务的繁杂程度等,所以前端优化的方式也很多。这些东西很零碎,容易被人遗忘。优化一: vue-router路由懒加载懒加载: 也叫延迟加载,即在需要的时候进行加载,随用随

    2020/03/20
  • node.js代理访问小白教程_代理使用帮助

    1. 场景本地开发,代理访问,防止跨域(一般通过webpack配置代理即可),特殊情况如携带一些自定义的登录cookie则需要通过自己写node,作为一种server中间层,单线程异步可以缓解服务器压力。长链接websocket通常使用node搭建2. 技术框架node – koa2 体量小,轻便易用。路由koa-router koa配套路由,中间件支持as

    2020/03/26