js 实现拖拽排序详情

作者:名字起太长会有傻子跟着念 时间:2024-06-07 15:24:23 

1、前言

拖拽排序对于小伙伴们来说应该不陌生,平时工作的时候,可能会选择使用类似Sortable.js这样的开源库来实现需求。但在完成需求后,大家有没有没想过拖拽排序是如何实现的呢?我花了点时间研究了一下,今天分享给大家。

2、实现


{
   margin: 0;
   padding: 0;
   box-sizing: border-box;
}

.grid {
   display: flex;
   flex-wrap: wrap;
   margin: 0 -15px -15px 0;
   touch-action: none;
   user-select: none;
}

.grid-item {
   width: 90px;
   height: 90px;
   line-height: 88px;
   text-align: center;
   margin: 0 15px 15px 0;
   background: #FFF;
   border: 1px solid #d6d6d6;
   list-style: none;
}

.active {
   background: #c8ebfb;
}

.clone-grid-item {
   position: fixed;
   left: 0;
   top: 0;
   z-index: 1;
   width: 90px;
   height: 90px;
   line-height: 88px;
   text-align: center;
   background: #FFF;
   border: 1px solid #d6d6d6;
   opacity: 0.8;
   list-style: none;
}


<ul class="grid">
   <li class="grid-item">item1</li>
   <li class="grid-item">item2</li>
   <li class="grid-item">item3</li>
   <li class="grid-item">item4</li>
   <li class="grid-item">item5</li>
   <li class="grid-item">item6</li>
   <li class="grid-item">item7</li>
   <li class="grid-item">item8</li>
   <li class="grid-item">item9</li>
   <li class="grid-item">item10</li>
</ul>

采用ES6 Class写法:


class Draggable {
   constructor(options) {
       this.parent = options.element; // 父级元素
       this.cloneElementClassName = options.cloneElementClassName; // 克隆元素类名
       this.isPointerdown = false;
       this.diff = { x: 0, y: 0 }; // 相对于上一次移动差值
       this.drag = { element: null, index: 0, lastIndex: 0 }; // 拖拽元素
       this.drop = { element: null, index: 0, lastIndex: 0 }; // 释放元素
       this.clone = { element: null, x: 0, y: 0 };
       this.lastPointermove = { x: 0, y: 0 };
       this.rectList = []; // 用于保存拖拽项getBoundingClientRect()方法获得的数据
       this.init();
   }
   init() {
       this.getRect();
       this.bindEventListener();
   }
   // 获取元素位置信息
   getRect() {
       this.rectList.length = 0;
       for (const item of this.parent.children) {
           this.rectList.push(item.getBoundingClientRect());
       }
   }
   handlePointerdown(e) {
       // 如果是鼠标点击,只响应左键
       if (e.pointerType === 'mouse' && e.button !== 0) {
           return;
       }
       if (e.target === this.parent) {
           return;
       }
       this.isPointerdown = true;
       this.parent.setPointerCapture(e.pointerId);
       this.lastPointermove.x = e.clientX;
       this.lastPointermove.y = e.clientY;
       this.drag.element = e.target;
       this.drag.element.classList.add('active');
       this.clone.element = this.drag.element.cloneNode(true);
       this.clone.element.className = this.cloneElementClassName;
       this.clone.element.style.transition = 'none';
       const i = [].indexOf.call(this.parent.children, this.drag.element);
       this.clone.x = this.rectList[i].left;
       this.clone.y = this.rectList[i].top;
       this.drag.index = i;
       this.drag.lastIndex = i;
       this.clone.element.style.transform = 'translate3d(' + this.clone.x + 'px, ' + this.clone.y + 'px, 0)';
       document.body.appendChild(this.clone.element);
   }
   handlePointermove(e) {
       if (this.isPointerdown) {
           this.diff.x = e.clientX - this.lastPointermove.x;
           this.diff.y = e.clientY - this.lastPointermove.y;
           this.lastPointermove.x = e.clientX;
           this.lastPointermove.y = e.clientY;
           this.clone.x += this.diff.x;
           this.clone.y += this.diff.y;
           this.clone.element.style.transform = 'translate3d(' + this.clone.x + 'px, ' + this.clone.y + 'px, 0)';
           for (let i = 0; i < this.rectList.length; i++) {
               // 碰撞检测
               if (e.clientX > this.rectList[i].left && e.clientX < this.rectList[i].right &&
                   e.clientY > this.rectList[i].top && e.clientY < this.rectList[i].bottom) {
                   this.drop.element = this.parent.children[i];
                   this.drop.lastIndex = i;
                   if (this.drag.element !== this.drop.element) {
                       if (this.drag.index < i) {
                           this.parent.insertBefore(this.drag.element, this.drop.element.nextElementSibling);
                           this.drop.index = i - 1;
                       } else {
                           this.parent.insertBefore(this.drag.element, this.drop.element);
                           this.drop.index = i + 1;
                       }
                       this.drag.index = i;
                       const dragRect = this.rectList[this.drag.index];
                       const lastDragRect = this.rectList[this.drag.lastIndex];
                       const dropRect = this.rectList[this.drop.index];
                       const lastDropRect = this.rectList[this.drop.lastIndex];
                       this.drag.lastIndex = i;
                       this.drag.element.style.transition = 'none';
                       this.drop.element.style.transition = 'none';
                       this.drag.element.style.transform = 'translate3d(' + (lastDragRect.left - dragRect.left) + 'px, ' + (lastDragRect.top - dragRect.top) + 'px, 0)';
                       this.drop.element.style.transform = 'translate3d(' + (lastDropRect.left - dropRect.left) + 'px, ' + (lastDropRect.top - dropRect.top) + 'px, 0)';
                       this.drag.element.offsetLeft; // 触发重绘
                       this.drag.element.style.transition = 'transform 150ms';
                       this.drop.element.style.transition = 'transform 150ms';
                       this.drag.element.style.transform = 'translate3d(0px, 0px, 0px)';
                       this.drop.element.style.transform = 'translate3d(0px, 0px, 0px)';
                   }
                   break;
               }
           }
       }
   }
   handlePointerup(e) {
       if (this.isPointerdown) {
           this.isPointerdown = false;
           this.drag.element.classList.remove('active');
           this.clone.element.remove();
       }
   }
   handlePointercancel(e) {
       if (this.isPointerdown) {
           this.isPointerdown = false;
           this.drag.element.classList.remove('active');
           this.clone.element.remove();
       }
   }
   bindEventListener() {
       this.handlePointerdown = this.handlePointerdown.bind(this);
       this.handlePointermove = this.handlePointermove.bind(this);
       this.handlePointerup = this.handlePointerup.bind(this);
       this.handlePointercancel = this.handlePointercancel.bind(this);
       this.getRect = this.getRect.bind(this);
       this.parent.addEventListener('pointerdown', this.handlePointerdown);
       this.parent.addEventListener('pointermove', this.handlePointermove);
       this.parent.addEventListener('pointerup', this.handlePointerup);
       this.parent.addEventListener('pointercancel', this.handlePointercancel);
       window.addEventListener('scroll', this.getRect);
       window.addEventListener('resize', this.getRect);
       window.addEventListener('orientationchange', this.getRect);
   }
   unbindEventListener() {
       this.parent.removeEventListener('pointerdown', this.handlePointerdown);
       this.parent.removeEventListener('pointermove', this.handlePointermove);
       this.parent.removeEventListener('pointerup', this.handlePointerup);
       this.parent.removeEventListener('pointercancel', this.handlePointercancel);
       window.removeEventListener('scroll', this.getRect);
       window.removeEventListener('resize', this.getRect);
       window.removeEventListener('orientationchange', this.getRect);
   }
}
// 实例化
new Draggable({
   element: document.querySelector('.grid'),
   cloneElementClassName: 'clone-grid-item'
});

