无法使用 Phantomjs 单击元素
Cannot click element with Phantomjs
来自post:How to ng-click an A directive in a PhantomJS test我知道我需要创建自己的函数来点击一个元素,但我该如何使用它?
下面的代码给我错误 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);
}
page.evaluate(function () {
document.getElementById("login").click();
});
有效...不是你的意思吗?
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();
});
参考文献:
page.evaluate()
的文档
- 在 PhantomJS 中 "click" 的方法:Q: PhantomJS; click an element
我找到问题了。该点击被网站的 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);
通过PhantomJS; click an element
找到上述解决方案
来自post:How to ng-click an A directive in a PhantomJS test我知道我需要创建自己的函数来点击一个元素,但我该如何使用它?
下面的代码给我错误 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);
}
page.evaluate(function () {
document.getElementById("login").click();
});
有效...不是你的意思吗?
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();
});
参考文献:
page.evaluate()
的文档
- 在 PhantomJS 中 "click" 的方法:Q: PhantomJS; click an element
我找到问题了。该点击被网站的 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);
通过PhantomJS; click an element
找到上述解决方案