H5中的draggable

基本语法及事件

draggable 属性规定元素是否可拖动。必须设置,否则没有拖拽效果及事件触发

提示: 链接和图像默认是可拖动的。

提示: draggable 属性经常用于拖放操作

语法 

<element draggable="true|false|auto">
描述
true规定元素是可拖动的。
false规定元素是不可拖动的。
auto使用浏览器的默认特性。

在拖放过程中会触发以下事件:

在拖动目标上触发事件(源元素),作用于被拖拽元素上

  • ondragstart - 开始拖动元素触发
  • ondrag - 元素正在拖动时触发
  • ondragend - 用户完成元素拖动后触发 
<div id="app">
  <div class="drag" draggable="true"></div>
  <div class="target"></div>
</div>

<script>
  const drag = document.querySelector('.drag')
  const target = document.querySelector('.target')

  drag.addEventListener('dragstart', ondragstart)
  drag.addEventListener('drag', ondrag)
  drag.addEventListener('dragend', ondragend)
  function ondragstart(event) {
    console.log('开始拖动')
  }
  function ondrag(event) {
    console.log('正在拖动')
  }
  function ondragend(event) {
    console.log('完成拖动')
  }
</script>

作用于目标元素

  • ondragenter - 进入其容器范围内触发(当被鼠标拖动的源元素进入目标容器范围内时触发) 
  • ondragover - 被拖动的源对象在目标容器中拖动时触发
  • ondragleave - 拖动的对象离开目标容器时触发
  • ondrop - 释放鼠标时触发

Event.preventDefault 

阻止默认事件,在ondragover中一定要执行preventDefault方法,否则ondrop不会被触发

<div id="app">
  <div class="drag" draggable="true"></div>
  <div class="target"></div>
</div>

<script>
  const drag = document.querySelector('.drag')
  const target = document.querySelector('.target')

  target.addEventListener('dragenter', ondragenter)
  target.addEventListener('dragover', ondragover)
  target.addEventListener('dragleave', ondragleave)
  target.addEventListener('drop', ondrop)
  function ondragenter(event) {
    console.log('进入入其容器范围内触发')
  }
  function ondragover(event) {
    event.preventDefault()
    console.log('容器中拖动时触发')
  }
  function ondragleave(event) {
    console.log('离开目标容器时触发')
  }
  function ondrop(event) {
    console.log('释放鼠标时触发')
  }
</script>

DataTransfer

在进行拖放操作的时候,dataTransfer对象可以用来保存被拖动的数据。它可以保存一项或多项数据、一种或多数数据类型。通谷一点讲,就是可以通过它来传输被拖动的数据,以便在拖拽结束的时候,对数据进行其他的操作。

  • setData(type, data): 用于声明所发送的数据
  • getData(type):用于获取指定type的数据 注意:只能在ondrop中获取的到
  • clearData(type):用于清除指定类型的数据
<div id="app">
  <div class="drag" draggable="true"></div>
  <div class="target"></div>
</div>

<script>
  const drag = document.querySelector('.drag')
  const target = document.querySelector('.target')

  drag.addEventListener('dragstart', dragstart)
  target.addEventListener('dragover', ondragover)
  target.addEventListener('drop', ondrop)

  function dragstart(event) {
    event.dataTransfer.setData('comp', JSON.stringify({name: 'WFT'}))
  }
  function ondragover(event) {
    event.preventDefault()
  }
  function ondrop(event) {
    let data = JSON.parse(event.dataTransfer.getData('comp'))
    console.log(data)
    // ...
    // 处理完数据 最后清除一下
    event.dataTransfer.clearData()
  }
</script>

在其它的事件(如ondragoverondragleave等),是无法获取dataTransfer里面的值了。这是由于W3C要求对dataTransfer里的值进行保护[参考]。因此,如果需要在这些事件里获取数据,只能通过一个全局变量等其它方式来实现了。

event.dataTransfer.setDragImage(p_w_picpath, x, y)

