拖拽页面元素+flip动画的案例

news2025/2/26 13:31:50

先上效果: 

实现思路和流程:

  1. 基础页面布局 给每个拖动元素加上 draggable="true"
  2. ondragstart(开始拖动某个元素时)做出 对应的处理 获得操作的具体元素 给目标元素添加对应的样式 显示透明 增加虚线描边
  3. ondragover 被拖动的元素hover到目标元素上时触发 阻止默认事件-默认不让元素拖动到自身
  4. ondragenter 拖动进行当中 对比 当前拖动的元素和 正在覆盖元素的索引 来判断操作 是上升还是下降
  5. ondragend 在结束时 样式由内部透明 虚线转为 原来的样子
  6. 配置flip动画

 基础页面布局:

<div class="list">
  <div draggable="true" class="list_item">1</div>
  <div draggable="true" class="list_item">2</div>
  <div draggable="true" class="list_item">3</div>
  <div draggable="true" class="list_item">4</div>
  <div draggable="true" class="list_item">5</div>
</div>

<style>
    .list{
      width: 750px;
      margin: 40px auto;
    }
    .list_item{
      width: 100%;
      border-radius: 8px;
      height: 52px;
      margin-bottom: 12px;
      background: rgba(40,142,145,0.9);
      color: white;
      line-height: 52px;
      padding-left: 16px;
      font-size: 18px;
      box-sizing: border-box;
      cursor: move;
      /*user-select: none;*/
    }
    .moving{
      background: transparent;
      color: transparent;
      border: 1px  dashed #ccc;
    }
</style>

实现拖动步骤

获取到总的外容器 便于下面事件委托
const list = document.querySelector('.list')
获取到所有可拖动的元素 用于记录起始位置
const item = document.querySelectorAll('.list_item')
let  sourceNode;  判断当前拖动的是哪个元素
 开始拖动的事件
list.ondragstart = e =>{
  sourceNode = e.target
  record(item)  传入item 记录起始位置
  setTimeout(()=>{
    e.target.classList.add('moving')
  },0)
  e.dataTransfer.effectAllowed = 'move'
}
list.ondragover = e => {
  e.preventDefault()
}
 拖动进行中的事件
list.ondragenter = e =>{
  e.preventDefault()
   托回到原来的位置了就什么也不做
  if(e.target === list || e.target === sourceNode){
    return false
  }
  const children = Array.from(list.children)
  const sourceIndex = children.indexOf(sourceNode)  当前劫持元素的索引值
  const targetIndex = children.indexOf(e.target)   覆盖到谁上面的索引值
  if(sourceIndex < targetIndex){
       父节点.insertBefore(要插入的节点,在谁前面) 从下向上拖动
    list.insertBefore(sourceNode,e.target.nextElementSibling)
  }else {
    list.insertBefore(sourceNode,e.target)
  }
  last([e.target,sourceNode]) 传入改变位置的两个元素 比较差异 执行filp动画
}
 拖动结束的时候取消虚线
list.ondragend = e =>{
  e.target.classList.remove('moving')
}

filp动画的函数

// 记录初始位置
function record(eleAll) {
  for( let i = 0;i < eleAll.length; i++ ) {
    const { top,left } = eleAll[i].getBoundingClientRect()
    eleAll[i]._top_ = top
    eleAll[i]._left_ = left
  }
}

/*  getBoundingClientRect()用于获得页面中某个元素的左,上,右和下分别相对浏览器视窗(可视范围不包含卷去的部分)的位置。*/
/**
 * requestAnimationFrame 比起 setTimeout、setInterval的优势主要有两点:
 1、requestAnimationFrame 会把每一帧中的所有DOM操作集中起来,在一次重绘或回流中就完成,并且重绘或回流的时间间隔紧紧跟随浏览器的刷新频率,一般来说,这个频率为每秒60帧。
 2、在隐藏或不可见的元素中,requestAnimationFrame将不会进行重绘或回流,这当然就意味着更少的的cpu,gpu和内存使用量。

 取消:cancelAnimationFrame(Id)

 * **/




​​​​​​​

