vue 实现可拖曳的树状结构图

 更新时间:2021年04月12日 10:25:35   作者:YouKnow_HRT  
这篇文章主要介绍了vue 实现可拖曳的树状结构图,帮助大家更好的理解和学习使用vue框架,感兴趣的朋友可以了解下

最近用vue做了一个小项目--可拖曳的树状结构图。

Vue递归组件

结构通过Vue的递归组件实现

布局使用flex,结构线由CSS伪类实现

需要注意的是居中布局,当X轴元素过多导致子元素宽度超出视图,元素居中后虽然有滚动条,但只能到达右边的内容,左边的内容会无法访问,可以把父元素设置为inline-flex,宽度设置为auto。当然,如果是上述的结构则不会有这个问题,但遇到大数据渲染,还是困扰我了一下午。

drag事件

首先在需要在拖动的元素上绑定draggable属性,除了<a>和<img>标签设默认为true,其他元素都需要设置下

然后是三个事件dragenddragoverdrop(用Vue写的,就不在事件前加on了)

注意dragover需要去除默认行为,在事件中给上$event.preventDefault(),否则拖曳时鼠标会有🚫标志,使拖动无效。

drag的元素把值传到drop的位置,需要使用$event.dataTransfer.setData("node", transNodeData)

"node"是相当于传递数据的变量名,需要先JSON.stringify()

methods:{
    dragstart(e,nodeObj){
      console.log('🐉drag移动的点位',nodeObj.name,);
      let transData=JSON.stringify(nodeObj)//拖曳传递过去的数据先转为JSON格式
      e.dataTransfer.setData("node",transData)
    },
    dragover(e){
      e.preventDefault()
    },
    drop(e,nodeObj){
      console.log('🐉drop到的点位',nodeObj.name);
      let getData=JSON.parse( e.dataTransfer.getData("node"))
      console.log('获取了数据',getData);
    }
  }

了解了这一点,接下来要做的就是把获取的拖曳点位数据放到放置点位的children数组中,并在dragend事件中把拖曳点位在父节点children数组中删除,节点的索引在dragstart事件触发时就获取,并通过eventBus这一组件通信方式传递给dragend(也可以使用Vuex)。

创建bus文件夹,新建index.js文件

import Vue from "vue"
const busEvent= new Vue({
  data(){
    return{
      dragNodeIndex:-1,//拖曳节点在父节点children数组中的index
    }
  },
  created(){
    this.$on("transDragNodeIndex", res=>{//通过$on来监听$emit,需要确保自定义事件在触发前被监听,也就是订阅先于发布,否则无法监听到数据,我eventBus没怎么用过,这算是个坑
      this.dragNodeIndex=res
    })
  }
})

export default busEvent

在组件中引入eventBus,此时在dragstart中通过$emit触发自定义事件后,$on就可以接收到这个数据,在dragend中,可以通过eventBus获取这一索引,然后在数组中删除

接下来就是做一些逻辑判断,例如父节点不能拖曳到子节点,先通过递归方式把父节点上所有子节点的name遍历进一个数组,如果drop位置的name在数组中就表明父到子了,设置状态为true,

ifFatherDragToSon(dragObj,dropObj){//判断是否父节点移动到了子节点
      if (dragObj.children.length === 0) return false;
      let newArr = [];
      function getAllName(dragObj) {
        newArr.push(...dragObj.children);
        if (dragObj.children.length === 0) {
          return;
        } else {
          for (let i = 0; i < dragObj.children.length; i++) {
            getAllName(dragObj.children[i]);
          }
        }
      }
      getAllName(dragObj);
      if (newArr.includes(dropObj)) {
        return true;
      }
      return false;
    }

通过eventBusdragend事件获取状态,为true直接return

拖曳到自身也不行,直接return。特意在拎壶冲点位下加了个子节点,主要就是为了验证通过节点在栈中的地址来判断,而不是依据name属性

再一个就是提一下drag事件中的$event.dataTransfer.dropEffect,可以获取的值为movecopynonelink,下图中当dropEffect为🚫时拖曳点位消失了,其实就是执行了dragend事件中的代码,而drop事件中的代码未执行,所以这一步需要先做流程控制。另外说明,不同浏览器的默认dropEffect也是不同的,比如360浏览器。

代码连接

以上就是vue 实现可拖曳的树状结构图的详细内容,更多关于vue 实现树状结构图的资料请关注脚本之家其它相关文章!

相关文章

  • Vue.js下拉菜单组件使用方法详解

    Vue.js下拉菜单组件使用方法详解

    这篇文章主要为大家详细介绍了Vue.js下拉菜单组件的使用方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-10-10
  • vue中项目页面空白但不报错产生的原因及分析

    vue中项目页面空白但不报错产生的原因及分析

    这篇文章主要介绍了vue中项目页面空白但不报错产生的原因及分析,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-05-05
  • vue中使用AJAX实现读取来自XML文件的信息

    vue中使用AJAX实现读取来自XML文件的信息

    这篇文章主要为大家详细介绍了vue中如何使用AJAX实现读取来自XML文件的信息,文中的示例代码讲解详细,具有一定的借鉴价值,需要的小伙伴可以参考下
    2023-12-12
  • vue全局数据管理示例详解

    vue全局数据管理示例详解

    这篇文章主要为大家介绍了vue全局数据管理示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07
  • vue3项目keepAlive使用方法详解

    vue3项目keepAlive使用方法详解

    keepalive是Vue的内置组件,作用是将组件缓存在内存当中,防止重复渲染DOM,属于消耗内存获取速度,这篇文章主要介绍了vue3项目keepAlive使用方法,主要将keepaliev在vue3.0中的用法,需要的朋友可以参考下
    2024-03-03
  • vue+elementui实现动态控制表格列的显示和隐藏

    vue+elementui实现动态控制表格列的显示和隐藏

    这篇文章主要介绍了vue+elementui实现动态控制表格列的显示和隐藏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-04-04
  • ElementUI中el-dropdown-item点击事件无效问题

    ElementUI中el-dropdown-item点击事件无效问题

    这篇文章主要介绍了ElementUI中el-dropdown-item点击事件无效问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-04-04
  • Vue实现子组件向父组件传递多个参数的方法

    Vue实现子组件向父组件传递多个参数的方法

    在Vue框架中,组件间的通信是一个常见的需求,特别是在子组件需要向父组件传递多个参数时,合理的通信方式可以显著提升代码的可读性和可维护性,本文将详细介绍如何在Vue中实现子组件向父组件传递多个参数,需要的朋友可以参考下
    2024-10-10
  • Vue 3 的<Teleport>功能与用法详解

    Vue 3 的<Teleport>功能与用法详解

    <Teleport> 是 Vue 3 的一个内置组件,允许将组件的内容渲染到 DOM 中的任意位置,而不改变其逻辑结构,这篇文章主要介绍了Vue 3 的<Teleport>功能与用法详解,需要的朋友可以参考下
    2025-04-04
  • 在vue中封装方法以及多处引用该方法详解

    在vue中封装方法以及多处引用该方法详解

    这篇文章主要介绍了在vue中封装方法以及多处引用该方法详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-08-08

最新评论