setDragImage方法用于在拖放操作过程中,修改鼠标指针所指向的图像

示例一: 

先看效果: 

先说一下,这种和鼠标去拖拽某个元素跟着鼠标移动一直移动不一样的,那种核心用到三个鼠标事件是 onmousedown、onmousemove、onmouseup。咱们这种是拖拽、就是本身是不动的状态 

 

下面是完整代码: 

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Document</title>
</head>
<body>
  <div id="app">
    <div class="drag" draggable="true"></div>
    <div class="target"></div>
  </div>

  <script>
    const drag = document.querySelector('.drag')
    const target = document.querySelector('.target')

    drag.addEventListener('dragstart', dragstart)
    target.addEventListener('dragover', ondragover)
    target.addEventListener('drop', ondrop)

    let curDragDom = null
    let curOffsetX = 0
    let curOffsetY = 0
    
    // 开始拖拽
    function dragstart(event) {
      curDragDom = this
      curOffsetX = event.offsetX
      curOffsetY = event.offsetY
    }
    // 在目标元素中拖拽 移除掉默认事件 否则不会触发 drop事件
    function ondragover(event) {
      event.preventDefault()
    }
    // 在目标容器中松开鼠标
    function ondrop(event) {
      if(!curDragDom) return
      let copyDom = curDragDom.cloneNode(true)
      copyDom.style.position = 'absolute'
      const { x, y } = this.getBoundingClientRect()
      copyDom.style.left = event.clientX - x - curOffsetX + 'px'
      copyDom.style.top = event.clientY - y - curOffsetY + 'px'
      copyDom.removeAttribute('draggable')
      this.appendChild(copyDom)
    }
  </script>

  <style>
    * {
      padding: 0;
      margin: 0;
      box-sizing: border-box;
    }
    #app {
      width: 100vw;
      height: 100vh;
      padding: 50px 0 0 50px;
      box-sizing: border-box;
      display: flex;
    }
    .drag {
      width: 100px;
      height: 100px;
      background-color: red;
    }
    .target {
      width: 400px;
      height: 400px;
      border: 1px solid red;
      margin-left: 200px;
      position: relative;
      overflow: hidden;
    }
  </style>
</body>
</html>

示例二: 

 

完整代码: 

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Document</title>
</head>
<body>
  <div id="app">
    <div class="upload" draggable="true">
      <div>将文件拖到此处进行上传</div>
    </div>
    <div class="images"></div>
  </div>

  <script>
    const upload = document.querySelector('.upload')
    const imageContainer = document.querySelector('.images')

    upload.addEventListener('dragover', ondragover)
    upload.addEventListener('drop', ondrop)

    function ondragover(event) {
      event.preventDefault()
    }

    function ondrop(event) {
      event.preventDefault()
      const { files } = event.dataTransfer
      Array.from(files).forEach(blob => {
        const url = window.URL.createObjectURL(new Blob([blob]))
        const img = document.createElement('img')
        img.src = url
        imageContainer.appendChild(img)
      })
    }
  </script>

  <style>
    * {
      padding: 0;
      margin: 0;
      box-sizing: border-box;
    }
    #app {
      width: 100vw;
      height: 100vh;
      padding: 50px 0 0 50px;
      box-sizing: border-box;
    }
    .upload {
      width: 360px;
      height: 180px;
      border: 1px dashed #c0c4cc;
      border-radius: 5px;
      display: flex;
      justify-content: center;
      align-items: center;
    }
    .upload > div {
      color: #606266;
      font-size: 14px;
    }
    .images {
      width: 100%;
      display: flex;
      flex-wrap: wrap;
      margin-top: 10px;
    }
    .images img {
      width: 300px;
      height: 180px;
      margin-right: 20px;
    }
  </style>
</body>
</html>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/62644.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

在服务器上搭建gitlab

