10分钟掌握HTML拖放API!让你的网页元素瞬间拥有拖拽功能,轻松提升用户体验!

在日常开发中,拖拽功能无疑是一个常见的需求场景。为了更好地满足这一需求,HTML5提供了一套便捷的拖放API。这些API不仅能够帮助开发者轻松实现拖拽效果,还可以提高排查拖拽问题的效率,甚至可以让我们更加灵活地自定义拖拽场景和设计能力。

要想完全掌握拖放API的使用及常见问题,本文将从三个步骤出发,带你深入了解并掌握这一技术。

首先,我们将通过案例演示来直观地展示拖拽功能在实际应用中的表现,帮助你快速理解拖拽的基本概念和操作流程。其次,我们将详细解析拖放流程并介绍相关的API,包括拖放事件的类型、事件处理函数以及关键属性的使用方法等。最后,我们将通过案例实现和讲解来巩固所学知识,让你能够亲自动手实践,并将所学知识应用到实际项目中。

通过本文的学习,你将能够轻松掌握HTML5拖放API的使用技巧,提高拖拽功能的开发效率,为你的项目增色添彩。

扫码加入前端交流群,期待你的加入!
描述文字

一、案例演示

实现一个可拖放自定义课程表,将左侧的科目拖拽到右侧的课程表中,支持增加、修改、删除、移动功能。

在这里插入图片描述

二、API介绍

1. 拖放过程

整个拖放过程中,存在两个关键元素:拖拽元素、放置元素
拖拽元素:被拖拽的元素

  • ondrag:元素被拖拽时触发,从开始拖拽到拖拽结束前整个过程会一直持续的触发 ◦ondragstart:元素被拖拽开始时触发
  • ondrop:拖拽元素被放置到放置元素内时触发,如果没有在放置元素内松手,则不会触发

放置元素:

  • ondragenter:有拖拽元素进入时触发
  • ondragover:有拖拽元素在该元素上时触发,在离开前会持续触发
  • ondragleave:拖拽元素离开时触发
  • ondragend:拖拽元素放置时触发

在这里插入图片描述

2. 可拖拽元素

在HTML中,文本、图片和链接是默认可以拖放的元素。
其他元素都是默认不可拖动的,如果需要让其他非默认可拖动的HTML元素变得可拖动,比如<div><span>等,你需要明确地为这些元素设置 draggable="true" 属性。这样,这些元素就能够接受拖放操作了。

<div draggable="true">语文</div>

3. 放置元素

所有HTML元素在默认情况下都不接受拖拽元素的放置,除非通过特定的事件处理来允许。
要使一个HTML元素能够接受被拖动的元素,需要对这个元素进行一些特定的设置和事件绑定:

  • dragover事件:当被拖动的元素在另一个元素上方移动时,会触发dragover事件。为了接受拖放,必须在- -
  • dragover事件处理器中调用event.preventDefault()方法,这会阻止浏览器的默认行为,也就是不接受任何被拖放的元素。
container.ondragover  = (e) => {
    e.preventDefault();
}

4. DataTransfer

DataTransfer对象用于保存拖动并放下(drag and drop)过程中的数据。它可以保存一项或多项数据,这些数据项可以是一种或者多种数据类型

  • dropEffect:获取当前选定的拖放操作类型,或设置为一个新的类型。值必须为 none、copy、link、move
  • effectAllowed:提供所有可用的操作类型。值必须为none、copy、copyLink、copyMove、link、copyMove、move、all、uninitialized
  • files:包含数据传输中的所有本地文件列表
  • items(只读):提供一个包含所有拖动数据列表的DataTransferItemList对象
  • types(只读):一个提供 dragstart 事件中设置的格式的 strings 数组

三、案例实现

1.把左侧科目设置为可拖拽

