整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:

轻松实现HTML表格拖拽和排序,实现复杂交互式数据表变得更简单

我们都知道普通的HTML自带的功能相对有限,很多复杂的交互式场景,如果手动去写功能的话会非常的复杂,而且可扩展性差,就拿HTML表格来说,对于初学者或者对于复杂的拖拽式交互编程不熟悉的话会很浪费时间,因此今天就介绍一个第三方的插件——Table-draagger,来轻松实现类似的功能。Table-draagger是用于构建可重排序的拖放表的极简主义纯Javascript库!



Github

https://github.com/sindu12jun/table-dragger


特征

Table-draagger因为其以下几个特征而让拖拽和排序的实现变得如此简单:

  • 非常容易配置
  • 能够同时对列或行进行排序
  • 排序时动画流畅
  • 没有臃肿的依赖
  • 提供触摸事件(意味着在触摸设备可以实现一些你想要的功能)


安装使用

  • 安装

可以在npm上获得它:

npm install table-dragger --save

或者引用压缩的js文件

<script src="../node_modules/table-dragger/dist/table-dragger.min.js"></script>

或者尝试开发中的不稳定版本

npm install table-dragger@next --save

  • 快速入门

请看以下代码:

import tableDragger from 'table-dragger'
tableDragger(el, options?)

<table id="table">
<thead>
<tr>
<th class='handle'>header1</th>
<th class='handle'>header2</th>
</tr>
</thead>
<tbody>
<tr>
<td>conten1</td>
<td>conten2</td>
</tr>
</tbody>
</table>

var el = document.getElementById('table');
var dragger = tableDragger(el, {
mode: 'row',
dragHandler: '.handle',
onlyBody: true,
});
dragger.on('drop',function(from, to){
console(from);
console(to);
});

你可以在不设置任何参数的情况下使用默认的拖拽和排序方式,当然以下是你可以配置的选项:

  • options.mode

1、将mode设置为column,用户拖动和排序表的列

2、将mode设置为row,用户拖动并排序表的行

3、设置mode为free,用户根据点击后鼠标移动的方向拖动行或列。注意,必须在自由模式下指定dragHandler。

  • options.dragHandler

dragHandler是表中的拖动句柄选择器默认情况下,在列模式下,dragHandler是表的第一行;在行模式下,则是第一列。

  • options.onlyBody

在行模式下将onlyBody设置为true时,用户只能在tbody中提升行。

  • API

下面是返回对象的API


tableDragger(document.querySelector('#event-table'), { mode: 'free', dragHandler: '.handle', onlyBody: true })
.on('drag', () => {
console.log('drag');
})
.on('drop', (from, to, el, mode) => {
console.log(`drop ${el.nodeName} from ${from} ${mode} to ${to} ${mode}`);
})
.on('shadowMove', (from, to, el, mode) => {
console.log(`move ${el.nodeName} from ${from} ${mode} to ${to} ${mode}`);
})
.on('out', (el, mode) => {
console.log(`move out or drop ${el.nodeName} in mode ${mode}`);
});


总结

Table-draagger为我们节省了很多手动封装表格排序和拖拽功能的时间,当然目前很多第三方框架已经实现了类似的功能,这更适用于原生的html表格,你还可以通过一些手段记忆用户拖拽,这只是其中一种思路,Enjoy it!~

本文中,我们将了解HTML拖放,并借助示例了解其实现.

拖放是一个非常交互式和用户友好的概念,它可以通过抓住对象更轻松地将对象移动到其他位置。这允许用户在元素上单击并按住鼠标按钮,将其拖动到其他位置,然后释放鼠标按钮以将元素放到该位置。在HTML 5中,拖放更容易编码,其中的任何元素都是可拖动的。

拖放事件:有各种拖放事件,其中一些在下面列出:

  • ondrag: 它用于在 HTML 中拖动元素或文本选择时使用。
  • ondragstart: 它用于调用一个函数 drag(event),该函数指定要拖动的数据。
  • ondragenter: 它用于确定丢弃目标是否接受丢弃。如果要接受丢弃,则必须取消此事件。
  • ondragleave:当鼠标在拖动发生时将元素留在有效放置目标之前时,就会发生这种情况。
  • ondragover: 它指定可以放置拖动数据的位置。
  • ondrop:它指定在拖动操作结束时发生丢弃的位置。
  • ondragend: 当用户完成拖动元素时,就会发生这种情况。

