/** * Copyright (c) Microsoft Corporation. * * 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 { contextTest as it, expect } from './config/browserTest'; it('should block all events when hit target is wrong', async ({ page, server }) => { await page.goto(server.PREFIX + '/input/button.html'); await page.evaluate(() => { const blocker = document.createElement('div'); blocker.style.position = 'absolute'; blocker.style.width = '400px'; blocker.style.height = '400px'; blocker.style.left = '0'; blocker.style.top = '0'; document.body.appendChild(blocker); const allEvents = []; (window as any).allEvents = allEvents; for (const name of ['mousedown', 'mouseup', 'click', 'dblclick', 'auxclick', 'contextmenu', 'pointerdown', 'pointerup']) { window.addEventListener(name, e => allEvents.push(e.type)); blocker.addEventListener(name, e => allEvents.push(e.type)); } }); const error = await page.click('button', { timeout: 1000 }).catch(e => e); expect(error.message).toContain('page.click: Timeout 1000ms exceeded.'); // Give it some time, just in case. await page.waitForTimeout(1000); const allEvents = await page.evaluate(() => (window as any).allEvents); expect(allEvents).toEqual([]); }); it('should block click when mousedown succeeds but mouseup fails', async ({ page, server }) => { await page.goto(server.PREFIX + '/input/button.html'); await page.$eval('button', button => { button.addEventListener('mousedown', () => { button.style.marginLeft = '100px'; }); const allEvents = []; (window as any).allEvents = allEvents; for (const name of ['mousedown', 'mouseup', 'click', 'dblclick', 'auxclick', 'contextmenu', 'pointerdown', 'pointerup']) button.addEventListener(name, e => allEvents.push(e.type)); }); await page.click('button'); expect(await page.evaluate('result')).toBe('Clicked'); const allEvents = await page.evaluate(() => (window as any).allEvents); expect(allEvents).toEqual([ // First attempt failed. 'pointerdown', 'mousedown', // Second attempt succeeded. 'pointerdown', 'mousedown', 'pointerup', 'mouseup', 'click', ]); }); it('should click when element detaches in mousedown', async ({ page, server }) => { await page.goto(server.PREFIX + '/input/button.html'); await page.$eval('button', button => { button.addEventListener('mousedown', () => { (window as any).result = 'Mousedown'; button.remove(); }); }); await page.click('button', { timeout: 1000 }); expect(await page.evaluate('result')).toBe('Mousedown'); }); it('should not block programmatic events', async ({ page, server }) => { await page.goto(server.PREFIX + '/input/button.html'); await page.$eval('button', button => { button.addEventListener('mousedown', () => { button.style.marginLeft = '100px'; button.dispatchEvent(new MouseEvent('click')); }); const allEvents = []; (window as any).allEvents = allEvents; button.addEventListener('click', e => { if (!e.isTrusted) allEvents.push(e.type); }); }); await page.click('button'); expect(await page.evaluate('result')).toBe('Clicked'); const allEvents = await page.evaluate(() => (window as any).allEvents); // We should get one programmatic click on each attempt. expect(allEvents).toEqual([ 'click', 'click', ]); }); it('should click the button again after document.write', async ({ page, server }) => { await page.goto(server.PREFIX + '/input/button.html'); await page.click('button'); expect(await page.evaluate('result')).toBe('Clicked'); await page.evaluate(() => { document.open(); document.write(''); document.close(); }); await page.click('button'); expect(await page.evaluate('result2')).toBe(true); });