当前位置:   article > 正文

原生拖拽,拖放事件(drag and drop)

js 通过拖放文件:可以使用`drag`和`drop`事件来实现文件的拖放操作,并获取文件路

原生拖拽,拖放事件(drag and drop)

拖拽,拖放事件可以通过拖拽实现数据传递,达到良好的交互效果,如:从操作系统拖拽文件实现文件选择,拖拽实现元素布局的修改.

drag and drop事件流程

一个完整的drag and drop流程通常包含以下几个步骤:

  1. 设置可拖拽目标.设置属性draggable="true"实现元素的可拖拽.
  2. 监听dragstart设置拖拽数据
  3. 为拖拽操作设置反馈图标(可选)
  4. 设置允许的拖放效果,如copy,move,link
  5. 设置拖放目标,默认情况下浏览器阻止所有的拖放操作,所以需要监听dragenter或者dragover取消浏览器默认行为使元素可拖放.
  6. 监听drop事件执行所需操作

拖拽事件

以下是拖拽产生的一系列事件,拖拽事件产生时不会产生对应的鼠标事件.

  • dragstart:拖拽开始时在被拖拽元素上触发此事件,监听器需要设置拖拽所需数据,从操作系统拖拽文件到浏览器时不触发此事件.
  • dragenter:拖拽鼠标进入元素时在该元素上触发,用于给拖放元素设置视觉反馈,如高亮
  • dragover:拖拽时鼠标在目标元素上移动时触发.监听器通过阻止浏览器默认行为设置元素为可拖放元素.
  • dragleave:拖拽时鼠标移出目标元素时在目标元素上触发.此时监听器可以取消掉前面设置的视觉效果.
  • drag:拖拽期间在被拖拽元素上连续触发
  • drop:鼠标在拖放目标上释放时,在拖放目标上触发.此时监听器需要收集数据并且执行所需操作.如果是从操作系统拖放文件到浏览器,需要取消浏览器默认行为.
  • dragend:鼠标在拖放目标上释放时,在拖拽元素上触发.将元素从浏览器拖放到操作系统时不会触发此事件.

DataTransfer对象

拖拽事件周期中会初始化一个DataTransfer对象,用于保存拖拽数据和交互信息.以下是它的属性和方法.

  • dropEffect: 拖拽交互类型,通常决定浏览器如何显示鼠标光标并控制拖放操作.常见的取值有copy,move,linknone
  • effectAllowed: 指定允许的交互类型,可以取值:copy,move,link,copyLink,copyMove,limkMove, all, none默认为uninitialized(允许所有操作)
  • files: 包含File对象的FileList对象.从操作系统向浏览器拖放文件时有用.
  • types: 保存DataTransfer对象中设置的所有数据类型.
  • setData(format, data): 以键值对设置数据,format通常为数据格式,如text,text/html
  • getData(format): 获取设置的对应格式数据,format与setData()中一致
  • clearData(format): 清除指定格式的数据
  • setDragImage(imgElement, x, y): 设置自定义图标

dataTransfer对象在传递给监听器的事件对象中可以访问,如下:

  1. draggableElement.addEventListener('dragstart', function (event) {
  2. event.dataTransfer.setData('text', 'Hello World');
  3. }, false);

推荐的拖拽元素和数据类型

详细参考MDN recommended drag type

文本

在页面中选择文本并拖拽,无需处理dragstart设置数据,浏览器自动设置选取的文本.相当于event.dataTransfer.setData("text/plain", "this is text to drag").只需要在拖放目标上读取对应格式的数据即可.

链接

实际案例

前面介绍了最基本的理论知识,下面进行实际操作

元素拖拽

