美文网首页让前端飞Web前端之路
实现一个美化原生拖拽的draggable-polyfill

实现一个美化原生拖拽的draggable-polyfill

作者: alanwhy | 来源:发表于2020-05-06 16:47 被阅读0次

    拖拽的实现

    在HTML5还未普及之前,实现元素的拖拽还算是一件比较麻烦的事,大概思路就是监听鼠标移动相关事件,下面是伪代码

    oDiv.onmousedown = function(ev){
        //记录起始位置
    }
    document.onmousemove = function(ev){
        //移动目标元素
    }
    document.onmouseup = function(ev){
        //取消鼠标移动事件
    }
    

    HTML5新增了拖放draggable标准,拖拽就变得简单了,只需要通过监听元素的拖放事件就能实现各种拖放功能。

    <div draggable="true">drag me</div>
    

    除了设置draggable="true"属性外,默认情况下img、链接默认是可拖拽的

    当然,设置draggable="true"元素仅仅是“可拖拽”,松手后就还原,如果需要拖拽到指定位置仅需要在drop记录一下就行了

    dropbox.addEventListener('drop',function(ev){
        dragbox.style.left = XX;
        dragbox.style.top = XX;
    })
    

    自定义原生的拖拽

    实现思路

    • 去除默认的预览图
    • 复制一份当前目标元素,cloneObj
    • 监听拖拽事件,改变cloneObj的位置
    • 拖拽结束移除cloneObj
    1.去除默认的预览图

    虽然setDragImage比较鸡肋,但是我们可以设置一张透明的图片就可以实现去除默认的预览图的效果了

    dragbox.addEventListener('dragstart', function (ev) {
        var img = new Image();
        img.src = "data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' %3E%3Cpath /%3E%3C/svg%3E";
        ev.dataTransfer.setDragImage(img, 0, 0);
    }
    

    普通元素设置draggable="true"后,Chrome可以直接拖拽,FireFox需要在dragstart设置ev.dataTransfer.setData('text','任意值')才行

    2.复制一份当前目标元素,cloneObj

    复制一份当前目标元素,并且设置position:fixed等属性,可以悬浮在页面之上,然后添加到body,如下

    var cloneObj = this.cloneNode(true);
    cloneObj.style = 'position:fixed;left:0;top:0;z-index:999;pointer-events:none;transform:translate3d( ' + left + 'px ,' + top + 'px,0);'
    document.body.appendChild(cloneObj);
    
    3.监听拖拽事件,改变cloneObj的位置

    拖拽元素时会触发drag事件,和鼠标移动事件类似,可以取到当前鼠标位置(在拖拽过程中并不触发mousemove事件),同时也能隐藏原目标

    dragbox.addEventListener('drag', function (ev) {
        if(cloneObj){
            cloneObj.style.transform = 'translate3d( ' + left + 'px ,' + top + 'px,0)';
            dragbox.style.visibility = 'visible';
        }
    })
    

    Chrome确实是这样,FireFox虽然也能触发drag事件,然后里面却取不到鼠标位置信息(均为0),所以,我们只能把监听放在dragover上,虽然不太完美,也是一个方法

    document.addEventListener('dragover', function (ev) {
        if(cloneObj){
            cloneObj.style.transform = 'translate3d( ' + left + 'px ,' + top + 'px,0)';    
        }
    })
    
    4.拖拽结束移除cloneObj

    拖拽结束移除cloneObj,并且还原原目标

    oDiv.addEventListener('dragend', function (ev) {
        document.body.removeChild(cloneObj);
        cloneObj = null;
        dragbox.style.visibility = 'visible';
    })
    

    效果图

    效果.gif

    原作者使用方式提供

    1.直接引用

    <script src="./lib/draggable-polyfill.js"></script>
    

    2.工程项目

    npm i draggable-polyfill
    
    import draggable-polyfill;
    

    需要注意的一点是:由于只是复制了当前节点,所以如果你的样式依赖于父级,那么复制出来的样式就会和原目标不一样

    .parent .dragbox{
        background:red
    }
    /*改为*/
    .dragbox{
        background:red
    }
    

    原文链接:# 实现一个美化原生拖拽的draggable-polyfill
    项目地址:https://github.com/XboxYan/draggable-polyfill
    更多示例:http://xboxyan.codelabo.cn/draggable-polyfill/example/index.html
    回收箱案例:https://www.zhangxinxu.com/study/201102/html5-drag-and-drop.html

    相关文章

      网友评论

        本文标题:实现一个美化原生拖拽的draggable-polyfill

        本文链接:https://www.haomeiwen.com/subject/fuvwghtx.html