无法使用Phantomjs单击元素

zol*_*tar 5 javascript click phantomjs

一篇如何在PhantomJS测试中ng单击A指令我知道我需要创建自己的函数来单击元素,但是如何使用它呢?

以下代码给我错误 TypeError: 'undefined' is not a function (evaluating 'click(obj)')

var page = require('webpage').create();
var url = 'http://somesite.com/document.html';

page.open(url, function(status) {
    page.evaluate(function() {
        var obj = document.querySelectorAll('button')[0];
        click(obj);
    });
    console.log(page.content);
    phantom.exit();
});


function click(el){
    var ev = document.createEvent('MouseEvent');
    ev.initMouseEvent(
        'click',
        true /* bubble */, true /* cancelable */,
        window, null,
        0, 0, 0, 0, /* coordinates */
        false, false, false, false, /* modifier keys */
        0 /*left*/, null
    );
    el.dispatchEvent(ev);
}
Run Code Online (Sandbox Code Playgroud)

Art*_* B. 5

PhantomJS有两个上下文。只有通过访问的页面上下文可以page.evaluate()访问DOM,并且可以触发综合事件。由于您的click()函数使用document,因此需要在页面上下文中运行它。

page.evaluate()是沙盒,这就是为什么您不能简单地使用外部定义的变量的原因。您要么需要在页面上下文中定义它们,要么需要以复杂的方式传递它们(因为只能将原始对象传入和传出页面上下文,而函数不是原始对象)。

page.open(url, function(status) {
    page.evaluate(function() {
        if (typeof window.click !== "function") {
            window.click = function(el){
                var ev = document.createEvent('MouseEvent');
                ev.initMouseEvent(
                    'click',
                    true /* bubble */, true /* cancelable */,
                    window, null,
                    0, 0, 0, 0, /* coordinates */
                    false, false, false, false, /* modifier keys */
                    0 /*left*/, null
                );
                el.dispatchEvent(ev);
            }
        }
        var obj = document.querySelectorAll('button')[0];
        click(obj);
    });
    console.log(page.content);
    phantom.exit();
});
Run Code Online (Sandbox Code Playgroud)

参考文献:


zol*_*tar 1

我解决了这个问题。该网站的 JavaScript 绕过了点击。幸运的是,PhantomJS 有一个 sendEvent 函数,可以像真实用户一样发送事件:

var rect = page.evaluate(function() {
    return $('a.whatever')[0].getBoundingClientRect();
});
page.sendEvent('click', rect.left + rect.width / 2, rect.top + rect.height / 2);
Run Code Online (Sandbox Code Playgroud)

通过 PhantomJS 找到上述解决方案;单击一个元素