fix(page.click): teach puppeteer click wrapped links (#2822)

This patch teaches Puppeteer to click elements that are
part of inline layout and that wrap on multiple lines.

Fixes #2798.
This commit is contained in:
Andrey Lushnikov 2018-06-29 12:03:02 -07:00 committed by GitHub
parent 59e7f7ebb6
commit 5955affab0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 79 additions and 22 deletions

View File

@ -15,7 +15,7 @@
*/ */
const path = require('path'); const path = require('path');
const {JSHandle} = require('./ExecutionContext'); const {JSHandle} = require('./ExecutionContext');
const {helper, debugError} = require('./helper'); const {helper, assert, debugError} = require('./helper');
class ElementHandle extends JSHandle { class ElementHandle extends JSHandle {
/** /**
@ -76,13 +76,29 @@ class ElementHandle extends JSHandle {
} }
/** /**
* @return {!Promise<{x: number, y: number}>} * @return {!Promise<!{x: number, y: number}>}
*/ */
async _boundingBoxCenter() { async _clickablePoint() {
const box = await this._assertBoundingBox(); const result = await this._client.send('DOM.getContentQuads', {
objectId: this._remoteObject.objectId
}).catch(debugError);
if (!result || !result.quads.length)
throw new Error('Node is either not visible or not an HTMLElement');
// Filter out quads that have too small area to click into.
const quads = result.quads.map(quad => this._fromProtocolQuad(quad)).filter(quad => computeQuadArea(quad) > 1);
if (!quads.length)
throw new Error('Node is either not visible or not an HTMLElement');
// Return the middle point of the first quad.
const quad = quads[0];
let x = 0;
let y = 0;
for (const point of quad) {
x += point.x;
y += point.y;
}
return { return {
x: box.x + box.width / 2, x: x / 4,
y: box.y + box.height / 2 y: y / 4
}; };
} }
@ -110,7 +126,7 @@ class ElementHandle extends JSHandle {
async hover() { async hover() {
await this._scrollIntoViewIfNeeded(); await this._scrollIntoViewIfNeeded();
const {x, y} = await this._boundingBoxCenter(); const {x, y} = await this._clickablePoint();
await this._page.mouse.move(x, y); await this._page.mouse.move(x, y);
} }
@ -119,7 +135,7 @@ class ElementHandle extends JSHandle {
*/ */
async click(options = {}) { async click(options = {}) {
await this._scrollIntoViewIfNeeded(); await this._scrollIntoViewIfNeeded();
const {x, y} = await this._boundingBoxCenter(); const {x, y} = await this._clickablePoint();
await this._page.mouse.click(x, y, options); await this._page.mouse.click(x, y, options);
} }
@ -135,7 +151,7 @@ class ElementHandle extends JSHandle {
async tap() { async tap() {
await this._scrollIntoViewIfNeeded(); await this._scrollIntoViewIfNeeded();
const {x, y} = await this._boundingBoxCenter(); const {x, y} = await this._clickablePoint();
await this._page.touchscreen.tap(x, y); await this._page.touchscreen.tap(x, y);
} }
@ -199,17 +215,6 @@ class ElementHandle extends JSHandle {
}; };
} }
/**
* @return {!Promise<?{x: number, y: number, width: number, height: number}>}
*/
async _assertBoundingBox() {
const boundingBox = await this.boundingBox();
if (boundingBox)
return boundingBox;
throw new Error('Node is either not visible or not an HTMLElement');
}
/** /**
* *
* @param {!Object=} options * @param {!Object=} options
@ -218,7 +223,8 @@ class ElementHandle extends JSHandle {
async screenshot(options = {}) { async screenshot(options = {}) {
let needsViewportReset = false; let needsViewportReset = false;
let boundingBox = await this._assertBoundingBox(); let boundingBox = await this.boundingBox();
assert(boundingBox, 'Node is either not visible or not an HTMLElement');
const viewport = this._page.viewport(); const viewport = this._page.viewport();
@ -234,7 +240,8 @@ class ElementHandle extends JSHandle {
await this._scrollIntoViewIfNeeded(); await this._scrollIntoViewIfNeeded();
boundingBox = await this._assertBoundingBox(); boundingBox = await this.boundingBox();
assert(boundingBox, 'Node is either not visible or not an HTMLElement');
const { layoutViewport: { pageX, pageY } } = await this._client.send('Page.getLayoutMetrics'); const { layoutViewport: { pageX, pageY } } = await this._client.send('Page.getLayoutMetrics');
@ -349,5 +356,17 @@ class ElementHandle extends JSHandle {
} }
} }
function computeQuadArea(quad) {
// Compute sum of all directed areas of adjacent triangles
// https://en.wikipedia.org/wiki/Polygon#Simple_polygons
let area = 0;
for (let i = 0; i < quad.length; ++i) {
const p1 = quad[i];
const p2 = quad[(i + 1) % quad.length];
area += (p1.x * p2.y - p2.x * p1.y) / 2;
}
return area;
}
module.exports = ElementHandle; module.exports = ElementHandle;
helper.tracePublicAPI(ElementHandle); helper.tracePublicAPI(ElementHandle);

View File

@ -0,0 +1,29 @@
<style>
:root {
font-family: monospace;
}
body {
display: flex;
align-items: center;
justify-content: center;
}
div {
width: 10ch;
word-wrap: break-word;
border: 1px solid blue;
transform: rotate(33deg);
line-height: 8ch;
padding: 2ch;
}
a {
margin-left: 7ch;
}
</style>
<div>
<a href='#clicked'>123321</a>
</div>
<script>
</script>

View File

@ -52,6 +52,15 @@ module.exports.addTests = function({testRunner, expect, DeviceDescriptors}) {
]); ]);
}); });
it('should click wrapped links', async({page, server}) => {
await page.goto(server.PREFIX + '/wrappedlink.html');
await Promise.all([
page.click('a'),
page.waitForNavigation()
]);
expect(page.url()).toBe(server.PREFIX + '/wrappedlink.html#clicked');
});
it('should click on checkbox input and toggle', async({page, server}) => { it('should click on checkbox input and toggle', async({page, server}) => {
await page.goto(server.PREFIX + '/input/checkbox.html'); await page.goto(server.PREFIX + '/input/checkbox.html');
expect(await page.evaluate(() => result.check)).toBe(null); expect(await page.evaluate(() => result.check)).toBe(null);