当前位置:首页 > VUE

vue实现树形拖拽

2026-01-17 23:14:38VUE

Vue 实现树形拖拽的方法

使用 vue-draggable 插件

安装 vue-draggable 插件:

npm install vuedraggable --save

在组件中引入并使用:

<template>
  <draggable v-model="treeData" group="nodes" @end="onDragEnd">
    <div v-for="(item, index) in treeData" :key="item.id">
      {{ item.name }}
      <draggable v-if="item.children" v-model="item.children" group="nodes">
        <div v-for="child in item.children" :key="child.id">
          {{ child.name }}
        </div>
      </draggable>
    </div>
  </draggable>
</template>

<script>
import draggable from 'vuedraggable'

export default {
  components: { draggable },
  data() {
    return {
      treeData: [
        { id: 1, name: 'Node 1', children: [
          { id: 11, name: 'Node 1-1' }
        ]},
        { id: 2, name: 'Node 2' }
      ]
    }
  },
  methods: {
    onDragEnd() {
      console.log('Drag end', this.treeData)
    }
  }
}
</script>

使用 Element UI 的树形组件

安装 Element UI:

npm install element-ui --save

实现拖拽功能:

<template>
  <el-tree
    :data="treeData"
    draggable
    @node-drop="handleDrop"
  />
</template>

<script>
export default {
  data() {
    return {
      treeData: [
        { label: 'Node 1', children: [
          { label: 'Node 1-1' }
        ]},
        { label: 'Node 2' }
      ]
    }
  },
  methods: {
    handleDrop(draggingNode, dropNode, dropType) {
      console.log('Drop event', draggingNode, dropNode, dropType)
    }
  }
}
</script>

自定义实现拖拽功能

通过 HTML5 的拖拽 API 实现:

<template>
  <div class="tree">
    <div
      v-for="item in treeData"
      :key="item.id"
      draggable="true"
      @dragstart="handleDragStart($event, item)"
      @dragover.prevent
      @drop="handleDrop($event, item)"
    >
      {{ item.name }}
      <div v-if="item.children" class="children">
        <div
          v-for="child in item.children"
          :key="child.id"
          draggable="true"
          @dragstart="handleDragStart($event, child)"
          @dragover.prevent
          @drop="handleDrop($event, child)"
        >
          {{ child.name }}
        </div>
      </div>
    </div>
  </div>
</template>

<script>
export default {
  data() {
    return {
      treeData: [
        { id: 1, name: 'Node 1', children: [
          { id: 11, name: 'Node 1-1' }
        ]},
        { id: 2, name: 'Node 2' }
      ],
      draggingNode: null
    }
  },
  methods: {
    handleDragStart(event, node) {
      this.draggingNode = node
      event.dataTransfer.setData('text/plain', node.id)
    },
    handleDrop(event, targetNode) {
      if (this.draggingNode === targetNode) return
      // 实现节点移动逻辑
      console.log('Move', this.draggingNode, 'to', targetNode)
    }
  }
}
</script>

<style>
.tree {
  user-select: none;
}
.children {
  margin-left: 20px;
}
</style>

使用 vue-tree-dnd 插件

专门用于树形拖拽的插件:

npm install vue-tree-dnd --save

基本用法:

<template>
  <tree :data="treeData" :options="options" @change="onTreeChange"/>
</template>

<script>
import { Tree } from 'vue-tree-dnd'

export default {
  components: { Tree },
  data() {
    return {
      treeData: [
        { id: 1, name: 'Node 1', children: [
          { id: 11, name: 'Node 1-1' }
        ]},
        { id: 2, name: 'Node 2' }
      ],
      options: {
        drag: true,
        drop: true
      }
    }
  },
  methods: {
    onTreeChange(newTree) {
      this.treeData = newTree
    }
  }
}
</script>

注意事项

  • 树形数据结构需要包含唯一标识符(如 id)
  • 拖拽时需要处理父子节点关系
  • 移动节点后需要更新整个树结构
  • 复杂操作可能需要递归处理数据

以上方法可以根据项目需求选择合适的方式实现树形拖拽功能。

vue实现树形拖拽

标签: 拖拽vue
分享给朋友:

相关文章

vue弹幕实现不重叠

vue弹幕实现不重叠

实现 Vue 弹幕不重叠的方法 CSS 定位与动画控制 通过绝对定位和动态计算弹幕轨道高度,确保弹幕在不同轨道上运行。每条弹幕占据固定高度的轨道,避免重叠。 // 弹幕轨道高度计算 const tr…

vue前端实现打印功能

vue前端实现打印功能

使用Vue实现前端打印功能 在Vue项目中实现打印功能可以通过多种方式完成,以下是几种常见的方法: 使用window.print()方法 这是最简单的打印方式,直接调用浏览器的打印功能。 me…

vue实现防抖节流

vue实现防抖节流

Vue 中实现防抖(Debounce) 防抖的核心是延迟执行函数,在连续触发时取消前一次的调用,只执行最后一次。适用于输入框搜索、窗口大小调整等场景。 方法1:使用 Lodash 的 _.debou…

vue实现继承语法糖

vue实现继承语法糖

在Vue中实现继承可以通过多种方式模拟类继承的效果,以下是常见的几种方法: 使用mixins实现逻辑复用 通过Vue的mixins机制共享组件选项,达到类似继承的效果: const Bas…

vue实现导航栏切图

vue实现导航栏切图

Vue 实现导航栏切图 使用 Vue Router 实现基础导航 在 Vue 项目中,可以通过 Vue Router 实现导航栏的路由切换。创建一个导航栏组件,并在其中使用 <router-li…

vue实现发送值接受值

vue实现发送值接受值

Vue 实现父子组件传值 父组件向子组件传值(props) 父组件通过 v-bind 或简写 : 向子组件传递数据。子组件通过 props 接收数据。 父组件代码: <templat…