如何在web自动化测试中模拟drag and drop动作

起因

有个小组在做一个页面构建的web站点,用来快速创建结构化的网站页面。该站点支持鼠标拖拽已定义好的各种页面插件,来拼凑成一个完整的页面。这些创建好的页面可以直接在公司的站点上使用,提高前端页面构建效率。
对此web站点,也是要做自动化进行验证,那么拖拽功能就是首先遇到的一个问题。

经过

思路1

selenium webdriver 中不就有拖拽命令,直接用不就行了。我们用的是webdriverio 来执行浏览器操作,首先就找到了这个链接 webdriverio dragAndDrop ,各种尝试都不行。

思路2

搜了一下internet看看网上是否有,发现这可能是一个selenium未解决的问题 selenium issue, chromedriver issue
网上也有很多人遇到同样的问题,最早从2015年开始一直到最近都陆续有人在selenium issue里面提问题,但是官方一直没有解决???这也是很神奇

思路3

网上搜索还是找到很多不同的解决方案,于是先找一些现成的代码试试看,阅尽千帆后发现没有一个能直接用上的,但是还是获取了一些信息,对后续的解决方案提供了帮助。同时也找到另一个有意思的端到端测试框架cypress.js,可以直接利用它来实现页面元素的拖拽测试,但是涉及到要换测试框架的问题,所以最终还是没有采用这个新框架,但该框架提供了一个用JS来解决拖拽问题的方向。

思路4

还是来分析一下测试目标的一些细节,可以从页面源码看到该站点的可拖拽元素是基于HTML drag and drop API 来做的,该元素增加了一个draggable 属性,就可以执行拖拽行为了。同时从站点源码看到其使用vuejs构建,vuejs的拖拽功能是通过vue.draggable.next来实现的,它是基于SortableJS这个开源包的支持来实现网页拖拽行为。
于是最终的想法是利用drag and drop API来模拟拖拽动作

结果