<div class="content">
    <div class="left"  data-drop="move">
        <div draggable="true" style="background:rgb(26, 231, 156)">语文</div>
        <div draggable="true" style="background:rgb(107, 219, 15)">数学</div>
        <div draggable="true" style="background:rgb(208, 133, 13)">英语</div>
        <div draggable="true" style="background:rgb(30, 98, 246)">物理</div>
        <div draggable="true" style="background:rgb(210, 40, 113)">化学</div>
        <div draggable="true" style="background:rgb(210, 224, 26)">生物</div>
    </div>
    <div class="right">
        <table>
            <thead>
                <tr>
                    <td>星期一</td>
                    <td>星期二</td>
                    <td>星期三</td>
                    <td>星期四</td>
                    <td>星期五</td>
                </tr>
            </thead>
            <body>
                <tr>
                    <td></td>
                    <td></td>
                    <td></td>
                    <td></td>
                    <td></td>
                </tr>
                <tr>
                    <td></td>
                    <td></td>
                    <td></td>
                    <td></td>
                    <td></td>
                </tr>
                <tr>
                    <td></td>
                    <td></td>
                    <td></td>
                    <td></td>
                    <td></td>
                </tr>
                <tr>
                    <td></td>
                    <td></td>
                    <td></td>
                    <td></td>
                    <td></td>
                </tr>
            </body>
        </table>
    </div>
</div>

2. 绑定事件

这里采用事件委托的方式,对整个父元素进行事件绑定

const container = document.querySelector('.content');
ondragover:让整个容器内的所有元素都可以接受拖拽元素,这里只需要阻止默认行为就可以
container.ondragover  = (e) => {
    e.preventDefault();
}

ondragstart:拖拽开始时,需要做以下处理

  • 拖拽元素原来的位置样式有些变化,这里就是整体颜色变得更透明
  • 从左侧拖到右侧,为新增,鼠标显示新增的手势。右侧拖到左侧为移除,显示普通手势。(拖拽过程中默认为新增手势)
  • 保存当前拖拽元素,后续操作需要用到

在这里插入图片描述

通过data属性给拖拽元素添加标识,标识此元素是新增还是移除,新增设置为 data-effect=“copy”

<div class="left"  data-drop="move">
    <div data-effect="copy" draggable="true" style="background:rgb(26, 231, 156)">语文</div>
    <div data-effect="copy" draggable="true" style="background:rgb(107, 219, 15)">数学</div>
    <div data-effect="copy" draggable="true" style="background:rgb(208, 133, 13)">英语</div>
    <div data-effect="copy" draggable="true" style="background:rgb(30, 98, 246)">物理</div>
    <div data-effect="copy" draggable="true" style="background:rgb(210, 40, 113)">化学</div>
    <div data-effect="copy" draggable="true" style="background:rgb(210, 224, 26)">生物</div>
</div>
let source = '';
container.ondragstart = (e) => {
    // 如果是移除操作,则设置手势为移除
    if (e.target.dataset.effect === "move") {
        e.dataTransfer.effectAllowed = "move";
    }
    source = e.target;
    // 设置拖拽元素的样式
    e.target.style.opacity = '0.2'
}

ondragenter:进入放置元素,需要做以下处理

  • 判断该元素以及该元素的所有上级元素是否可以接收拖拽元素。哪些元素可以接收,也通过data属性进行标识 data-drop=“copy”
  • 如果可以接收拖拽元素,需要设置其样式用于标识
<tr>
    <td data-drop="copy"></td>
    <td data-drop="copy"></td>
    <td data-drop="copy"></td>
    <td data-drop="copy"></td>
    <td data-drop="copy"></td>
</tr>
container.ondragenter  = (e) => {
    const dropNode = getDropNode(e.target);
    // 判断放置元素是否可以接收拖拽元素,当 data-effect 和 data-drop 的值相等时,说明可以
    if (dropNode && dropNode.dataset.drop === source.dataset.effect) {
        // 给放置元素添加样式
        dropNode.classList.add('hover-background');
    }
}

// 获取最近的可接受拖拽元素的父节点
function getDropNode(node) {
    while (node) {
        //  判断元素是否设置了data-drop属性,如果设置了,说明此元素是一个放置元素
        if (node.dataset && node.dataset.drop) {
            return node;
        }
        node = node.parentNode;
    }
    return node;
}

