在日常开发中,拖拽功能无疑是一个常见的需求场景。为了更好地满足这一需求,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)的资源中心中下载本地案例的完整源码
扫码加入前端交流群,期待你的加入!
描述文字

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

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