Vue实现拖拽改变列表顺序

当我们构建前端应用时,有时需要实现一些交互性强的功能,比如拖拽改变列表顺序。在本文中,我将演示如何使用 Vue.js 实现这样一个功能。

首先,我们需要一个基本的 Vue 组件结构,包括 HTML 模板、JavaScript 逻辑和 CSS 样式。我们将在 Vue 组件中实现拖拽改变列表顺序的功能。

html 复制代码
<template>
  <div ref="list" 
    :ondragstart="dragstart"
    :ondragenter="dragenter"
    :ondragend="dragend"
    class="list">
    <div draggable="true" class="item" v-for="i in 10">{{ i }}</div>
  </div>
</template>

<script setup>
import { ref } from 'vue'

// 被拖拽的父元素
const list = ref(null)
// 当前被拖拽的元素
const sourceNode = ref(null)

// 拖拽开始事件
const dragstart = (e) => {
  setTimeout(() => {
    // 需要异步 不然元素直接消失
    sourceNode.value = e.target
    e.target.classList.add('moving')
  }, 0)
}

const dragenter = (e) => {
  // 阻止浏览器默认行为,否则放手时会回到原位
  e.preventDefault()
  if(e.target === list.value || e.target === sourceNode.value) {
    // 如果是在父元素上,或在自身元素上,不做任何处理
    return;
  }
  const children = [... list.value.children]
  const sourceIndex = children.indexOf(sourceNode.value)
  const targetIndex = children.indexOf(e.target)
  // 插入到对应位置
  if(sourceIndex < targetIndex) {
    list.value.insertBefore(sourceNode.value, e.target.nextElementSilbling)
  }else {
    list.value.insertBefore(sourceNode.value, e.target)    
  }
}

const dragend = (e) => {
  e.target.classList.remove('moving')
}
</script>

<style scoped>
.box {
  height: 80vw;
}
.item {
  height: 40px;
  background-color: antiquewhite;
  margin: 10px;
  cursor: pointer;
}

.item.moving {
  background: #ccc;
  color: transparent;
  border: 1px dashed #ccc;
}
</style>

以上是完整的代码。现在让我们逐步解释这段代码是如何工作的。

  1. <template> 部分包含了我们的 HTML 结构。我们有一个包含一些可拖拽元素的容器,每个元素都有一个拖拽事件。
  2. <script setup> 部分是 Vue.js 3 的新特性,用于编写组件的逻辑。在这里,我们引入了 ref 函数,用来创建响应式数据。我们创建了两个 ref,一个用于跟踪被拖拽的父元素,另一个用于跟踪当前被拖拽的元素。然后我们定义了三个函数,分别处理拖拽开始、拖拽进入和拖拽结束事件。
  3. dragstart 函数中,我们设置了一个异步定时器,以确保获取到正确的被拖拽元素。然后我们给被拖拽的元素添加一个 moving 类,用于在拖拽时改变其样式。
  4. dragenter 函数处理拖拽进入事件。我们首先阻止了浏览器的默认行为,然后判断当前拖拽的目标元素是否是列表容器本身或者是拖拽元素本身,如果是则不进行任何处理。接着,我们获取了列表容器中所有子元素,并计算出被拖拽元素和目标元素的索引,然后根据这些索引将被拖拽元素插入到正确的位置。
  5. dragend 函数在拖拽结束时移除了被拖拽元素的 moving 类,恢复其原始样式。
  6. 最后,<style scoped> 部分包含了组件的 CSS 样式,包括列表容器和拖拽元素的样式定义。
相关推荐
sunly_12 分钟前
Flutter:自定义Tab切换,订单列表页tab,tab吸顶
开发语言·javascript·flutter
Zero_pl25 分钟前
vue学习路线
vue.js
咔咔库奇31 分钟前
【TypeScript】命名空间、模块、声明文件
前端·javascript·typescript
NoneCoder33 分钟前
JavaScript系列(42)--路由系统实现详解
开发语言·javascript·网络
2013crazy1 小时前
Java 基于 SpringBoot+Vue 的校园兼职平台(附源码、部署、文档)
java·vue.js·spring boot·兼职平台·校园兼职·兼职发布平台
兩尛1 小时前
订单状态定时处理、来单提醒和客户催单(day10)
java·前端·数据库
又迷茫了1 小时前
vue + element-ui 组件样式缺失导致没有效果
前端·javascript·vue.js
哇哦Q1 小时前
原生HTML集合
前端·javascript·html
SoWhat~1 小时前
随遇随记篇
前端·javascript
孟健1 小时前
重磅首发:国产AI编程助手Trae实测!免费用上Claude是什么体验?
前端·aigc·visual studio code