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

基本用法:

vue实现树形拖拽

<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懒加载的实现并不复杂,核心逻辑是通过动态导入(Dynamic Imports)和路由配置或组件异步加载完成。以下是具体实现方法: 路由懒加载实现 在Vue Router…

vue实现视窗

vue实现视窗

Vue 实现视窗功能 在 Vue 中实现视窗功能通常涉及监听浏览器窗口大小变化、响应式布局或创建自定义弹窗组件。以下是几种常见实现方式: 监听浏览器窗口大小变化 使用 Vue 的 mounted 和…

vue实现ios

vue实现ios

Vue 实现 iOS 风格应用 使用 Vue 实现 iOS 风格的应用,可以通过结合 UI 框架和自定义样式来达成目标。以下是具体方法和步骤: 选择 iOS 风格的 UI 框架 Vue 生态中有多…

vue实现弹幕

vue实现弹幕

Vue 实现弹幕功能 弹幕功能通常包括动态生成、滚动显示、颜色和速度随机化等特性。以下是基于 Vue 3 的实现方法。 核心思路 使用 CSS 动画控制弹幕从右向左滚动。 动态生成弹幕数据,随机设置…

vue实现mui

vue实现mui

使用 Vue 实现 MUI(Material-UI)风格的界面 Vue 本身不直接提供 MUI(Material Design)组件,但可以通过集成第三方库或手动实现来达到类似效果。以下是几种常见方法…

proxy实现vue

proxy实现vue

使用 Proxy 实现 Vue 响应式系统 Vue 3 的响应式系统基于 JavaScript 的 Proxy API,通过代理对象实现对属性的拦截和依赖追踪。以下是实现的核心逻辑: 创建响应式对象…