js实现拖拽功能

news/2025/1/2 2:05:11/

基于onMouseDown 、onMouseMove 、onMouseUp

使用 mousedownmousemovemouseup 事件来实现拖拽的基本思路是:

  1. mousedown 事件中,开始追踪拖拽操作并记录鼠标按下的位置。

  2. mousemove 事件中,根据鼠标的移动,更新被拖拽元素的位置。

  3. mouseup 事件中,停止拖拽操作,清除事件监听器,以及执行必要的清理操作。

以下是一些关键的步骤和注意事项:

实现拖拽的步骤:

  1. 在元素上监听 mousedown 事件。当鼠标按下时,标记拖拽操作的开始。

  2. mousedown 事件处理程序中,记录鼠标按下时的初始位置(通常是相对于被拖拽元素的偏移量)。这将帮助你计算鼠标相对于元素的位置。

  3. 添加 mousemove 事件监听器。在 mousemove 事件处理程序中,计算鼠标位置的变化,并根据变化来更新被拖拽元素的位置。

  4. mouseup 事件处理程序中,停止拖拽操作。清除 mousemovemouseup 事件监听器,以确保不再处理鼠标移动和释放事件。

注意事项:

  1. 记录鼠标偏移:确保在 mousedown 事件处理程序中记录鼠标按下时的初始位置。这将帮助你计算鼠标相对于元素的位置,以正确更新元素的位置。

  2. 防止文字选中:在拖拽元素上防止文字的选中,可以使用 CSS 的 user-select: none; 样式或在 mousedown 事件中取消选择文字的默认行为。

  3. 移动边界:确保在 mousemove 事件处理程序中更新元素的位置时,限制元素的移动范围,以防止它移出可见区域或父元素的边界。

  4. 清理:在 mouseup 事件处理程序中,清除事件监听器,以避免内存泄漏或不必要的事件处理。

  5. 可访问性:考虑到可访问性,确保你的拖拽操作不会影响键盘用户。同时,确保在视觉上指示被拖拽元素的状态。

  6. 兼容性:确保你的拖拽实现在不同浏览器中正常工作,可能需要处理一些浏览器特定的问题或事件。

以下是一个简单的示例,演示了如何使用 mousedownmousemovemouseup 事件来实现一个可拖拽的元素:

const element = document.getElementById('draggableElement');
let isDragging = false;
let offsetX, offsetY;element.addEventListener('mousedown', (e) => {isDragging = true;offsetX = e.clientX - element.getBoundingClientRect().left;offsetY = e.clientY - element.getBoundingClientRect().top;element.style.cursor = 'grabbing';
});document.addEventListener('mousemove', (e) => {if (isDragging) {element.style.left = e.clientX - offsetX + 'px';element.style.top = e.clientY - offsetY + 'px';}
});document.addEventListener('mouseup', () => {isDragging = false;element.style.cursor = 'grab';
});

另外注意:

  1. 绝对定位: 通常,要实现拖拽效果,需要将元素设置为绝对定位,这样它可以脱离文档流,使其能够在页面上自由移动。这是拖拽操作的一个基本前提。

  2. 事件绑定: 拖拽元素的 mousedown 事件和 mousemove 事件通常绑定在 document 上,而不是拖拽元素本身。这是因为拖拽元素可能会在页面上移动,而 document 会捕获全局的鼠标事件,确保无论鼠标在哪里释放,都可以正确处理拖拽操作。

<!DOCTYPE html>
<html>
<head><style>.draggable {width: 100px;height: 100px;background-color: lightblue;text-align: center;line-height: 100px;position: absolute;cursor: grab;}</style>
</head>
<body><div class="draggable" id="myElement">拖拽我</div><script>const element = document.getElementById('myElement');let isDragging = false;let offsetX, offsetY;element.addEventListener('mousedown', (e) => {isDragging = true;offsetX = e.clientX - element.getBoundingClientRect().left;offsetY = e.clientY - element.getBoundingClientRect().top;element.style.cursor = 'grabbing';});document.addEventListener('mousemove', (e) => {if (isDragging) {element.style.left = e.clientX - offsetX + 'px';element.style.top = e.clientY - offsetY + 'px';}});document.addEventListener('mouseup', () => {isDragging = false;element.style.cursor = 'grab';});</script>
</body>
</html>

Web API 提供了内置的拖放

Web API 提供了内置的拖放功能,主要涉及到以下事件和方法,以及一些注意事项:

拖放的思路:

  1. ondragstart 事件: 当你想要元素成为可拖拽的源时,你可以监听 ondragstart 事件,该事件在用户开始拖动元素时触发。你可以设置 dataTransfer 对象的数据,以便在拖放操作中传递信息。

  2. ondragover 事件: 当你希望元素成为拖放目标时,你可以监听 ondragover 事件,该事件在拖动元素悬停在目标元素上时触发。你可以通过 ondragover 事件的默认行为阻止拒绝拖放,或者通过 JavaScript 来决定是否接受拖放操作。

  3. ondrop 事件: 当你希望在目标元素上接受拖放操作时,你可以监听 ondrop 事件,该事件在用户完成拖放操作时触发。在 ondrop 事件处理程序中,你可以访问拖放数据,执行相应的操作,以及取消默认行为。