目录 1.在服务器上下载gitlab 2.编辑站点位置 3.重载配置 4.访问gitlab 最终效果展示&#xff1a; 官方文档&#xff1a; 安装部署GitLab服务 1.在服务器上下载gitlab wget https://mirrors.tuna.tsinghua.edu.cn/gitlab-ce/yum/el7/gitlab-ce-12.9.0-ce.0.el7.x86_64.r…

Wavefront .OBJ文件格式解读【3D】

OBJ&#xff08;或 .OBJ&#xff09;是一种几何定义文件格式&#xff0c;最初由 Wavefront Technologies 为其高级可视化器动画包开发。 该文件格式是开放的&#xff0c;已被其他 3D 图形应用程序供应商采用。 OBJ 文件格式是一种简单的数据格式&#xff0c;仅表示 3D 几何体&…

Linux文本处理工具和正则表达式

Linux文本处理工具和正则表达式 一.查看、截取和修改文本的工具 1.查看文本的工具 cat 最常用的文件查看命令&#xff1b;当不指明文件或者文件名为一杠’-时&#xff0c;读取标准输入。 cat [OPTION]... [FILE]... -A&#xff1a;显示所有控制符(tab键:^I;行结束符:$) -…

01Mysql创建表

目录 一、题目要求 二、具体操作代码 三、查看结果&#xff1a; 一、题目要求 建立一张表&#xff1a; 表里面有多个字段&#xff0c;每一个字段对应一种数据类型 注意&#xff1a;表名&#xff0c;字段名都要起的有意义 二、具体操作代码 CREATE TABLE DataInfo (id INT …

Vercel 部署的项目发现APIkeys过期了怎么办

好不容易部署的Vercel&#xff0c;发现APIkeys过期了显示&#xff0c;查了查资料发现只要更新下新的apikeys&#xff0c;然后再重新部署下就好了。 重新设置APIkeys 1.1. 进去 Vercel 项目内部控制台&#xff0c;点击顶部的 Settings 按钮&#xff1b; 1.2 点击环境变量Enviorn…

计算机网络-性能指标

计算机网络-性能指标 文章目录 计算机网络-性能指标简介速率比特速率 带宽吞吐量时延时延计算 时延带宽积往返时间网络利用率丢包率总结 简介 性能指标可以从不同的方面来度量计算机网络的性能 常用的计算机网络的性能指标有以下8个 速率带宽吞吐量时延时延带宽积往返时间利…

远程连接身份验证错误,又找不到加密Oracle修正

一、问题描述 远程连接服务器出现了错误&#xff0c;错误信息为&#xff1a;远程连接身份验证错误&#xff0c;又找不到加密Oracle修正。 二、原因分析 出错原因&#xff1a;Windows的CVE-2018-0886 的 CredSSP 更新将CredSSP 身份验证协议默认设置成了“缓解”&#xff0c;…

华为发布数字资产继承功能

在华为开发者大会2023&#xff08;HDC.Together&#xff09;上&#xff0c;华为常务董事、终端BG CEO、智能汽车解决方案BU CEO余承东正式发布了数字资产继承功能&#xff0c;HarmonyOS提供了安全便捷的数字资产继承路径。 在鸿蒙世界中&#xff0c;我们每个人在每台设备、应用…

【C++】C++11 新特性总结 | C++ 常见设计模式总结(秋招篇)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言介绍几种C11新特性介绍一下自动类型推导auto和decltype关键字的用法举例讲一下范围基于的for循环介绍一下列表初始化讲一下右值引用&#xff0c;和左值引用的区…

微服务 云原生:搭建 Harbor 私有镜像仓库

Harbor官网 写在文前&#xff1a; 本文中用到机器均为虚拟机 CentOS-7-x86_64-Minimal-2009 镜像。 基础设施要求 虚拟机配置达到最低要求即可&#xff0c;本次系统中使用 docker 24.0.4、docker-compose 1.29.2。docker 及 docker-compose 的安装可以参考上篇文章 微服务 &am…