// 记录最后的位置 并且执行动画
function last(eleAll) {
  for( let i = 0;i < eleAll.length; i++ ) {
    const dom = eleAll[i]
    const { top,left } = dom.getBoundingClientRect()
    // 新增dom时,逻辑应为 原有dom后移动,新增dom不动,故记录了位置的才添加动画 确定上一步有记录起始位置再进行下一步
    if(dom._left_) {
      // 恢复至开始位置
      dom.style.transform = `translate3d(${ dom._left_ - left }px, ${ dom._top_ - top }px,0px)`
      // play 过程,移除开始位置的设置,添加过渡
      let rafId = requestAnimationFrame(function() {
        //启用tansition,并移除翻转的改变  可以内置样式也可以用 外部类
         //dom.classList.add('active')
        dom.style.transition = 'transform 300ms ease-out'
        dom.style.transform = 'none'
      })
      dom.addEventListener('transitionend', () => {
        dom.style.transition = 'none'
         //dom.classList.remove('active')
        cancelAnimationFrame(rafId)
      })
    }
  }
}
 flip 动画思路
 f - first 记录动画开始前的位置、大小等信息 ( translateY(0px) )
 l - last  记录动画结束时的位置、大小等信息 ( translateY(100px) )
 i - invert 对动画前后数据信息的计算(translateY --> 100px,同时利用translate等操作,将dom恢复到 first位置)
 p - play 开始动画,并移除 i 步骤恢复至 first 的操作,启用tansition,动画就开始了

 整个过程其实就是,先记录好动画前后的dom位置等数据信息
 然后,利用css将dom恢复至初始位置
 最后,移除上一步恢复的状态(此时dom会自动回到last位置,只不过没有过渡效果,生硬的闪现),添加过渡效果,完成动画

完整代码:

const list = document.querySelector('.list')
  const item = document.querySelectorAll('.list_item')
  let  sourceNode; 

  list.ondragstart = e =>{
    sourceNode = e.target
    record(item)
    setTimeout(()=>{
      e.target.classList.add('moving')
    },0)
    e.dataTransfer.effectAllowed = 'move'
  }

  list.ondragover = e => {
    e.preventDefault()
  }


  list.ondragenter = e =>{
    e.preventDefault()
    if(e.target === list || e.target === sourceNode){
      return false
    }
    const children = Array.from(list.children)
    const sourceIndex = children.indexOf(sourceNode) 
    const targetIndex = children.indexOf(e.target) 
    if(sourceIndex < targetIndex){
      list.insertBefore(sourceNode,e.target.nextElementSibling)
    }else {
      list.insertBefore(sourceNode,e.target)
    }
    last([e.target,sourceNode])
  }

  list.ondragend = e =>{
    e.target.classList.remove('moving')
  }


function record(eleAll) {
  for( let i = 0;i < eleAll.length; i++ ) {
    const { top,left } = eleAll[i].getBoundingClientRect()
    eleAll[i]._top_ = top
    eleAll[i]._left_ = left
  }
}



