2018-11-21 03:43:07 +00:00
|
|
|
/**
|
2024-01-03 10:11:33 +00:00
|
|
|
* @license
|
|
|
|
* Copyright 2018 Google Inc.
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2018-11-21 03:43:07 +00:00
|
|
|
*/
|
2020-06-23 05:18:46 +00:00
|
|
|
import os from 'os';
|
2023-02-15 23:09:31 +00:00
|
|
|
|
2020-06-23 05:18:46 +00:00
|
|
|
import expect from 'expect';
|
2023-06-19 08:12:43 +00:00
|
|
|
import {MouseButton} from 'puppeteer-core/internal/api/Input.js';
|
2023-09-26 16:24:24 +00:00
|
|
|
import type {Page} from 'puppeteer-core/internal/api/Page.js';
|
|
|
|
import type {KeyInput} from 'puppeteer-core/internal/common/USKeyboardLayout.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
|
2023-07-03 12:01:29 +00:00
|
|
|
import {getTestState, setupTestBrowserHooks} from './mocha-utils.js';
|
2018-11-21 03:43:07 +00:00
|
|
|
|
2023-06-12 17:59:28 +00:00
|
|
|
interface ClickData {
|
|
|
|
type: string;
|
|
|
|
detail: number;
|
|
|
|
clientX: number;
|
|
|
|
clientY: number;
|
|
|
|
isTrusted: boolean;
|
|
|
|
button: number;
|
|
|
|
buttons: number;
|
|
|
|
}
|
|
|
|
|
2020-06-23 05:18:46 +00:00
|
|
|
interface Dimensions {
|
|
|
|
x: number;
|
|
|
|
y: number;
|
|
|
|
width: number;
|
|
|
|
height: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
function dimensions(): Dimensions {
|
2022-06-15 10:09:22 +00:00
|
|
|
const rect = document.querySelector('textarea')!.getBoundingClientRect();
|
2018-11-21 03:43:07 +00:00
|
|
|
return {
|
|
|
|
x: rect.left,
|
|
|
|
y: rect.top,
|
|
|
|
width: rect.width,
|
2020-05-07 10:54:55 +00:00
|
|
|
height: rect.height,
|
2018-11-21 03:43:07 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-05-07 10:54:55 +00:00
|
|
|
describe('Mouse', function () {
|
2023-07-03 12:01:29 +00:00
|
|
|
setupTestBrowserHooks();
|
|
|
|
|
2020-05-07 10:54:55 +00:00
|
|
|
it('should click the document', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page} = await getTestState();
|
2018-11-21 03:43:07 +00:00
|
|
|
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.evaluate(() => {
|
2022-06-22 13:25:44 +00:00
|
|
|
(globalThis as any).clickPromise = new Promise(resolve => {
|
|
|
|
document.addEventListener('click', event => {
|
2020-04-09 05:56:25 +00:00
|
|
|
resolve({
|
|
|
|
type: event.type,
|
|
|
|
detail: event.detail,
|
|
|
|
clientX: event.clientX,
|
|
|
|
clientY: event.clientY,
|
|
|
|
isTrusted: event.isTrusted,
|
2020-05-07 10:54:55 +00:00
|
|
|
button: event.button,
|
2019-02-03 01:49:12 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2018-11-21 03:43:07 +00:00
|
|
|
});
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.mouse.click(50, 60);
|
feat!: type inference for evaluation types (#8547)
This PR greatly improves the types within Puppeteer:
- **Almost everything** is auto-deduced.
- Parameters don't need to be specified in the function. They are deduced from the spread.
- Return types don't need to be specified. They are deduced from the function. (More on this below)
- Selections based on tag names correctly deduce element type, similar to TypeScript's mechanism for `getElementByTagName`.
- [**BREAKING CHANGE**] We've removed the ability to declare return types in type arguments for the following reasons:
1. Setting them will indubitably break auto-deduction.
2. You can just use `as ...` in TypeScript to coerce the correct type (given it makes sense).
- [**BREAKING CHANGE**] `waitFor` is officially gone.
To migrate to these changes, there are only four things you may need to change:
- If you set a return type using the `ReturnType` type parameter, remove it and use `as ...` and `HandleFor` (if necessary).
⛔ `evaluate<ReturnType>(a: number, b: number) => {...}, a, b)`
✅ `(await evaluate(a, b) => {...}, a, b)) as ReturnType`
⛔ `evaluateHandle<ReturnType>(a: number, b: number) => {...}, a, b)`
✅ `(await evaluateHandle(a, b) => {...}, a, b)) as HandleFor<ReturnType>`
- If you set any type parameters in the *parameters* of an evaluation function, remove them.
⛔ `evaluate(a: number, b: number) => {...}, a, b)`
✅ `evaluate(a, b) => {...}, a, b)`
- If you set any type parameters in the method's declaration, remove them.
⛔ `evaluate<(a: number, b: number) => void>((a, b) => {...}, a, b)`
✅ `evaluate(a, b) => {...}, a, b)`
2022-06-23 09:29:46 +00:00
|
|
|
const event = await page.evaluate(() => {
|
2022-06-15 10:09:22 +00:00
|
|
|
return (globalThis as any).clickPromise;
|
|
|
|
});
|
2020-04-09 05:56:25 +00:00
|
|
|
expect(event.type).toBe('click');
|
|
|
|
expect(event.detail).toBe(1);
|
|
|
|
expect(event.clientX).toBe(50);
|
|
|
|
expect(event.clientY).toBe(60);
|
|
|
|
expect(event.isTrusted).toBe(true);
|
|
|
|
expect(event.button).toBe(0);
|
|
|
|
});
|
2021-11-19 07:50:31 +00:00
|
|
|
it('should resize the textarea', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server} = await getTestState();
|
2020-04-09 05:56:25 +00:00
|
|
|
|
|
|
|
await page.goto(server.PREFIX + '/input/textarea.html');
|
feat!: type inference for evaluation types (#8547)
This PR greatly improves the types within Puppeteer:
- **Almost everything** is auto-deduced.
- Parameters don't need to be specified in the function. They are deduced from the spread.
- Return types don't need to be specified. They are deduced from the function. (More on this below)
- Selections based on tag names correctly deduce element type, similar to TypeScript's mechanism for `getElementByTagName`.
- [**BREAKING CHANGE**] We've removed the ability to declare return types in type arguments for the following reasons:
1. Setting them will indubitably break auto-deduction.
2. You can just use `as ...` in TypeScript to coerce the correct type (given it makes sense).
- [**BREAKING CHANGE**] `waitFor` is officially gone.
To migrate to these changes, there are only four things you may need to change:
- If you set a return type using the `ReturnType` type parameter, remove it and use `as ...` and `HandleFor` (if necessary).
⛔ `evaluate<ReturnType>(a: number, b: number) => {...}, a, b)`
✅ `(await evaluate(a, b) => {...}, a, b)) as ReturnType`
⛔ `evaluateHandle<ReturnType>(a: number, b: number) => {...}, a, b)`
✅ `(await evaluateHandle(a, b) => {...}, a, b)) as HandleFor<ReturnType>`
- If you set any type parameters in the *parameters* of an evaluation function, remove them.
⛔ `evaluate(a: number, b: number) => {...}, a, b)`
✅ `evaluate(a, b) => {...}, a, b)`
- If you set any type parameters in the method's declaration, remove them.
⛔ `evaluate<(a: number, b: number) => void>((a, b) => {...}, a, b)`
✅ `evaluate(a, b) => {...}, a, b)`
2022-06-23 09:29:46 +00:00
|
|
|
const {x, y, width, height} = await page.evaluate(dimensions);
|
2020-04-09 05:56:25 +00:00
|
|
|
const mouse = page.mouse;
|
|
|
|
await mouse.move(x + width - 4, y + height - 4);
|
|
|
|
await mouse.down();
|
|
|
|
await mouse.move(x + width + 100, y + height + 100);
|
|
|
|
await mouse.up();
|
feat!: type inference for evaluation types (#8547)
This PR greatly improves the types within Puppeteer:
- **Almost everything** is auto-deduced.
- Parameters don't need to be specified in the function. They are deduced from the spread.
- Return types don't need to be specified. They are deduced from the function. (More on this below)
- Selections based on tag names correctly deduce element type, similar to TypeScript's mechanism for `getElementByTagName`.
- [**BREAKING CHANGE**] We've removed the ability to declare return types in type arguments for the following reasons:
1. Setting them will indubitably break auto-deduction.
2. You can just use `as ...` in TypeScript to coerce the correct type (given it makes sense).
- [**BREAKING CHANGE**] `waitFor` is officially gone.
To migrate to these changes, there are only four things you may need to change:
- If you set a return type using the `ReturnType` type parameter, remove it and use `as ...` and `HandleFor` (if necessary).
⛔ `evaluate<ReturnType>(a: number, b: number) => {...}, a, b)`
✅ `(await evaluate(a, b) => {...}, a, b)) as ReturnType`
⛔ `evaluateHandle<ReturnType>(a: number, b: number) => {...}, a, b)`
✅ `(await evaluateHandle(a, b) => {...}, a, b)) as HandleFor<ReturnType>`
- If you set any type parameters in the *parameters* of an evaluation function, remove them.
⛔ `evaluate(a: number, b: number) => {...}, a, b)`
✅ `evaluate(a, b) => {...}, a, b)`
- If you set any type parameters in the method's declaration, remove them.
⛔ `evaluate<(a: number, b: number) => void>((a, b) => {...}, a, b)`
✅ `evaluate(a, b) => {...}, a, b)`
2022-06-23 09:29:46 +00:00
|
|
|
const newDimensions = await page.evaluate(dimensions);
|
2020-04-09 05:56:25 +00:00
|
|
|
expect(newDimensions.width).toBe(Math.round(width + 104));
|
|
|
|
expect(newDimensions.height).toBe(Math.round(height + 104));
|
|
|
|
});
|
2021-11-19 07:50:31 +00:00
|
|
|
it('should select the text with mouse', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server} = await getTestState();
|
2020-04-09 05:56:25 +00:00
|
|
|
|
2020-05-07 10:54:55 +00:00
|
|
|
const text =
|
|
|
|
"This is the text that we are going to try to select. Let's see how it goes.";
|
2023-09-21 20:50:36 +00:00
|
|
|
|
|
|
|
await page.goto(`${server.PREFIX}/input/textarea.html`);
|
|
|
|
await page.focus('textarea');
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.keyboard.type(text);
|
2023-09-21 20:50:36 +00:00
|
|
|
using handle = await page
|
|
|
|
.locator('textarea')
|
|
|
|
.filterHandle(async element => {
|
|
|
|
return await element.evaluate((element, text) => {
|
|
|
|
return element.value === text;
|
|
|
|
}, text);
|
|
|
|
})
|
|
|
|
.waitHandle();
|
2022-06-22 13:25:44 +00:00
|
|
|
const {x, y} = await page.evaluate(dimensions);
|
2020-05-07 10:54:55 +00:00
|
|
|
await page.mouse.move(x + 2, y + 2);
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.mouse.down();
|
2020-05-07 10:54:55 +00:00
|
|
|
await page.mouse.move(100, 100);
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.mouse.up();
|
2020-05-07 10:54:55 +00:00
|
|
|
expect(
|
2023-09-21 20:50:36 +00:00
|
|
|
await handle.evaluate(element => {
|
|
|
|
return element.value.substring(
|
|
|
|
element.selectionStart,
|
|
|
|
element.selectionEnd
|
2020-05-07 10:54:55 +00:00
|
|
|
);
|
|
|
|
})
|
|
|
|
).toBe(text);
|
2020-04-09 05:56:25 +00:00
|
|
|
});
|
2022-09-08 10:32:39 +00:00
|
|
|
it('should trigger hover state', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server} = await getTestState();
|
2020-04-09 05:56:25 +00:00
|
|
|
|
|
|
|
await page.goto(server.PREFIX + '/input/scrollable.html');
|
|
|
|
await page.hover('#button-6');
|
2020-05-07 10:54:55 +00:00
|
|
|
expect(
|
2022-06-15 10:09:22 +00:00
|
|
|
await page.evaluate(() => {
|
|
|
|
return document.querySelector('button:hover')!.id;
|
|
|
|
})
|
2020-05-07 10:54:55 +00:00
|
|
|
).toBe('button-6');
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.hover('#button-2');
|
2020-05-07 10:54:55 +00:00
|
|
|
expect(
|
2022-06-15 10:09:22 +00:00
|
|
|
await page.evaluate(() => {
|
|
|
|
return document.querySelector('button:hover')!.id;
|
|
|
|
})
|
2020-05-07 10:54:55 +00:00
|
|
|
).toBe('button-2');
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.hover('#button-91');
|
2020-05-07 10:54:55 +00:00
|
|
|
expect(
|
2022-06-15 10:09:22 +00:00
|
|
|
await page.evaluate(() => {
|
|
|
|
return document.querySelector('button:hover')!.id;
|
|
|
|
})
|
2020-05-07 10:54:55 +00:00
|
|
|
).toBe('button-91');
|
2020-04-09 05:56:25 +00:00
|
|
|
});
|
2022-09-08 10:32:39 +00:00
|
|
|
it('should trigger hover state with removed window.Node', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server} = await getTestState();
|
2020-04-09 05:56:25 +00:00
|
|
|
|
2022-09-08 10:32:39 +00:00
|
|
|
await page.goto(server.PREFIX + '/input/scrollable.html');
|
|
|
|
await page.evaluate(() => {
|
|
|
|
// @ts-expect-error Expected.
|
|
|
|
return delete window.Node;
|
|
|
|
});
|
|
|
|
await page.hover('#button-6');
|
|
|
|
expect(
|
2022-06-15 10:09:22 +00:00
|
|
|
await page.evaluate(() => {
|
2022-09-08 10:32:39 +00:00
|
|
|
return document.querySelector('button:hover')!.id;
|
|
|
|
})
|
|
|
|
).toBe('button-6');
|
|
|
|
});
|
2020-05-07 10:54:55 +00:00
|
|
|
it('should set modifier keys on click', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server, isFirefox} = await getTestState();
|
2020-04-09 05:56:25 +00:00
|
|
|
|
|
|
|
await page.goto(server.PREFIX + '/input/scrollable.html');
|
2022-06-15 10:09:22 +00:00
|
|
|
await page.evaluate(() => {
|
|
|
|
return document.querySelector('#button-3')!.addEventListener(
|
|
|
|
'mousedown',
|
2022-06-22 13:25:44 +00:00
|
|
|
e => {
|
2022-06-15 10:09:22 +00:00
|
|
|
return ((globalThis as any).lastEvent = e);
|
|
|
|
},
|
|
|
|
true
|
|
|
|
);
|
|
|
|
});
|
2020-06-23 05:18:46 +00:00
|
|
|
const modifiers = new Map<KeyInput, string>([
|
|
|
|
['Shift', 'shiftKey'],
|
|
|
|
['Control', 'ctrlKey'],
|
|
|
|
['Alt', 'altKey'],
|
|
|
|
['Meta', 'metaKey'],
|
|
|
|
]);
|
2020-04-09 05:56:25 +00:00
|
|
|
// In Firefox, the Meta modifier only exists on Mac
|
2022-06-14 11:55:35 +00:00
|
|
|
if (isFirefox && os.platform() !== 'darwin') {
|
2022-06-15 10:09:22 +00:00
|
|
|
modifiers.delete('Meta');
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2020-06-23 05:18:46 +00:00
|
|
|
for (const [modifier, key] of modifiers) {
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.keyboard.down(modifier);
|
2018-11-21 03:43:07 +00:00
|
|
|
await page.click('#button-3');
|
2020-07-10 10:52:13 +00:00
|
|
|
if (
|
2022-06-15 10:09:22 +00:00
|
|
|
!(await page.evaluate((mod: string) => {
|
|
|
|
return (globalThis as any).lastEvent[mod];
|
|
|
|
}, key))
|
2022-06-14 11:55:35 +00:00
|
|
|
) {
|
2020-06-23 05:18:46 +00:00
|
|
|
throw new Error(key + ' should be true');
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2020-04-09 05:56:25 +00:00
|
|
|
await page.keyboard.up(modifier);
|
|
|
|
}
|
|
|
|
await page.click('#button-3');
|
2020-06-23 05:18:46 +00:00
|
|
|
for (const [modifier, key] of modifiers) {
|
2022-06-14 11:55:35 +00:00
|
|
|
if (
|
2022-06-15 10:09:22 +00:00
|
|
|
await page.evaluate((mod: string) => {
|
|
|
|
return (globalThis as any).lastEvent[mod];
|
|
|
|
}, key)
|
2022-06-14 11:55:35 +00:00
|
|
|
) {
|
2022-06-15 10:09:22 +00:00
|
|
|
throw new Error(modifiers.get(modifier) + ' should be false');
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2020-04-09 05:56:25 +00:00
|
|
|
}
|
|
|
|
});
|
2022-09-08 10:32:39 +00:00
|
|
|
it('should send mouse wheel events', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server} = await getTestState();
|
2020-07-06 07:27:17 +00:00
|
|
|
|
|
|
|
await page.goto(server.PREFIX + '/input/wheel.html');
|
2023-08-30 10:02:59 +00:00
|
|
|
using elem = (await page.$('div'))!;
|
2022-06-15 10:09:22 +00:00
|
|
|
const boundingBoxBefore = (await elem.boundingBox())!;
|
2020-07-06 07:27:17 +00:00
|
|
|
expect(boundingBoxBefore).toMatchObject({
|
|
|
|
width: 115,
|
|
|
|
height: 115,
|
|
|
|
});
|
|
|
|
|
|
|
|
await page.mouse.move(
|
|
|
|
boundingBoxBefore.x + boundingBoxBefore.width / 2,
|
|
|
|
boundingBoxBefore.y + boundingBoxBefore.height / 2
|
|
|
|
);
|
|
|
|
|
2022-06-22 13:25:44 +00:00
|
|
|
await page.mouse.wheel({deltaY: -100});
|
2020-07-06 07:27:17 +00:00
|
|
|
const boundingBoxAfter = await elem.boundingBox();
|
|
|
|
expect(boundingBoxAfter).toMatchObject({
|
|
|
|
width: 230,
|
|
|
|
height: 230,
|
|
|
|
});
|
|
|
|
});
|
2022-09-08 10:32:39 +00:00
|
|
|
it('should tween mouse movement', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page} = await getTestState();
|
2020-04-09 05:56:25 +00:00
|
|
|
|
|
|
|
await page.mouse.move(100, 100);
|
|
|
|
await page.evaluate(() => {
|
2022-06-15 10:09:22 +00:00
|
|
|
(globalThis as any).result = [];
|
2022-06-22 13:25:44 +00:00
|
|
|
document.addEventListener('mousemove', event => {
|
2022-06-15 10:09:22 +00:00
|
|
|
(globalThis as any).result.push([event.clientX, event.clientY]);
|
2018-11-21 03:43:07 +00:00
|
|
|
});
|
|
|
|
});
|
2022-06-22 13:25:44 +00:00
|
|
|
await page.mouse.move(200, 300, {steps: 5});
|
2020-04-09 05:56:25 +00:00
|
|
|
expect(await page.evaluate('result')).toEqual([
|
|
|
|
[120, 140],
|
|
|
|
[140, 180],
|
|
|
|
[160, 220],
|
|
|
|
[180, 260],
|
2020-05-07 10:54:55 +00:00
|
|
|
[200, 300],
|
2020-04-09 05:56:25 +00:00
|
|
|
]);
|
|
|
|
});
|
|
|
|
// @see https://crbug.com/929806
|
2021-11-19 07:50:31 +00:00
|
|
|
it('should work with mobile viewports and cross process navigations', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server} = await getTestState();
|
2020-04-09 05:56:25 +00:00
|
|
|
|
2021-11-19 07:50:31 +00:00
|
|
|
await page.goto(server.EMPTY_PAGE);
|
2022-06-22 13:25:44 +00:00
|
|
|
await page.setViewport({width: 360, height: 640, isMobile: true});
|
2021-11-19 07:50:31 +00:00
|
|
|
await page.goto(server.CROSS_PROCESS_PREFIX + '/mobile.html');
|
|
|
|
await page.evaluate(() => {
|
2022-06-22 13:25:44 +00:00
|
|
|
document.addEventListener('click', event => {
|
|
|
|
(globalThis as any).result = {x: event.clientX, y: event.clientY};
|
2019-02-11 22:00:06 +00:00
|
|
|
});
|
2021-11-19 07:50:31 +00:00
|
|
|
});
|
2019-02-11 22:00:06 +00:00
|
|
|
|
2021-11-19 07:50:31 +00:00
|
|
|
await page.mouse.click(30, 40);
|
2019-02-11 22:00:06 +00:00
|
|
|
|
2022-06-22 13:25:44 +00:00
|
|
|
expect(await page.evaluate('result')).toEqual({x: 30, y: 40});
|
2021-11-19 07:50:31 +00:00
|
|
|
});
|
2023-09-20 15:58:02 +00:00
|
|
|
it('should not throw if buttons are pressed twice', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server} = await getTestState();
|
2023-04-17 08:56:51 +00:00
|
|
|
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
|
|
|
|
await page.mouse.down();
|
2023-09-20 15:58:02 +00:00
|
|
|
await page.mouse.down();
|
2023-04-17 08:56:51 +00:00
|
|
|
});
|
|
|
|
|
2023-06-12 17:59:28 +00:00
|
|
|
interface AddMouseDataListenersOptions {
|
|
|
|
includeMove?: boolean;
|
|
|
|
}
|
2023-04-17 08:56:51 +00:00
|
|
|
|
2023-06-12 17:59:28 +00:00
|
|
|
const addMouseDataListeners = (
|
|
|
|
page: Page,
|
|
|
|
options: AddMouseDataListenersOptions = {}
|
|
|
|
) => {
|
|
|
|
return page.evaluate(({includeMove}) => {
|
2023-04-17 08:56:51 +00:00
|
|
|
const clicks: ClickData[] = [];
|
|
|
|
const mouseEventListener = (event: MouseEvent) => {
|
|
|
|
clicks.push({
|
|
|
|
type: event.type,
|
|
|
|
detail: event.detail,
|
|
|
|
clientX: event.clientX,
|
|
|
|
clientY: event.clientY,
|
|
|
|
isTrusted: event.isTrusted,
|
|
|
|
button: event.button,
|
|
|
|
buttons: event.buttons,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
document.addEventListener('mousedown', mouseEventListener);
|
2023-06-12 17:59:28 +00:00
|
|
|
if (includeMove) {
|
|
|
|
document.addEventListener('mousemove', mouseEventListener);
|
|
|
|
}
|
2023-04-17 08:56:51 +00:00
|
|
|
document.addEventListener('mouseup', mouseEventListener);
|
|
|
|
document.addEventListener('click', mouseEventListener);
|
2023-09-20 15:58:02 +00:00
|
|
|
document.addEventListener('auxclick', mouseEventListener);
|
2023-04-17 08:56:51 +00:00
|
|
|
(window as unknown as {clicks: ClickData[]}).clicks = clicks;
|
2023-06-12 17:59:28 +00:00
|
|
|
}, options);
|
|
|
|
};
|
|
|
|
|
|
|
|
it('should not throw if clicking in parallel', async () => {
|
2023-06-21 19:41:09 +00:00
|
|
|
const {page, server} = await getTestState();
|
2023-06-12 17:59:28 +00:00
|
|
|
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await addMouseDataListeners(page);
|
2023-04-17 08:56:51 +00:00
|
|
|
|
|
|
|
await Promise.all([page.mouse.click(0, 5), page.mouse.click(6, 10)]);
|
|
|
|
|
|
|
|
const data = await page.evaluate(() => {
|
|
|
|
return (window as unknown as {clicks: ClickData[]}).clicks;
|
|
|
|
});
|
|
|
|
const commonAttrs = {
|
|
|
|
isTrusted: true,
|
|
|
|
detail: 1,
|
|
|
|
clientY: 5,
|
|
|
|
clientX: 0,
|
|
|
|
button: 0,
|
|
|
|
};
|
|
|
|
expect(data.splice(0, 3)).toMatchObject({
|
|
|
|
0: {
|
|
|
|
type: 'mousedown',
|
|
|
|
buttons: 1,
|
|
|
|
...commonAttrs,
|
|
|
|
},
|
|
|
|
1: {
|
|
|
|
type: 'mouseup',
|
|
|
|
buttons: 0,
|
|
|
|
...commonAttrs,
|
|
|
|
},
|
|
|
|
2: {
|
|
|
|
type: 'click',
|
|
|
|
buttons: 0,
|
|
|
|
...commonAttrs,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
Object.assign(commonAttrs, {
|
|
|
|
clientX: 6,
|
|
|
|
clientY: 10,
|
|
|
|
});
|
|
|
|
expect(data).toMatchObject({
|
|
|
|
0: {
|
|
|
|
type: 'mousedown',
|
|
|
|
buttons: 1,
|
|
|
|
...commonAttrs,
|
|
|
|
},
|
|
|
|
1: {
|
|
|
|
type: 'mouseup',
|
|
|
|
buttons: 0,
|
|
|
|
...commonAttrs,
|
|
|
|
},
|
|
|
|
2: {
|
|
|
|
type: 'click',
|
|
|
|
buttons: 0,
|
|
|
|
...commonAttrs,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
2023-06-12 17:59:28 +00:00
|
|
|
|
|
|
|
it('should reset properly', async () => {
|
2023-09-20 15:58:02 +00:00
|
|
|
const {page, server, isChrome} = await getTestState();
|
2023-06-12 17:59:28 +00:00
|
|
|
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
|
|
|
|
await page.mouse.move(5, 5);
|
|
|
|
await Promise.all([
|
|
|
|
page.mouse.down({button: MouseButton.Left}),
|
|
|
|
page.mouse.down({button: MouseButton.Middle}),
|
|
|
|
page.mouse.down({button: MouseButton.Right}),
|
|
|
|
]);
|
|
|
|
|
|
|
|
await addMouseDataListeners(page, {includeMove: true});
|
|
|
|
await page.mouse.reset();
|
|
|
|
|
|
|
|
const data = await page.evaluate(() => {
|
|
|
|
return (window as unknown as {clicks: ClickData[]}).clicks;
|
|
|
|
});
|
|
|
|
const commonAttrs = {
|
|
|
|
isTrusted: true,
|
|
|
|
clientY: 5,
|
|
|
|
clientX: 5,
|
|
|
|
};
|
2023-09-20 15:58:02 +00:00
|
|
|
|
|
|
|
expect(data.slice(0, 2)).toMatchObject([
|
2023-06-12 17:59:28 +00:00
|
|
|
{
|
|
|
|
...commonAttrs,
|
2023-09-20 15:58:02 +00:00
|
|
|
button: 2,
|
|
|
|
buttons: 5,
|
2023-06-12 17:59:28 +00:00
|
|
|
detail: 1,
|
|
|
|
type: 'mouseup',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
...commonAttrs,
|
2023-09-20 15:58:02 +00:00
|
|
|
button: 2,
|
|
|
|
buttons: 5,
|
2023-06-12 17:59:28 +00:00
|
|
|
detail: 1,
|
2023-09-20 15:58:02 +00:00
|
|
|
type: 'auxclick',
|
2023-06-12 17:59:28 +00:00
|
|
|
},
|
2023-09-20 15:58:02 +00:00
|
|
|
]);
|
|
|
|
// TODO(crbug/1485040): This should align with the firefox implementation.
|
|
|
|
if (isChrome) {
|
|
|
|
expect(data.slice(2)).toMatchObject([
|
|
|
|
{
|
|
|
|
...commonAttrs,
|
|
|
|
button: 1,
|
|
|
|
buttons: 1,
|
|
|
|
detail: 0,
|
|
|
|
type: 'mouseup',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
...commonAttrs,
|
|
|
|
button: 0,
|
|
|
|
buttons: 0,
|
|
|
|
detail: 0,
|
|
|
|
type: 'mouseup',
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
expect(data.slice(2)).toMatchObject([
|
2023-06-12 17:59:28 +00:00
|
|
|
{
|
|
|
|
...commonAttrs,
|
|
|
|
button: 1,
|
2023-09-20 15:58:02 +00:00
|
|
|
buttons: 1,
|
|
|
|
detail: 1,
|
2023-06-12 17:59:28 +00:00
|
|
|
type: 'mouseup',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
...commonAttrs,
|
2023-09-20 15:58:02 +00:00
|
|
|
button: 1,
|
|
|
|
buttons: 1,
|
|
|
|
detail: 1,
|
|
|
|
type: 'auxclick',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
...commonAttrs,
|
|
|
|
button: 0,
|
2023-06-12 17:59:28 +00:00
|
|
|
buttons: 0,
|
2023-09-20 15:58:02 +00:00
|
|
|
detail: 1,
|
2023-06-12 17:59:28 +00:00
|
|
|
type: 'mouseup',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
...commonAttrs,
|
|
|
|
button: 0,
|
|
|
|
buttons: 0,
|
2023-09-20 15:58:02 +00:00
|
|
|
detail: 1,
|
|
|
|
type: 'click',
|
2023-06-12 17:59:28 +00:00
|
|
|
},
|
|
|
|
]);
|
|
|
|
});
|
2023-12-08 09:38:49 +00:00
|
|
|
|
|
|
|
it('should evaluate before mouse event', async () => {
|
|
|
|
const {page, server} = await getTestState();
|
|
|
|
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await page.goto(server.CROSS_PROCESS_PREFIX + '/input/button.html');
|
|
|
|
|
|
|
|
using button = await page.waitForSelector('button');
|
|
|
|
|
|
|
|
const point = await button!.clickablePoint();
|
|
|
|
|
|
|
|
const result = page.evaluate(() => {
|
|
|
|
return new Promise(resolve => {
|
|
|
|
document
|
|
|
|
.querySelector('button')
|
|
|
|
?.addEventListener('click', resolve, {once: true});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
await page.mouse.click(point?.x, point?.y);
|
|
|
|
await result;
|
|
|
|
});
|
2020-04-09 05:56:25 +00:00
|
|
|
});
|