Java基础入门篇——Java虚拟机和运行环境(一)

目录 一、Java的发展简史和语言特点 1.1发展简史&#xff1a; 1.2语言特点&#xff1a; 二、Java运行机制 三、Java虚拟机 四、Java的运行环境 五、第一个Java程序——HelloWorld&#xff01; 六、总结 一、Java的发展简史和语言特点 Java是一种广泛使用的编程语言&a…

并查集练习 — 岛屿问题(二)

题目&#xff1a; 同样是岛的问题&#xff0c;但是参数有所变化&#xff0c;一共3个参数&#xff0c;m、n、int[][] position。根据position&#xff0c;求出每一步的岛屿的数量。 代表的意思是&#xff1a;m * n是二维数组的行和列&#xff0c;通过 m * n可以构建一个值都为0的…

python 合并多个excel文件

使用 openpyxl 思路&#xff1a; 读取n个excel的文件&#xff0c;存储在一个二维数组中&#xff0c;注意需要转置。将二维数组的数据写入excel。 安装软件&#xff1a; pip install openpyxl源代码&#xff1a; import os import openpyxl # 将n个excel文件数据合并到一个…

Android性能优化—数据结构优化

优化数据结构是提高Android应用性能的重要一环。在Android开发中&#xff0c;ArrayList、LinkedList和HashMap等常用的数据结构的正确使用对APP性能的提升有着重大的影响。 一、ArrayList ArrayList内部使用的是数组&#xff0c;默认大小10&#xff0c;当数组长度不足时&…

Spring源码——初识Spring容器

Spring源码之工厂&#xff08;容器&#xff09; 为什么把Spring的工厂又叫做容器呢&#xff1f; 工厂的责任是创建对象&#xff0c;但是创建完对象后还要进行存储&#xff08;针对于单例的对象来讲&#xff09;&#xff0c;以供其他地方使用&#xff0c;这就是容器。为了能存…

设计模式行为型——迭代器模式

什么是迭代器模式 迭代器模式&#xff08;Iterator Pattern&#xff09;属于行为型模式&#xff0c;其提供一种方法顺序访问一个聚合对象中的各种元素&#xff0c;而又不暴露该对象的内部表示&#xff0c;即不需要知道集合对象的底层表示。编程环境中非常常用的设计模式。 迭代…

详解PHP反射API

PHP中的反射API就像Java中的java.lang.reflect包一样。它由一系列可以分析属性、方法和类的内置类组成。它在某些方面和对象函数相似&#xff0c;比如get_class_vars()&#xff0c;但是更加灵活&#xff0c;而且可以提供更多信息。反射API也可与PHP最新的面向对象特性一起工作&…

谷歌语音助手战略调整:开发 AI 新版,调整裁员计划

北京时间8月2日晚间&#xff0c;谷歌通过对 “谷歌助手” 团队进行调整和裁员&#xff0c;意图改变其开发方向。经过此次变动&#xff0c;谷歌计划借助最新的生成式人工智能技术和大型语言模型来提升 谷歌助手 的能力。此次调整表明语音助手市场未达到先前的预期。 亚马逊旗下的…

【从零开始学习JAVA | 三十四篇】IO流

目录 前言&#xff1a; IO流介绍&#xff1a; IO流的常见方法&#xff1a; 1.字节流类&#xff1a; 2.字符流类&#xff1a; 总结&#xff1a; 前言&#xff1a; IO流就是存入和读取数据的解决方案&#xff0c;并且他是一个知识点很多的章节&#xff0c;因此我们关于IO流…

Apache+Tomcat 整合

目录 方式一&#xff1a;JK 1、下载安装包 2、添加依赖 3、启动服务&#xff0c;检查端口是否监听 4、提供apxs命令 5、检查是否确实依赖 6、编译安装 7、重要配置文件 方式二&#xff1a;http_proxy 方式三&#xff1a;ajp_proxy 方式一&#xff1a;JK 1、下载安装…