拖放 Drag and Drop实现步骤:

  • 步骤1:使对象可拖动 首先将可拖动属性设置为 true,以使该元素可拖动 <img draggable = “true”>然后,指定拖动元素时应发生的情况。ondragstart 属性调用一个函数 drag(event),该函数指定要拖动的数据。dataTransfer.setData() 方法设置数据类型和拖动数据的值事件侦听器 ondragstart 将设置允许的效果(复制、移动、链接或某种组合).
  • 步骤2: 放下对象 ondragover 事件指定可以放下拖动数据的位置。默认情况下,不能将数据/元素放在其他元素中。若要允许删除,它必须阻止元素的默认处理。这是通过调用 event.preventDefault() 方法最后,drop 事件来完成的,它允许执行实际的丢弃

例1:

<!DOCTYPE HTML>
<html>
<head>
    <style>
    #getData {
        width: 250px;
        height: 200px;
        padding: 10px;
        border: 1px solid #4f4d4d;
    }
    </style>
    <script>
    function allowDrop(even) {
        even.preventDefault();
    }
 
    function drag(even) {
        even.dataTransfer.setData("text", even.target.id);
    }
 
    function drop(even) {
        even.preventDefault();
        var fetchData = even.dataTransfer.getData("text");
        even.target.appendChild(document.getElementById(fetchData));
    }
    </script>
</head>
 
<body>
    <h3>Drag the GeekforGeeks image into the rectangle:</h3>
    <div id="getData"
         ondrop="drop(event)"
         ondragover="allowDrop(event)">
      </div>
    <br>
    <img id="dragData"
         src="gfg.png"
         draggable="true"
         ondragstart="drag(event)"
         width="250"
         height="200">
</body>
</html>


Output:

Dragging the image into the box

拖放过程的数据传递: 当拖放的整个过程发生时,将使用数据传输属性。它用于保存从源拖放到所需位置的数据。这些是与之关联的属性:

  • dataTransfer.setData(format, data): 它用于设置要拖动的数据.
  • dataTransfer.clearData(format): 它用于调用此函数,没有清除所有数据的参数。使用 format 参数调用它仅删除该特定数据。
  • dataTransfer.getData(format): 它返回指定格式的数据。如果没有此类数据,则返回空字符串.
  • dataTransfer.types:此属性返回在 dragstart 事件中设置的所有格式的数组。
  • dataTransfer.files: 它用于返回要删除的所有文件。
  • dataTransfer.setDragImage(element, x, y): 它用于将现有图像显示为拖动图像,而不是光标旁边的默认图像。坐标指定放置位置。
  • dataTransfer.addElement(element): 它用于添加要绘制的指定元素作为拖动反馈图像。
  • dataTransfer.effectAllowed(value): 它将告诉浏览器只允许用户使用列出的操作类型。该属性可以设置为以下值:none、copy、copyLink、copyMove、link、linkMove、move、all 和未初始化。
  • dataTransfer.dropEffect(value): 它用于控制在拖动和拖动事件期间向用户提供的反馈。当用户将鼠标悬停在目标元素上时,浏览器的光标将指示将要发生的操作类型(例如复制、移动等)。效果可以采用以下值之一:无、复制、链接、移动。

例2:

<!DOCTYPE HTML>
<html>
<head>
<title>Drag and Drop box</title>
<script>
function allowDrop(ev) {
ev.preventDefault();
}
function dragStart(ev) {
ev.dataTransfer.setData("text", ev.target.id);
}
function dragDrop(ev) {
ev.preventDefault();
var data = ev.dataTransfer.getData("text");
ev.target.appendChild(document.getElementById(data));
}
</script>
<style>
#box {
margin: auto;
width: 50%;
height: 200px;
border: 3px solid green;
padding: 10px;
}
#box1,
#box2,
#box3 {
float: left;
margin: 5px;
padding: 10px;
}
#box1 {
width: 50px;
height: 50px;
background-color: #F5B5C5;
}
#box2 {
width: 100px;
height: 100px;
background-color: #B5D5F5;
}
#box3 {
width: 150px;
height: 150px;
background-color: #BEA7CC;
}
p {
font-size: 20px;
font-weight: bold;
text-align: center;
}
.gfg {
font-size: 40px;
color: #009900;
font-weight: bold;
text-align: center;
}
</style>
</head>
<body>
<div class="gfg">GeeksforGeeks</div>
<p>Drag and drop of boxes</p>
<div id="box">
<div id="box1" draggable="true"
ondragstart="dragStart(event)">
</div>
<div id="box2" draggable="true"
ondragstart="dragStart(event)">
</div>
<div id="box3" ondrop="dragDrop(event)"
ondragover="allowDrop(event)">
</div>
</div>
</body>
</html>


