playwright/tests/page/page-dispatchevent.spec.ts
2024-05-31 14:44:26 -07:00

232 lines
10 KiB
TypeScript

/**
* Copyright Microsoft Corporation. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { test as it, expect } from './pageTest';
it('should dispatch click event @smoke', async ({ page, server }) => {
await page.goto(server.PREFIX + '/input/button.html');
await page.dispatchEvent('button', 'click');
expect(await page.evaluate(() => window['result'])).toBe('Clicked');
});
it('should dispatch click event properties', async ({ page, server }) => {
await page.goto(server.PREFIX + '/input/button.html');
await page.dispatchEvent('button', 'click');
expect(await page.evaluate('bubbles')).toBeTruthy();
expect(await page.evaluate('cancelable')).toBeTruthy();
expect(await page.evaluate('composed')).toBeTruthy();
});
it('should dispatch click svg', async ({ page }) => {
await page.setContent(`
<svg height="100" width="100">
<circle onclick="javascript:window.__CLICKED=42" cx="50" cy="50" r="40" stroke="black" stroke-width="3" fill="red" />
</svg>
`);
await page.dispatchEvent('circle', 'click');
expect(await page.evaluate(() => window['__CLICKED'])).toBe(42);
});
it('should dispatch click on a span with an inline element inside', async ({ page }) => {
await page.setContent(`
<style>
span::before {
content: 'q';
}
</style>
<span onclick='javascript:window.CLICKED=42'></span>
`);
await page.dispatchEvent('span', 'click');
expect(await page.evaluate(() => window['CLICKED'])).toBe(42);
});
it('should dispatch click after navigation ', async ({ page, server }) => {
await page.goto(server.PREFIX + '/input/button.html');
await page.dispatchEvent('button', 'click');
await page.goto(server.PREFIX + '/input/button.html');
await page.dispatchEvent('button', 'click');
expect(await page.evaluate(() => window['result'])).toBe('Clicked');
});
it('should dispatch click after a cross origin navigation ', async ({ page, server }) => {
await page.goto(server.PREFIX + '/input/button.html');
await page.dispatchEvent('button', 'click');
await page.goto(server.CROSS_PROCESS_PREFIX + '/input/button.html');
await page.dispatchEvent('button', 'click');
expect(await page.evaluate(() => window['result'])).toBe('Clicked');
});
it('should not fail when element is blocked on hover', async ({ page }) => {
await page.setContent(`<style>
container { display: block; position: relative; width: 200px; height: 50px; }
div, button { position: absolute; left: 0; top: 0; bottom: 0; right: 0; }
div { pointer-events: none; }
container:hover div { pointer-events: auto; background: red; }
</style>
<container>
<button onclick="window.clicked=true">Click me</button>
<div></div>
</container>`);
await page.dispatchEvent('button', 'click');
expect(await page.evaluate(() => window['clicked'])).toBeTruthy();
});
it('should dispatch click when node is added in shadow dom', async ({ page, server }) => {
await page.goto(server.EMPTY_PAGE);
const watchdog = page.dispatchEvent('span', 'click');
await page.evaluate(() => {
const div = document.createElement('div');
div.attachShadow({ mode: 'open' });
document.body.appendChild(div);
});
await page.waitForTimeout(100);
await page.evaluate(() => {
const span = document.createElement('span');
span.textContent = 'Hello from shadow';
span.addEventListener('click', () => window['clicked'] = true);
document.querySelector('div').shadowRoot.appendChild(span);
});
await watchdog;
expect(await page.evaluate(() => window['clicked'])).toBe(true);
});
it('should be atomic', async ({ playwright, page }) => {
const createDummySelector = () => ({
query(root, selector) {
const result = root.querySelector(selector);
if (result)
void Promise.resolve().then(() => result.onclick = '');
return result;
},
queryAll(root: HTMLElement, selector: string) {
const result = Array.from(root.querySelectorAll(selector));
for (const e of result)
void Promise.resolve().then(() => (e as HTMLElement).onclick = null);
return result;
}
});
await playwright.selectors.register('dispatchEvent', createDummySelector);
await page.setContent(`<div onclick="window._clicked=true">Hello</div>`);
await page.dispatchEvent('dispatchEvent=div', 'click');
expect(await page.evaluate(() => window['_clicked'])).toBe(true);
});
it('should dispatch drag drop events', async ({ page, server }) => {
await page.goto(server.PREFIX + '/drag-n-drop.html');
const dataTransfer = await page.evaluateHandle(() => new DataTransfer());
await page.dispatchEvent('#source', 'dragstart', { dataTransfer });
await page.dispatchEvent('#target', 'drop', { dataTransfer });
const source = await page.$('#source');
const target = await page.$('#target');
expect(await page.evaluate(({ source, target }) => {
return source.parentElement === target;
}, { source, target })).toBeTruthy();
});
it('should dispatch drag drop events via ElementHandles', async ({ page, server }) => {
await page.goto(server.PREFIX + '/drag-n-drop.html');
const dataTransfer = await page.evaluateHandle(() => new DataTransfer());
const source = await page.$('#source');
await source.dispatchEvent('dragstart', { dataTransfer });
const target = await page.$('#target');
await target.dispatchEvent('drop', { dataTransfer });
expect(await page.evaluate(({ source, target }) => {
return source.parentElement === target;
}, { source, target })).toBeTruthy();
});
it('should dispatch click event via ElementHandles', async ({ page, server }) => {
await page.goto(server.PREFIX + '/input/button.html');
const button = await page.$('button');
await button.dispatchEvent('click');
expect(await page.evaluate(() => window['result'])).toBe('Clicked');
});
it('should dispatch wheel event', async ({ page, server }) => {
it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/15562' });
await page.goto(server.PREFIX + '/input/scrollable.html');
const eventsHandle = await page.locator('body').evaluateHandle(e => {
const events = [];
e.addEventListener('wheel', event => {
events.push(event);
console.log(event);
});
return events;
});
await page.locator('body').dispatchEvent('wheel', { deltaX: 100, deltaY: 200 });
expect(await eventsHandle.evaluate(e => e.length)).toBe(1);
expect(await eventsHandle.evaluate(e => e[0] instanceof WheelEvent)).toBeTruthy();
expect(await eventsHandle.evaluate(e => ({ deltaX: e[0].deltaX, deltaY: e[0].deltaY }))).toEqual({ deltaX: 100, deltaY: 200 });
});
it('should dispatch device orientation event', async ({ page, server, isAndroid }) => {
it.skip(isAndroid, 'DeviceOrientationEvent is only available in a secure context. While Androids loopback is not treated as secure.');
await page.goto(server.PREFIX + '/device-orientation.html');
await page.locator('html').dispatchEvent('deviceorientation', { alpha: 10, beta: 20, gamma: 30 });
expect(await page.evaluate('result')).toBe('Oriented');
expect(await page.evaluate('alpha')).toBe(10);
expect(await page.evaluate('beta')).toBe(20);
expect(await page.evaluate('gamma')).toBe(30);
expect(await page.evaluate('absolute')).toBeFalsy();
});
it('should dispatch absolute device orientation event', async ({ page, server, isAndroid }) => {
it.skip(isAndroid, 'DeviceOrientationEvent is only available in a secure context. While Androids loopback is not treated as secure.');
await page.goto(server.PREFIX + '/device-orientation.html');
await page.locator('html').dispatchEvent('deviceorientationabsolute', { alpha: 10, beta: 20, gamma: 30, absolute: true });
expect(await page.evaluate('result')).toBe('Oriented');
expect(await page.evaluate('alpha')).toBe(10);
expect(await page.evaluate('beta')).toBe(20);
expect(await page.evaluate('gamma')).toBe(30);
expect(await page.evaluate('absolute')).toBeTruthy();
});
it('should dispatch device motion event', async ({ page, server, isAndroid }) => {
it.skip(isAndroid, 'DeviceOrientationEvent is only available in a secure context. While Androids loopback is not treated as secure.');
await page.goto(server.PREFIX + '/device-motion.html');
await page.locator('html').dispatchEvent('devicemotion', {
acceleration: { x: 10, y: 20, z: 30 },
accelerationIncludingGravity: { x: 15, y: 25, z: 35 },
rotationRate: { alpha: 5, beta: 10, gamma: 15 },
interval: 16,
});
expect(await page.evaluate('result')).toBe('Moved');
expect(await page.evaluate('acceleration.x')).toBe(10);
expect(await page.evaluate('acceleration.y')).toBe(20);
expect(await page.evaluate('acceleration.z')).toBe(30);
expect(await page.evaluate('accelerationIncludingGravity.x')).toBe(15);
expect(await page.evaluate('accelerationIncludingGravity.y')).toBe(25);
expect(await page.evaluate('accelerationIncludingGravity.z')).toBe(35);
expect(await page.evaluate('rotationRate.alpha')).toBe(5);
expect(await page.evaluate('rotationRate.beta')).toBe(10);
expect(await page.evaluate('rotationRate.gamma')).toBe(15);
expect(await page.evaluate('interval')).toBe(16);
});
it('should throw if argument is from different frame', async ({ page, server }) => {
it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/28690' });
await page.goto(server.PREFIX + '/frames/one-frame.html');
{
const dataTransfer = await page.frames()[1].evaluateHandle(() => new DataTransfer());
await page.frameLocator('iframe').locator('div').dispatchEvent('drop', { dataTransfer });
}
{
const dataTransfer = await page.evaluateHandle(() => new DataTransfer());
await expect(page.frameLocator('iframe').locator('div').dispatchEvent('drop', { dataTransfer }))
.rejects.toThrow('JSHandles can be evaluated only in the context they were created!');
}
});