原生 JS 快速实现拖放

拖放是很常见的一种交互效果,很多时候我们都会借助于第三方的控件来实现,其实用原生 js 实现起来也非常的方便。接下来我们就用原生 js 和 css 快速实现拖放效果

html

<body>
<div class="droppable">
<div class="draggable" draggable="true"></div>
</div>
<div class="droppable"></div>
<div class="droppable"></div>
<div class="droppable"></div>
<div class="droppable"></div>
</body>

注意点: 1. 容器的的 class 为 droppable,用于接收被拖拽的元素,可被拖拽的元素 class 为 draggable,同时设置 draggable 属性为 true,表示该元素可以被拖拽。 2. 默认情况下,只有图片、链接还有被选中的文字能被拖拽,其他元素需要设置 draggable 为 true 才能被拖拽。所以为了凸显 draggable 的用法,这里使用 <div> 而不是 <image> 来作为被拖拽的元素。

css

在实现样式的时候,除了实现静态的样式,一些过渡状态也需要增加样式以提升视觉体验:

  1. 元素被拖动的过程中增加边框等效果; 2. 当元素被拖动到容器上方时,容器也应增加样式表明容器可以接收一个被拖拽的元素。
body {
background-color: darksalmon;
} .draggable {
background-image: url("http://source.unsplash.com/random/150x150");
position: relative;
height: 150px;
width: 150px;
top: 5px;
left: 5px;
cursor: pointer;
} .droppable {
display: inline-block;
height: 160px;
width: 160px;
margin: 10px;
border: 3px salmon solid;
background-color: white;
} .dragging {
border: 4px yellow solid;
} .drag-over {
background-color: #f4f4f4;
border-style: dashed;
} .invisible {
display: none;
}

注意点: 1. 图片来源于 百度; 2. .dragging 为 draggable 元素正在被拖动的状态,增加黄色 border; 3. .drag-over 为 draggable 元素被拖动到容器上方时容器的状态,增加灰色虚线 border。

js

最后,我们需要通过 js 监听 draggable 和 droppable 的相关的事件。

// 查询draggable和droppable
const draggable = document.querySelector(".draggable");
const droppables = document.querySelectorAll(".droppable"); // 监听draggable的相关事件
draggable.addEventListener("dragstart", dragStart);
draggable.addEventListener("dragend", dragEnd); function dragStart() {
this.className += " dragging";
setTimeout(() => {
this.className = "invisible";
}, 0);
} function dragEnd() {
this.className = "draggable";
} // 监听droppable的相关事件
for (const droppable of droppables) {
droppable.addEventListener("dragover", dragOver);
droppable.addEventListener("dragleave", dragLeave);
droppable.addEventListener("dragenter", dragEnter);
droppable.addEventListener("drop", dragDrop);
} function dragOver(e) {
e.preventDefault();
} function dragEnter(e) {
e.preventDefault();
this.className += " drag-over";
} function dragLeave(e) {
this.className = "droppable";
} function dragDrop(e) {
this.className = "droppable";
this.append(draggable);
}

注意点: 1. 当 draggable 元素被拖动时,原来容器中的 draggable 元素并不会消失,需要我们手动将其隐藏(class 设置为 invisible),如果同步操作会立马触发 dragend 事件导致拖动效果消失,所以在 setTimeout 的回调中异步设置可确保拖动操作开始后再隐藏 draggable 元素; 2. 在 dragEnter 和 dragOver 方法中我们需要通过 preventDefault 来取消事件以表明容器是一个合法的 droppable 元素,不然容器的 drop 事件将无法触发,接下来的操作也将无法进行,详细解释请参考 MDN DropTarget; 3.在 dragDrop 方法中直接使用 append 方法将 draggable 元素移动至当前容器下面。


完整示例演示:https://codehhr.gitee.io/web/jsdrag/

The_End