说明:

  • 通过将要拖动的元素的可拖动属性draggable设置为 true 来开始拖动。
  • 使用 dataTransfer.getData() 方法获取拖动的数据。此方法将返回在 setData() 方法中设置为相同类型的任何数据。
  • 调用 event.preventDefault() 方法,通过阻止元素的默认处理来允许删除其他元素中的元素。
  • 该元素存储在我们附加到放置元素中的变量数据中。

效果:

简介】

拖放是一种常见的特性, 属于html5标准的一部分, 即抓取对象以后拖动到另一个位置, 在html5中, 任何元素都可被设置拖放。首先, 我们要给需要拖动的HTML元素启用拖动功能, 设置属性draggable="true",

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

提示:a标签和img标签默认是启用该属性的, 可不需要设置draggable属性。

draggable有三个值, 如下所示:

draggable = true(元素可以被拖动)

draggable = false(元素不能被拖动)

draggable = auto(浏览器可以自主决定某个元素是否可以被拖动)

【用法】

当我们用鼠标拖拽目标元素过程中会触发的事件:

ondragstart:用户按下鼠标开始拖动时触发

ondrag:用户正在拖动时反复触发

ondragend:用户结束拖动后触发

<img id="imgs" ondragstart="startFun()" ondrag="ondragFun()" ondragend="ondragendFun()" src="../img/a.png"/>

当拖动元素进入目标容器内触发的事件:

ondragenter:鼠标拖动对象进入释放区时触发

ondragover:被拖动物体进入目标容器内移动时反复触发

ondragleave:拖动对象在释放区没有释放就离开容器时触发

ondrop:被拖动物体在目标容器内释放时触发

<div id="container" ondragenter="ondragenterFun(event)" ondragover="ondragoverFun(event)" ondragleave="ondragleaveFun()" ondrop="drop()"></div>

ondragenter和ondragover事件的默认行为是拒绝接受任何被拖放的项目, 所以我们必须要做的最重要的事情就是防止这种默认行为的发生。

因此, 我们只需要在这两个事件调用的函数中传入event对象, 使用event.preventDefault()就可取消这种默认行为;举个例子, 在drop事件时, Firefox浏览器会关闭网页, 转而显示被拖动图片img元素src所引用的地址。

取消元素默认行为:

function ondragenterFun(e){
    e.preventDefault();
}
function ondragoverFun(e){
    e.preventDefault();
}

在event对象中, 我们会使用dataTransfer属性来获取DataTransfer对象, 在DataTransfer对象中有我们操作数据的属性和方法, 具体如下:

datatransfer:转移释放元素的数据到释放区, 返回Datatransfer对象

event.dataTransfer //返回DataTransfer对象

DataTransfer对象的属性:

files:处理从操作系统拖动并释放到释放区的文件;

types:返回一个字符串数组, 该对象包含了dataTransfer对象中数据的所有类型;

items:返回DataTransferItems对象, 该对象代表了拖动数据;

dropEffect:设置拖放目标允许发生的拖放行为, 如果此处设置的拖放行为不在effectAllowed属性设置的可拖放行为内, 拖放操作将会失败。该属性值只允许为"null"、"copy"、"link"或"move";

effectAllowed:设置拖动元素允许发生的拖动行为, 该属性值可为"none"、"copy"、"copyLink"、"copyMove"、"link"、"linkMove"、"move"、"all"或"uninitialized";

DataTransfer对象的方法:

setData( format , data ):将指定格式的数据赋值给dataTransfer对象,参数format定义数据的格式也就是数据的类型,data为待赋值的数据。

getData( format ):从dataTransfer对象中获取指定格式的数据,format代表数据格式,data为数据。

clearData( [format] ):从dataTransfer对象中删除指定格式的数据,参数可选,若不给参数,将删除对象中所有的数据。