注意事项:

  1. 阻止默认行为: 如果你希望某个元素成为拖放源或目标,通常需要在 ondragstartondragover 事件处理程序中使用 event.preventDefault() 阻止默认行为,以便正确处理拖放。

  2. 数据传递: 使用 dataTransfer 对象来传递数据,这可以是文本、URL、自定义数据等。在 ondragstart 事件中设置数据,然后在 ondrop 事件中获取数据。

  3. 拖放目标状态: 使用 ondragenterondragleave 事件来处理拖放目标的状态变化。这些事件在拖动元素进入和离开目标元素时触发。

  4. 拖放效果: 你可以使用 dataTransfer.effectAlloweddataTransfer.dropEffect 来指定拖放操作的效果,如移动、复制或链接。这有助于提供用户反馈。

  5. 跨浏览器兼容性: 考虑跨浏览器兼容性,因为不同浏览器可能会有一些差异,特别是在数据格式方面。

  6. 拖放元素样式: 为了提供视觉效果,你可以在 ondragstart 事件中更改拖放元素的样式,以使用户知道它可以被拖动。

代码示例:

<!DOCTYPE html>
<html>
<head><style>.draggable {width: 100px;height: 100px;background-color: lightblue;text-align: center;line-height: 100px;position: absolute;cursor: grab;}.droppable {width: 200px;height: 200px;background-color: lightgray;position: absolute;top: 150px;left: 150px;}</style>
</head>
<body><divclass="draggable"draggable="true"ondragstart="dragStart(event)">拖拽我</div><divclass="droppable"ondragover="dragOver(event)"ondrop="drop(event)">放置区域</div><script>function dragStart(event) {event.dataTransfer.setData("text", event.target.id);event.target.style.cursor = 'grabbing';}function dragOver(event) {event.preventDefault();}function drop(event) {event.preventDefault();const data = event.dataTransfer.getData("text");const draggableElement = document.getElementById(data);event.target.appendChild(draggableElement);draggableElement.style.cursor = 'grab';}</script>
</body>
</html>

在这个示例中,创建了两个元素:一个可拖拽的元素(draggable)和一个接受拖动的目标元素(droppable)。draggable 元素被标记为可拖拽(draggable=“true”),并且定义了 ondragstart 事件处理程序,它在拖拽开始时触发。

droppable 元素定义了 ondragover 和 ondrop 事件处理程序,它们用于处理拖放区域的事件。ondragover 阻止默认行为,以允许元素可以放置在该区域,而 ondrop 处理实际的拖放操作,将拖拽的元素追加到目标元素中。

在学习时看到其他值得一看的文章,把 HTML 拖放 API描述的很详细


http://www.ppmy.cn/news/1159861.html

相关文章

在C++和Python的项目中使用ROS

如果搜索如何使用ROS&#xff0c;搜索结果肯定是先建立工作空间&#xff0c;在创建功能包等等步骤&#xff0c;但其实不需要这么麻烦。 在Python中使用ROS&#xff0c;只需要在Pycharm的Project Structure中的Add Content Root加入ros的packages就可以了&#xff0c;如下图 在…

数据结构之手撕链表(讲解➕源代码)

0.引言 我们在学习过顺序表之后&#xff0c;会发现两点不是很优秀的操作&#xff1a; 1.顺序表的头插和中间的插入&#xff1a; 非常麻烦&#xff0c;需要不断的覆盖数据。 2.动态开辟空间&#xff1a; a.一般动态开辟的空间都是以2倍的形式开辟&#xff0c;当…

网格大师如何把b3dm转为osgb格式?

答&#xff1a;在网格大师的倾斜数据处理工具中选中“3DTiles转OSGB”&#xff0c;设定数据输入路径和输出路径提交任务即可。 网格大师是一款能够解决实景三维模型空间参考、原点、瓦块大小不统一&#xff0c;重叠区域处理问题的工具“百宝箱”&#xff0c;集格式转换、坐标转…

SpringBootCMS漏洞复现分析

SpringBootCMS&#xff0c;极速开发&#xff0c;动态添加字段&#xff0c;自定义标签&#xff0c;动态创建数据库表并crud数据&#xff0c;数据库备份、还原&#xff0c;动态添加站点(多站点功能)&#xff0c;一键生成模板代码&#xff0c;让您轻松打造自己的独立网站&#xff…

算法练习14——除自身以外数组的乘积

LeetCode 除自身以外数组的乘积 给你一个整数数组 nums&#xff0c;返回 数组 answer &#xff0c;其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&a…

LeetCode 2906. 构造乘积矩阵【前后缀分解,数组】中等

本文属于「征服LeetCode」系列文章之一&#xff0c;这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁&#xff0c;本系列将至少持续到刷完所有无锁题之日为止&#xff1b;由于LeetCode还在不断地创建新题&#xff0c;本系列的终止日期可能是永远。在这一系列刷题文章…

Python 批量修改文件名

仅个人需求&#xff0c;有需要的可以自取。 前段时间为家里孩子下载了一批课程&#xff0c;但是文件命名就很奇怪也很乱&#xff0c;就想着将文件名修改掉便于查看。 这批视频下载下来后前边都给了诸如001之类的编号&#xff0c;当然是序列。可是这批序列又非常的乱&#xff0…

NLG(自然语言生成)评估指标介绍

诸神缄默不语-个人CSDN博文目录 本文介绍自然语言生成任务中的各种评估指标。 因为我是之前做文本摘要才接触到这一部分内容的&#xff0c;所以本文也是文本摘要中心。 持续更新。 文章目录 1. 常用术语2. ROUGE (Recall Oriented Understudy for Gisting Evaluation)1. 计算…