function last(eleAll) {
  for( let i = 0;i < eleAll.length; i++ ) {
    const dom = eleAll[i]
    const { top,left } = dom.getBoundingClientRect()
    if(dom._left_) {
      dom.style.transform = `translate3d(${ dom._left_ - left }px, ${ dom._top_ - top }px,0px)`

      let rafId = requestAnimationFrame(function() {
        dom.style.transition = 'transform 300ms ease-out'
        dom.style.transform = 'none'
      })
      dom.addEventListener('transitionend', () => {
        dom.style.transition = 'none'
        cancelAnimationFrame(rafId)
      })
    }
  }
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

java计算机毕业设计ssm养老管理系统-敬老院系统

项目介绍 1.登录页面要有验证码 2.在健康信息模块中有三个小模块分别是饮食信息、身体信息、医疗常识,饮食信息就是护工每天负责老人的饮食搭配包括;早中晚餐;各种菜谱、图片、注意事项等,身体信息就是老人的各项身体指标,医疗常识就是每一种病对应一种病例还有每天推送更新的关…

互联网获客经验分享(一)

现在的流量都很贵了。特别是公域的流量。在这样的情况下,我们如何从互联网上获取客户?满足我业务的需求呢。 今天为大家分享基础的方法,后续我们会不定期的分享一些互联网软件获客的经验,希望能够帮助到你。 流量的本质 不管是平台方还是用户,流量都是生命线。在这样的…

好心情心理咨询平台:独处≠孤独,独处对心理健康有多重要?

说到独处&#xff0c;有人认为那不就是自己一个人呆着嘛。 非也非也&#xff0c;独处其实不是要求远离人群&#xff0c;而是在信息与情感上与他人无交流&#xff0c;做到无社会互动&#xff0c;与人在网上聊天、打电话可都不能算是独处。 相反&#xff0c;自己一个人在图书馆…

ClickHouse Senior Course Ⅳ

序言 梳理一下ClickHouse的数据库引擎 首先了解下什么是数据库引擎(cuiyaonan2000163.com) 数据库引擎是用于存储、处理和保护数据的核心服务。利用数据库引擎可控制访问权限并快速处理事务&#xff0c;从而满足企业内大多数需要处理大量数据的应用程序的要求。 使用数据库引…

[附源码]java毕业设计健身房管理系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

《无障碍环境认证实施方案》发布

市场监管总局 中国残联关于印发 《无障碍环境认证实施方案》的通知 国市监认证发〔2022〕94号各省、自治区、直辖市和新疆生产建设兵团市场监管局&#xff08;厅、委&#xff09;、残联&#xff0c;各有关单位&#xff1a;现将《无障碍环境认证实施方案》印发给你们&#xff0…

【附源码】Python计算机毕业设计团结里小区疫情防控系统

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

【Java|golang】775. 全局倒置与局部倒置

给你一个长度为 n 的整数数组 nums &#xff0c;表示由范围 [0, n - 1] 内所有整数组成的一个排列。 全局倒置 的数目等于满足下述条件不同下标对 (i, j) 的数目&#xff1a; 0 < i < j < n nums[i] > nums[j] 局部倒置 的数目等于满足下述条件的下标 i 的数目&a…

KEGG信号通路富集分析

KEGG信号通路富集分析 当我们手头有转录组测序后的差异分析数据后&#xff0c;想更进一步的去分析这下差异基因在信号通路富集情况&#xff0c;其中一种办法是使用KEGG网站进行信号通路富集分析。 1. 通过uniprot将Gene name转换为GeneID UniProt网站中的ID mapping具有强大…

关于HTTP/1.1,HTTP/2和HTTP/3的区别与联系

HTTP/1.1 每当说起http/1.1就会想起以前叫外卖的方式&#xff0c;那个时候很多店都没有专门的外卖员&#xff0c;打电话叫外卖&#xff0c;老板那就会叫人送货来&#xff0c;但是这样的方式有一个很大的问题&#xff0c;店员总是忘记放筷子。于是&#xff0c;店员送完外卖以后…

揭秘:如何用主题公园的思路做一款VR高尔夫游戏

作为一个从没玩过高尔夫球的人&#xff0c;我竟然第一次VR游戏中找到了高尔夫球的乐趣。由Mighty Coconut开发的《Walkabout Mini Golf》是这样一款VR游戏&#xff0c;它将迷你高尔夫与各种虚拟场景结合&#xff0c;高尔夫并不是游戏的全部&#xff0c;但你必须进球才能解锁一个…

【Python开发】Flask开发实战:个人博客(四)

Flask开发实战&#xff1a;个人博客&#xff08;四&#xff09;本篇博客将是《Flask开发实战&#xff1a;个人博客》的最后一篇。本篇文章将会详细介绍博客后台的编写。 为了支持管理员管理文章、分类、评论和链接&#xff0c;我们需要提供后台管理功能。通常来说&#xff0c;…

PDF转HTML转换器哪个好用?快看看这里

PDF格式是我们经常使用的文件格式&#xff0c;但是难免会需要将其转换成其他的格式。比如转为HTML&#xff0c;大家是不是对它很陌生呢。其实HTML是一种网页浏览格式&#xff0c;平时我们打开的浏览器页面就是用HTML代码设置而成的。那我来考考大家&#xff0c;你们知道如何将P…

redis缓存雪崩、缓存击穿和缓存穿透还傻傻分不清?看这篇就够了

前言 关于Redis的高频问题&#xff0c;缓存雪崩、缓存击穿和缓存穿透一定少不了&#xff0c;相信大家在面试中都被问到过类似的问题。为什么这些问题一直热门呢&#xff1f;因为我们在使用Redis缓存时&#xff0c;这些问题都是很容易遇到的。接下来我们就来看看这些问题都是怎…

【附源码】Python计算机毕业设计网上宠物店预约系统

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

Excel打开第二个文件很慢的解决方法

简介&#xff1a; 该问题具体表现为&#xff1a;打开第一个Excel文件后&#xff0c;在不关闭它的情况下接着打开第二个Excel文件&#xff0c;第二个Excel文件会延迟几秒之后才会正常打开。 注意前提是第一个Excel文件打开速度是正常的&#xff0c;否则本解决方案大概率对你无效…

Allegro分Partition操作指导

Allegro分Partition操作指导 Allegro支持多人协同操作,可以将版本分割成很多个部分,让多人操作,具体操作步骤如下 Change editor 选择Team Design,点击OK 点击place design Partition,选择Create Patitions 空白的地方右击选择 ADD Rectangle 选择需要分割的区域 除…

可观测性-可视化-Grafana热图Heatmap

文章目录可观测性-可视化-Grafana热图Heatmap前言选项可观测性-可视化-Grafana热图Heatmap 前言 Heatmap&#xff08;热图&#xff09;的用途&#xff0c;在Grafana官网是这样描述的&#xff1a;使用热图&#xff0c;将允许您查看随时间变化的直方图。 所以要使用热图的前提…

数据类型优化

schema与数据类型优化 数据类型优化原则 最小数据类型 尽量使用可以正确存储数据的最小数据类型&#xff0c;更小的数据类型通常更快&#xff0c;因为它们占用更少的磁盘、内存和CPU缓存。 更简单的数据类型 简单数据类型的操作通常需要更少的CPU周期 1.整型比字符操作代…

全球10大智慧港口介绍

智慧港口&#xff0c;包括数字化和利益相关者深切关注的港口&#xff0c;正在改变海运和航运业的未来。商业活动不断变化&#xff0c;人工智慧、物联网、大数据、区块链和 5G 等智慧技术正在推动全球港口创新。 为了举例说明是什么让港口变得「聪明」&#xff0c;我们决定寻找…