下面的代码参考了很多资料,真实有效的解决了vuejs上拖拽操作的模拟。代码中有注释做简单说明,希望对大家有帮助。
拖拽动作到底要触发哪些事件,其实是参考了cypress.js 的执行结果,发现它使用了以下事件来模拟拖拽:

  1. pointerdown

  2. mousedown

  3. dragstart

  4. dragover

  5. mousemove

  6. pointermove

  7. drop

  8. mouseup

  9. pointerup
    所以下面的代码总的来说就是使用JS创建以上的事件,然后按照顺序执行这一系列事件,从而达到模拟拖拽行为。
    比较重要的一点,是要意识到:拖拽元素是通过drag event的dataTransfer来传输,所以需要在dragstart事件构造一个dataTransfer对象,然后把该dataTransfer对象传给后续的drag事件,以此绑定元素到拖拽动作中。

     var triggerDragAndDrop = function (source, target, offsetY = 0) {
         var DELAY_INTERVAL_MS = 100;
         var MAX_TRIES = 10;
         var dragStartEvent;
    
         function createNewDataTransfer() {
             var data = {};
             return {
                 clearData: function (key) {
                     if (key === undefined) {
                         data = {};
                     } else {
                         delete data[key];
                     }
                 },
                 getData: function (key) {
                     return data[key];
                 },
                 setData: function (key, value) {
                     data[key] = value;
                 },
                 setDragImage: function () { },
                 dropEffect: 'move',
                 files: [],
                 items: [],
                 types: ['text/plain'],
                 effectAllowed: 'move'
             }
         };
         var fireDragEvent = function (type, elem, clientX, clientY, dataTransfer) {
             let event = new DragEvent(type, {
                 bubbles: true,
                 cancelable: true,
                 view: window,
                 clientX: clientX,
                 clientY: clientY,
                 pageX: clientX,
                 pageY: clientY,
                 screenX: clientX,
                 screenY: clientY,
                 relatedTarget: elem
             });
             event.dataTransfer = dataTransfer || createNewDataTransfer();
             elem.dispatchEvent(event);
             return event;
         }
         var firePointerEvent = function (type, elem, clientX, clientY) {
             let event = new PointerEvent(type, {
                 bubbles: true,
                 cancelable: true,
                 view: window,
                 pageX: clientX,
                 pageY: clientY,
                 screenX: clientX,
                 screenY: clientY,
                 button: 0,
                 which: 1
             });
             event.preventDefault()
             elem.dispatchEvent(event)
         }
         var firePointerMoveEvent = function (type, elem, clientX, clientY) {
             let event = new PointerEvent(type, {
                 bubbles: true,
                 cancelable: true,
                 view: window,
                 pageX: clientX,
                 pageY: clientY,
                 screenX: clientX,
                 screenY: clientY,
             });
             event.preventDefault()
             elem.dispatchEvent(event)
         }
         var fireMouseMoveEvent = function (type, elem, clientX, clientY) {
             var event = new MouseEvent(type, {
                 bubbles: true,
                 cancelable: true,
                 view: window,
                 pageX: clientX,
                 pageY: clientY,
                 screenX: clientX,
                 screenY: clientY,
             });
             event.preventDefault()
             elem.dispatchEvent(event)
         }
         var fireMouseEvent = function (type, elem, clientX, clientY) {
             var event = new MouseEvent(type, {
                 bubbles: true,
                 cancelable: true,
                 button: 0,
                 view: window,
                 pageX: clientX,
                 pageY: clientY,
                 screenX: clientX,
                 screenY: clientY,
                 which: 1
             });
             event.preventDefault()
             elem.dispatchEvent(event)
         }
         // fetch target elements
         var elemDrag = document.querySelector(source)
         var elemDrop = document.querySelector(target)
         if (!elemDrag || !elemDrop) return false;
    
         // calculate positions
         var pos = elemDrag.getBoundingClientRect()
         var center1X = Math.floor((pos.left + (pos.width / 2)))
         var center1Y = Math.floor((pos.top + (pos.height / 2)))
         pos = elemDrop.getBoundingClientRect()
         var center2X = Math.floor((pos.left + (pos.width / 2)))
         var center2Y = pos.bottom
         var counter = 0;
         var startingDropRect = elemDrop.getBoundingClientRect();
    
         function rectsEqual(r1, r2) {
             return r1.top === r2.top && r1.right === r2.right && r1.bottom === r2.bottom && r1.left === r2.left;
         }
         function dragover() {
             counter++;
             console.log('DRAGOVER #' + counter);
    
             var currentDropRect = elemDrop.getBoundingClientRect();
             if (rectsEqual(startingDropRect, currentDropRect) && counter < MAX_TRIES) {
                 if (counter != 1) console.log("drop target rect hasn't changed, trying again");
    
                 fireDragEvent("dragover", elemDrop, center2X, center2Y + offsetY, dragStartEvent.dataTransfer)
                 fireMouseMoveEvent("mousemove", elemDrop, center2X, center2Y + offsetY)
                 firePointerMoveEvent("pointermove", elemDrop, center2X, center2Y + offsetY)
                 setTimeout(dragover, DELAY_INTERVAL_MS);
             } else {
                 if (rectsEqual(startingDropRect, currentDropRect)) {
                     console.log("wasn't able to budge drop target after " + MAX_TRIES + " tries, aborting");
                     fireDragEvent("drop", elemDrop, center2X, center2Y + offsetY, dragStartEvent.dataTransfer)
                     fireMouseEvent("mouseup", elemDrop, center2X, center2Y + offsetY)
                     firePointerEvent("pointerup", elemDrop, center2X, center2Y + offsetY)
    
                 } else {
                     setTimeout(drop, DELAY_INTERVAL_MS);
                 }
                 setTimeout(drop, DELAY_INTERVAL_MS);
             }
         }
    
         function drop() {
             console.log('DROP');
             fireDragEvent("drop", elemDrop, center2X, center2Y + offsetY, dragStartEvent.dataTransfer)
             fireMouseEvent("mouseup", elemDrop, center2X, center2Y + offsetY)
             firePointerEvent("pointerup", elemDrop, center2X, center2Y + offsetY)  
    
         }
         // start dragging process
         console.log('DRAGSTART');
    
         firePointerEvent("pointerdown", elemDrag, center1X, center1Y);
         fireMouseEvent("mousedown", elemDrag, center1X, center1Y)
         dragStartEvent = fireDragEvent("dragstart", elemDrag, center1X, center1Y)
    
         setTimeout(dragover, DELAY_INTERVAL_MS);
         return true
     }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343

推荐阅读更多精彩内容