此时会发现,所有经过的放置元素背景色都发生了变化
在这里插入图片描述

每次在进入元素时,清除所有的背景色。

container.ondragenter  = (e) => {
    const dropNode = getDropNode(e.target);
    // 判断放置元素是否可以接收拖拽元素,当 data-effect 和 data-drop 的值相等时,说明可以
    if (dropNode && dropNode.dataset.drop === source.dataset.effect) {
        // 给放置元素添加样式
        dropNode.classList.add('hover-background');
    }
}

function clearHoverClass() {
    document.querySelectorAll('.hover-background').forEach(ele => ele.classList.remove("hover-background"));
}

ondrop:松手放置时
对放置元素的处理

  • 清除hover时的背景样式
  • 【新增课程】清除放置元素内的内容;克隆拖拽元素;设置克隆后的元素data-effect属性只为move;恢复拖拽元素本身的样式;添加到放置元素中
  • 【移除课程】删除掉拖拽元素

对放置元素的处理

  • 恢复元素样式
// 对放置元素触发的事件
container.ondrop = (e) => {
    // 清除hover时的样式
    clearHoverClass();
    // 获取最近的放置节点
    const dropNode = getDropNode(e.target);
    if (dropNode && dropNode.dataset.drop === source.dataset.effect) {
        // 如果是新增课程
        if (dropNode.dataset.drop === "copy") {
            dropNode.innerHTML = "";
            const cloned = source.cloneNode(true);
            cloned.dataset.effect = "move";
            cloned.style.opacity = "1";
            dropNode.appendChild(cloned);
        // 移除课程
        } else {
            source.remove();
        }
    }
}

// 对拖拽元素触发的事件
container.ondragend = (e) => {
    e.target.style.opacity = "1";
}

写在最后

可以到我的个人网站(www.dengzhanyong.com)的资源中心中下载本地案例的完整源码
扫码加入前端交流群,期待你的加入!
描述文字

欢迎关注我的公众号,不错过每一篇推送
在这里插入图片描述

  • 30
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
实现自动生成相框的功能,需要使用HTML5的拖放API,并结合Canvas绘图功能。 以下是一个简单的实现步骤及代码示例: 1. HTML结构 ```html <div id="frame-container"> <canvas id="canvas"></canvas> <img id="image" src="your-image-url" draggable="true"> </div> ``` 2. JS代码 ```javascript // 获取canvas和image元素 var canvas = document.getElementById('canvas'); var ctx = canvas.getContext('2d'); var image = document.getElementById('image'); // 设置canvas的宽高为image的宽高 canvas.width = image.width; canvas.height = image.height; // 监听image元素拖放事件 image.addEventListener('dragstart', function(e) { // 将拖放的数据设置为image元素的src属性值 e.dataTransfer.setData('text/plain', image.src); }); // 监听canvas元素拖放事件 canvas.addEventListener('dragover', function(e) { // 阻止默认事件,允许拖放 e.preventDefault(); }); canvas.addEventListener('drop', function(e) { // 阻止默认事件,允许拖放 e.preventDefault(); // 获取拖放的数据 var imageUrl = e.dataTransfer.getData('text/plain'); // 创建一个新的Image对象,加载拖放的图片 var newImage = new Image(); newImage.src = imageUrl; // 在canvas上绘制相框和图片 newImage.onload = function() { // 绘制相框 ctx.strokeStyle = '#000'; ctx.lineWidth = 10; ctx.strokeRect(0, 0, canvas.width, canvas.height); // 绘制图片 ctx.drawImage(newImage, 10, 10, canvas.width - 20, canvas.height - 20); } }); ``` 通过以上代码,当用户将图片拖放到canvas元素上时,会在canvas上生成一个相框,并将拖放的图片绘制在相框内部。 需要注意的是,由于浏览器安全策略的限制,当使用本地图片时,需要使用本地服务器进行测试,否则会报错。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

前端筱园

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值