Demo:jsdemo.codeman.top/html/dragga…

3、为何不使用HTML拖放API实现?

因为原生HTML拖放API在移动端无法使用,所以为了兼容PC端和移动端,使用了PointerEvent事件实现拖拽逻辑。

4、总结

拖拽排序的基本功能已经实现,但还存在很多不足。像嵌套拖拽,跨列表拖拽,拖拽到底部自动滚动等功能都未实现。

来源:https://juejin.cn/post/7022824391163510821

标签:js,拖拽,排序
0
投稿

猜你喜欢

  • asp.net FindControl方法误区和解析

    2024-06-05 09:28:08
  • 编写和优化SQL Server的存储过程

    2009-04-13 10:13:00
  • 分析python并发网络通信模型

    2023-12-15 11:13:59
  • 微信小程序picker组件简单用法示例

    2023-07-23 10:49:32
  • Python-Tkinter Text输入内容在界面显示的实例

    2023-03-21 13:50:58
  • python实现nao机器人身体躯干和腿部动作操作

    2021-07-02 07:39:47
  • ubuntu 16.04下python版本切换的方法

    2021-07-10 16:36:04
  • 对比分析BN和dropout在预测和训练时区别

    2022-09-05 11:46:55
  • python Manager 之dict KeyError问题的解决

    2022-12-17 07:38:09
  • asp 简单分页代码

    2011-03-11 10:53:00
  • Go 结构体、数组、字典和 json 字符串的相互转换方法

    2024-05-05 09:26:42
  • 教你用Python写一个植物大战僵尸小游戏

    2021-07-19 22:59:37
  • Python制作CSDN免积分下载器

    2021-12-25 03:46:35
  • Pytorch深度学习经典卷积神经网络resnet模块训练

    2022-12-02 01:43:23
  • python同时替换多个字符串方法示例

    2021-11-25 00:37:54
  • 在Python编程过程中用单元测试法调试代码的介绍

    2023-12-10 02:16:46
  • PyCharm更改字体和界面样式的方法步骤

    2021-12-24 09:15:25
  • DW中如何使用Library

    2007-02-03 11:39:00
  • WEB设计经验-来自Microsoft

    2008-05-15 07:30:00
  • python 多线程中join()的作用

    2022-11-27 12:24:24
  • asp之家 网络编程 m.aspxhome.com