目标: 拖拽元素到达目的区域,改变在DOM中的位置,同时设置反馈视觉效果在线demo

  1. <div id="demo1">
  2. <ul class="panel-list">
  3. <li class="panel-item"></li>
  4. <li class="panel-item"></li>
  5. <li class="panel-item"></li>
  6. <li class="panel-item"></li>
  7. <li class="panel-item"></li>
  8. </ul>
  9. <h2>拖拽下面的方块到上面任意容器中</h2>
  10. <!-- 设置draggable使元素成为可拖拽元素 -->
  11. <span class="movable" id="demo1-src" draggable="true"></span>
  12. <style>
  13. #demo1 {
  14. margin: 20px;
  15. }
  16. #demo1 .panel-list {
  17. overflow: hidden;
  18. list-style: none;
  19. margin: 0;
  20. padding: 0;
  21. }
  22. #demo1 .panel-item {
  23. float: left;
  24. margin-right: 30px;
  25. width: 100px;
  26. height: 100px;
  27. background: #ddd;
  28. border: 1px solid #ddd;
  29. }
  30. #demo1-src {
  31. display: inline-block;
  32. width: 50px;
  33. height: 50px;
  34. background: purple;
  35. }
  36. #demo1 .over {
  37. border: 1px dashed #000;
  38. -webkit-transform: scale(0.8, 0.8);
  39. }
  40. </style>
  41. <script>
  42. (function () {
  43. var dnd = {
  44. // 初始化
  45. init: function () {
  46. var me = this;
  47. me.src = document.querySelector('#demo1-src');
  48. me.panelList = document.querySelector('.panel-list');
  49. // 为拖拽源监听dragstart,设置关联数据
  50. me.src.addEventListener('dragstart', me.onDragStart, false);
  51. // 拖拽鼠标移入元素,在拖放目标上设置视觉反馈
  52. me.panelList.addEventListener('dragenter', me.onDragEnter, false);
  53. // 取消元素dragover默认行为,使其可拖放
  54. me.panelList.addEventListener('dragover', me.onDragOver, false);
  55. // 拖拽移出元素,清除视觉反馈
  56. me.panelList.addEventListener('dragleave', me.onDragLeave, false);
  57. // 鼠标释放,在拖放目标上接收数据并处理
  58. me.panelList.addEventListener('drop', me.onDrop, false);
  59. },
  60. onDragStart: function (e) {
  61. e.dataTransfer.setData('text/plain', 'demo1-src');
  62. },
  63. onDragEnter: function (e) {
  64. if (e.target.classList.contains('panel-item')) {
  65. e.target.classList.add('over');
  66. }
  67. },
  68. onDragLeave: function (e) {
  69. if (e.target.classList.contains('panel-item')) {
  70. e.target.classList.remove('over');
  71. }
  72. },
  73. onDragOver: function (e) {
  74. e.preventDefault();
  75. },
  76. onDrop: function (e) {
  77. var id = e.dataTransfer.getData('text/plain');
  78. var src = document.getElementById(id);
  79. var target = e.target;
  80. if (target.classList.contains('panel-item')) {
  81. target.appendChild(src);
  82. target.classList.remove('over');
  83. }
  84. }
  85. };
  86. dnd.init();
  87. }());
  88. </script>
  89. </div>

从操作系统拖拽图片到指定区域进行预览

从操作系统拖拽文件到浏览器中.不会触发dragstart,dragend,只需取消拖放区域的默认行为,设置反馈,并在拖放发生时取消浏览器默认行为,通过e.dataTransfer.files获取文件信息进行操作.在线demo

  1. <div id="demo2">
  2. <h3>从文件夹中拖拽图片到下面的区域进行预览</h3>
  3. <ul class="preview"></ul>
  4. <style>
  5. #demo2 {
  6. margin: 20px;
  7. }
  8. #demo2 .preview {
  9. height: 300px;
  10. background: #ddd;
  11. }
  12. #demo2 li {
  13. float: left;
  14. margin-left: 40px;
  15. }
  16. #demo2 img {
  17. max-height: 150px;
  18. width: auto;
  19. }
  20. </style>
  21. <script>
  22. (function (w) {
  23. var doc = w.document;
  24. var dnd = {
  25. init: function () {
  26. var me = this;
  27. var preview = doc.querySelector('#demo2 .preview');
  28. preview.addEventListener('dragover', function (e) {
  29. e.preventDefault();
  30. }, false);
  31. preview.addEventListener('drop', function (e) {
  32. // 操作系统拖放文件到浏览器需要取消默认行为
  33. e.preventDefault();
  34. [].forEach.call(e.dataTransfer.files, function (file) {
  35. if (file && file.type.match('image.*')) {
  36. var reader = new FileReader();
  37. reader.onload = function (e) {
  38. var img = doc.createElement('img');
  39. img.src = e.target.result;
  40. var li = doc.createElement('li');
  41. li.appendChild(img);
  42. preview.appendChild(li);
  43. };
  44. reader.readAsDataURL(file);
  45. }
  46. });
  47. }, false);
  48. }
  49. };
  50. dnd.init();
  51. }(window));
  52. </script>
  53. </div> <!-- demo2 -->

参考资料

本文内容由网友自发贡献,转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号