当前位置:首页 > 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:

vue实现树形拖拽

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 实现:

vue实现树形拖拽

<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 实现走势图的方法 使用 ECharts 库 ECharts 是一个强大的数据可视化库,支持多种图表类型,包括走势图(折线图)。在 Vue 项目中可以通过 vue-echarts 封装库或直接使…

vue实现3d宇宙

vue实现3d宇宙

Vue实现3D宇宙效果 使用Three.js库 Three.js是一个强大的JavaScript 3D库,可以轻松集成到Vue项目中。安装Three.js依赖: npm install three…

vue实现旋转

vue实现旋转

Vue 实现旋转效果的方法 在 Vue 中实现旋转效果可以通过多种方式完成,以下是几种常见的方法: 使用 CSS 动画 通过 Vue 的 :class 或 :style 绑定 CSS 动画实现旋转效…

vue element实现

vue element实现

Vue Element UI 实现 Vue Element UI 是基于 Vue.js 的组件库,提供丰富的 UI 组件,适合快速开发企业级中后台产品。以下是一些关键实现方法和步骤: 安装 Vue…

vue实现公告

vue实现公告

Vue 实现公告功能的方法 公告功能通常需要实现滚动展示、自动切换或固定显示的效果。以下是几种常见的实现方式: 使用 marquee 标签实现滚动公告 <template> <…

vue实现socket

vue实现socket

Vue 中实现 WebSocket 通信 在 Vue 项目中实现 WebSocket 通信可以通过原生 WebSocket API 或第三方库(如 socket.io-client)完成。以下是两种常…