原生JS快速实现拖放的更多相关文章

  1. 原生js快速渲染dom节点

    function renderDom(str){ var _div = document.createElement('div'); _div.innerHTML = str; var dom_tem ...

  2. 原生JS实现"旋转木马"效果的图片轮播插件

    一.写在最前面 最近都忙一些杂七杂八的事情,复习软考.研读经典...好像都好久没写过博客了... 我自己写过三个图片轮播,一个是简单的原生JS实现的,没有什么动画效果的,一个是结合JQuery实现的, ...

  3. 【前端性能】必须要掌握的原生JS实现JQuery

    很多时候,我们经常听见有人说jquery有多快多快.在这个各种类库满天飞的时候,不得不说的是,能有原生JS快吗? 是的,明显原生JS要更快,因为诸如JQuery这样的库必须要兼容各种浏览器和低版本和许 ...

  4. 用原生JS读写CSS样式的方法总结

    为了日后方便查询,本人翻阅了一些资料总结了以下方法,仅限原生JS,如有不对的地方欢迎指出!只求大家看完觉得有学到点什么就OK了!   一.可以通过DOM节点对象的style对象(即CSSStyleDe ...

  5. AJAX请求和跨域请求详解(原生JS、Jquery)

    一.概述 AJAX 是一种在无需重新加载整个网页的情况下,能够更新部分网页的技术. AJAX = 异步 JavaScript 和 XML,是一种用于创建快速动态网页的技术.通过在后台与服务器进行少量数 ...

  6. 原生js仿jquery--animate效果

    效果 代码 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF ...

  7. JQuery&原生js ——实现剪刀石头布小游戏

    前言 jQuery是一个快速.简洁的JavaScript框架,是继Prototype之后又一个优秀的JavaScript代码库( 或JavaScript框架).jQuery设计的宗旨是“write L ...

  8. 通过游戏认识 --- JQuery与原生JS的差异

      前言 jQuery是一个快速.简洁的JavaScript框架,是继Prototype之后又一个优秀的JavaScript代码库( 或JavaScript框架).jQuery设计的宗旨是“write ...

  9. 原生js写ajax请求(复习)

    今天本地想测试一个接口,不想用框架想用js快速完成,突然发现,我居然忘了这个最基本的代码.好吧,只能复习一波. 在框架泛滥的今天,用惯$.ajax(),axios,superAgent等框架的你们,还 ...

随机推荐

  1. 最短Hamilton路径(状压dp)

    最短Hamilton路径实际上就是状压dp,而且这是一道作为一个初学状压dp的我应该必做的题目 题目描述 给定一张 n(n≤20) 个点的带权无向图,点从 0~n-1 标号,求起点 0 到终点 n-1 ...

  2. HDU 3449 依赖背包

    这道题虽然水水的,但是还是成功地给我增加了10多个WA. 最开始拿着题,一看,依赖背包嘛~直接DFS树形DP嗨起来,甚至连内存都没有算一下,3MLE: 然后又仔细看了一下题,没有必要用树形背包来做嘛, ...

  3. Codeforces Round #647 (Div. 2) A. Johnny and Ancient Computer

    题目链接:https://codeforces.com/contest/1362/problem/A 题意 有一个正整数 $a$,可选择的操作如下: $a \times 2$ $a \times 4$ ...

  4. Educational Codeforces Round 88 (Rated for Div. 2) E. Modular Stability(数论)

    题目链接:https://codeforces.com/contest/1359/problem/E 题意 有一大小为 $k$ 的数组,每个元素的值在 $[1,n]$ 间,若元素间两两不等,问有多少数 ...

  5. HDU 3537 Daizhenyang's Coin 翻硬币博弈

    题意: 给你n个硬币,你可以从中拿出来1.2.3个硬币,它们不一定要连续,你只需要保证拿出来的硬币中那个下标最大的硬币一定要是正面朝上,最后谁不能操作,谁就输了 题解: 翻硬币游戏 结论: 局面的SG ...

  6. .net webapi 中使用session是出错 HttpContext.Current.Session==null

    最近在写.net webapi时发现 HttpContext.Current.Session==null  ,导致报错,后来查资料发现webapi中使用session时首先需要开启session功能, ...

  7. Netty(三)基于Bio和Netty 的简易版Tomcat

    参考代码: https://github.com/FLGBetter/tomcat-rpc-demo

  8. php 安装 yii framework notice-error 的解决方案!

    1 问题描述: 2 解决方案: error_reporting(0); //解决error_notice 的最简单最有效的方法在每一个php文件的头部都加上error_reporting(0); 3. ...

  9. Navigator.registerProtocolHandler All In One

    Navigator.registerProtocolHandler All In One Web API custom protocol URL Schemes URL Protocols https ...

  10. Swift private(set) All In One

    Swift private(set) All In One SwiftUI Getters and Setters https://docs.swift.org/swift-book/Language ...