setDragImage(el, x, y):设置拖放操作的图标,其中el代表自定义图标,x代表图标与鼠标在水平方向上的距离,y代表图标与鼠标在垂直方向上的距离。


了解了H5拖动使用的api以后我们接下来看一个综合的案例, 功能如下:

1)、实现图片拖动功能;

2)、实现图片复制功能;

3)、过滤不能拖动的元素;

4)、实现拖动本地图片到浏览器指定位置;

公共css部分:

<style>
#dropIn{
border:1px solid #AAAAAA;
height:100px;
margin-bottom: 10px;
padding: 10px;
}
#dropIn>img{
margin-right: 10px;
border:2px solid deepskyblue;
}
img{
width:100px;
border-radius: 10px;
border:2px solid red;
}
</style>

html部分:

<body>
<div id="dropIn"></div> <!--释放区-->
<img id="drop1" src="img/a.png" alt="" />
<!--拖动的图片元素-->
</body>

js部分:

<script type="text/javascript">
var darggID;
function getId(el){
    return document.getElementById(el)
}
var dropId1 = getId("drop1");
var dropInId = getId("dropIn");
//取消事件默认行为
dropInId.ondragenter = cancelDefault;
dropInId.ondragover = cancelDefault; //绑定拖动元素释放时触发的事件
dropInId.ondrop = drop; //绑定
dropId1.ondragstart = startFun;
function cancelDefault(ev){ //取消默认行为
    ev.preventDefault();
}
function startFun(ev){
    darggID = ev.target.id;
    //获取被拖动元素的id
    //从源对象上的事件处理中保存数据,数据类型为"Text"
    ev.dataTransfer.setData("Text",darggID);
}
function drop(ev){
    ev.preventDefault();
    // 从目标对象上的事件处理中读取"Text"类型数据
    var data=ev.dataTransfer.getData("Text");
    // 插入到目标对象中
    ev.target.appendChild(document.getElementById(data));
}
</script>

接下来我们添加两张图, "drop2"是实现复制的图片, "drop3"是既不能复制也不能拖动的图片;

<img id="drop2" src="img/b.png" alt="" /><img id="drop3" src="img/c.png" alt="" />

添加js代码:

//获取页面元素
var dropId2 = getId("drop2");
var dropId3 = getId("drop3");
//绑定事件
dropId2.ondragstart = startFun;dropId3.ondragstart = startFun;
//修改drop函数为
function drop(ev){
    ev.preventDefault();
    // 从目标对象上的事件处理中读取"Text"类型数据
    var data=ev.dataTransfer.getData("Text");
    if(data=='drop1'){
    //移动
        ev.target.appendChild(document.getElementById(data));
    }
    if(data=='drop2'){//复制
        var nreEl=document.getElementById(darggID).cloneNode(false);
        getId("dropIn").appendChild(nreEl);
    }
    if(data=='drop3'){//过滤drop3,drop3不做任何操作
        alert('过滤drop3')
    }
}

接下来我们实现拖动本地图片到浏览器, 我们就将图片拖动到id为"dropIn"的这个div中;添加js:

/*document 监听drop 并阻止浏览器打开客户端的图片*/
document.ondragover = function (e) {
    //只有在ondragover中阻止默认行为
    e.preventDefault();
};
document.ondrop = function (e) {
    //阻止 document.ondrop的默认行为
    e.preventDefault();
};
//dropIn是div的id
dropIn.ondrop = function (e) {
    var list = e.dataTransfer.files;
    for (var i = 0; i < list.length; i++) {
    var f = list[i];
        reader(f);
    }
};
function reader(f) {
    var reader = new FileReader();
    //读取数据
    reader.readAsDataURL(f);
    reader.onload = function () {
    var img = new Image();
    img.src = reader.result;
    dropIn.appendChild(img);
    }
}

【浏览器支持】

目前只有Internet Explorer 9、Firefox、Opera 12、Chrome 以及 Safari5支持拖放,在 Safari5.1.2 中不支持拖放。

最后再和大家分享一个技巧, 这种拖动行为还能跨浏览器工作, 这里说的跨浏览器不是浏览器之间的跨窗口, 而是可以从Chrome浏览器拖动到Firefox浏览器, 因为拖放功能的支持是集成在操作系统里面的, 有着相同的特性。