mirror of
				https://github.com/microsoft/playwright.git
				synced 2025-06-26 21:40:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			458 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			458 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /**
 | |
|  * 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 type { ElementHandle, Route } from 'playwright-core';
 | |
| import { test as it, expect } from './pageTest';
 | |
| import { attachFrame } from '../config/utils';
 | |
| 
 | |
| it.skip(({ browserName, browserMajorVersion }) => browserName === 'chromium' && browserMajorVersion < 91);
 | |
| it.fixme(({ headless, isLinux }) => isLinux && !headless, 'Stray mouse events on Linux headed mess up the tests.');
 | |
| it.fixme(({ headless, isWindows, browserName }) => isWindows && !headless && browserName === 'webkit', 'WebKit win also send stray mouse events.');
 | |
| 
 | |
| it.describe('Drag and drop', () => {
 | |
|   it.skip(({ isAndroid }) => isAndroid, 'No drag&drop on Android.');
 | |
| 
 | |
|   it('should work @smoke', async ({ page, server }) => {
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     await page.hover('#source');
 | |
|     await page.mouse.down();
 | |
|     await page.hover('#target');
 | |
|     await page.mouse.up();
 | |
|     expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true); // could not find source in target
 | |
|   });
 | |
| 
 | |
|   it('should send the right events', async ({ server, page, browserName }) => {
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     const events = await trackEvents(await page.$('body'));
 | |
|     await page.hover('#source');
 | |
|     await page.mouse.down();
 | |
|     await page.hover('#target');
 | |
|     await page.mouse.up();
 | |
|     expect(await events.jsonValue()).toEqual([
 | |
|       'mousemove at 120;86',
 | |
|       'mousedown at 120;86',
 | |
|       browserName === 'firefox' ? 'dragstart at 120;86' : 'mousemove at 240;350',
 | |
|       browserName === 'firefox' ? 'mousemove at 240;350' : 'dragstart at 120;86',
 | |
|       'dragenter at 240;350',
 | |
|       'dragover at 240;350',
 | |
|       'drop at 240;350',
 | |
|       'dragend',
 | |
|     ]);
 | |
|   });
 | |
| 
 | |
|   it('should not send dragover on the first mousemove', async ({ server, page, browserName }) => {
 | |
|     it.fixme(browserName !== 'chromium');
 | |
| 
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     const events = await trackEvents(await page.$('body'));
 | |
|     await page.hover('#source');
 | |
|     await page.mouse.down();
 | |
|     await page.hover('#target');
 | |
|     expect(await events.jsonValue()).toEqual([
 | |
|       'mousemove at 120;86',
 | |
|       'mousedown at 120;86',
 | |
|       browserName === 'firefox' ? 'dragstart at 120;86' : 'mousemove at 240;350',
 | |
|       browserName === 'firefox' ? 'mousemove at 240;350' : 'dragstart at 120;86',
 | |
|       'dragenter at 240;350',
 | |
|     ]);
 | |
|   });
 | |
| 
 | |
|   it('should work inside iframe', async ({ page, server, browserName, isElectron, isWindows }) => {
 | |
|     it.fixme(isElectron && isWindows, 'Fails on the bots');
 | |
|     await page.goto(server.EMPTY_PAGE);
 | |
|     const frame = await attachFrame(page, 'myframe', server.PREFIX + '/drag-n-drop.html');
 | |
|     await page.$eval('iframe', iframe => {
 | |
|       iframe.style.width = '500px';
 | |
|       iframe.style.height = '600px';
 | |
|       iframe.style.marginLeft = '80px';
 | |
|       iframe.style.marginTop = '60px';
 | |
|     });
 | |
|     const pageEvents = await trackEvents(await page.$('body'));
 | |
|     const frameEvents = await trackEvents(await frame.$('body'));
 | |
|     await frame.hover('#source');
 | |
|     await page.mouse.down();
 | |
|     await frame.hover('#target');
 | |
|     await page.mouse.up();
 | |
|     expect(await frame.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true); // could not find source in target
 | |
|     expect(await frameEvents.jsonValue()).toEqual([
 | |
|       'mousemove at 120;86',
 | |
|       'mousedown at 120;86',
 | |
|       browserName === 'firefox' ? 'dragstart at 120;86' : 'mousemove at 240;350',
 | |
|       browserName === 'firefox' ? 'mousemove at 240;350' : 'dragstart at 120;86',
 | |
|       'dragenter at 240;350',
 | |
|       'dragover at 240;350',
 | |
|       'drop at 240;350',
 | |
|       'dragend',
 | |
|     ]);
 | |
|     expect(await pageEvents.jsonValue()).toEqual([]);
 | |
|   });
 | |
| 
 | |
|   it('should cancel on escape', async ({ server, page, browserName }) => {
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     const events = await trackEvents(await page.$('body'));
 | |
|     await page.hover('#source');
 | |
|     await page.mouse.down();
 | |
|     await page.hover('#target');
 | |
|     await page.keyboard.press('Escape');
 | |
|     await page.mouse.up();
 | |
|     expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(false); // found source in target
 | |
|     expect(await events.jsonValue()).toEqual([
 | |
|       'mousemove at 120;86',
 | |
|       'mousedown at 120;86',
 | |
|       browserName === 'firefox' ? 'dragstart at 120;86' : 'mousemove at 240;350',
 | |
|       browserName === 'firefox' ? 'mousemove at 240;350' : 'dragstart at 120;86',
 | |
|       'dragenter at 240;350',
 | |
|       browserName === 'chromium' ? null : 'dragover at 240;350',
 | |
|       'dragend',
 | |
|       'mouseup at 240;350',
 | |
|     ].filter(Boolean));
 | |
|   });
 | |
| 
 | |
|   it.describe('iframe', () => {
 | |
|     it.fixme(true, 'implement dragging with iframes');
 | |
| 
 | |
|     it('should drag into an iframe', async ({ server, page, browserName }) => {
 | |
|       await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|       const frame = await attachFrame(page, 'oopif', server.PREFIX + '/drag-n-drop.html');
 | |
|       await page.$eval('iframe', iframe => {
 | |
|         iframe.style.width = '500px';
 | |
|         iframe.style.height = '600px';
 | |
|         iframe.style.marginLeft = '500px';
 | |
|         iframe.style.marginTop = '60px';
 | |
|       });
 | |
|       await page.waitForTimeout(5000);
 | |
|       const pageEvents = await trackEvents(await page.$('body'));
 | |
|       const frameEvents = await trackEvents(await frame.$('body'));
 | |
|       await page.hover('#source');
 | |
|       await page.mouse.down();
 | |
|       await frame.hover('#target');
 | |
|       await page.mouse.up();
 | |
|       expect(await frame.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true); // could not find source in target
 | |
|       expect(await pageEvents.jsonValue()).toEqual([
 | |
|         'mousemove',
 | |
|         'mousedown',
 | |
|         browserName === 'firefox' ? 'dragstart' : 'mousemove',
 | |
|         browserName === 'firefox' ? 'mousemove' : 'dragstart',
 | |
|       ]);
 | |
|       expect(await frameEvents.jsonValue()).toEqual([
 | |
|         'dragenter',
 | |
|         'dragover',
 | |
|         'drop',
 | |
|       ]);
 | |
|     });
 | |
| 
 | |
|     it('should drag out of an iframe', async ({ server, page }) => {
 | |
|       await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|       const frame = await attachFrame(page, 'oopif', server.PREFIX + '/drag-n-drop.html');
 | |
|       const pageEvents = await trackEvents(await page.$('body'));
 | |
|       const frameEvents = await trackEvents(await frame.$('body'));
 | |
|       await frame.hover('#source');
 | |
|       await page.mouse.down();
 | |
|       await page.hover('#target');
 | |
|       await page.mouse.up();
 | |
|       expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true); // could not find source in target
 | |
|       expect(await frameEvents.jsonValue()).toEqual([
 | |
|         'mousemove',
 | |
|         'mousedown',
 | |
|         'dragstart',
 | |
|         'dragend',
 | |
|       ]);
 | |
|       expect(await pageEvents.jsonValue()).toEqual([
 | |
|         'dragenter',
 | |
|         'dragover',
 | |
|         'drop',
 | |
|       ]);
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   it('should respect the drop effect', async ({ page, browserName, isLinux, isMac, headless, trace }) => {
 | |
|     it.fixme(browserName === 'webkit' && !isLinux, 'WebKit doesn\'t handle the drop effect correctly outside of linux.');
 | |
|     it.fixme(browserName === 'webkit' && isLinux && !headless, 'https://github.com/microsoft/playwright/issues/21646');
 | |
|     it.fixme(browserName === 'chromium' && !isMac && !headless, 'https://github.com/microsoft/playwright/issues/21646');
 | |
|     it.slow(trace === 'on');
 | |
| 
 | |
|     expect(await testIfDropped('copy', 'copy')).toBe(true);
 | |
|     expect(await testIfDropped('copy', 'move')).toBe(false);
 | |
|     expect(await testIfDropped('all', 'link')).toBe(true);
 | |
|     expect(await testIfDropped('all', 'none')).toBe(false);
 | |
| 
 | |
|     expect(await testIfDropped('copyMove', 'copy')).toBe(true);
 | |
|     expect(await testIfDropped('copyLink', 'copy')).toBe(true);
 | |
|     expect(await testIfDropped('linkMove', 'copy')).toBe(false);
 | |
| 
 | |
|     expect(await testIfDropped('copyMove', 'link')).toBe(false);
 | |
|     expect(await testIfDropped('copyLink', 'link')).toBe(true);
 | |
|     expect(await testIfDropped('linkMove', 'link')).toBe(true);
 | |
| 
 | |
|     expect(await testIfDropped('copyMove', 'move')).toBe(true);
 | |
|     expect(await testIfDropped('copyLink', 'move')).toBe(false);
 | |
|     expect(await testIfDropped('linkMove', 'move')).toBe(true);
 | |
| 
 | |
|     expect(await testIfDropped('uninitialized', 'copy')).toBe(true);
 | |
| 
 | |
|     async function testIfDropped(effectAllowed: string, dropEffect: string) {
 | |
|       await page.setContent(`
 | |
|         <div draggable="true">drag target</div>
 | |
|         <drop-target>this is the drop target</drop-target>
 | |
|       `);
 | |
|       await page.evaluate(({ effectAllowed, dropEffect }) => {
 | |
|         window['dropped'] = false;
 | |
| 
 | |
|         document.querySelector('div').addEventListener('dragstart', event => {
 | |
|           event.dataTransfer.effectAllowed = effectAllowed as any;
 | |
|           event.dataTransfer.setData('text/plain', 'drag data');
 | |
|         });
 | |
| 
 | |
|         const dropTarget: HTMLElement = document.querySelector('drop-target');
 | |
|         dropTarget.addEventListener('dragover', event => {
 | |
|           event.dataTransfer.dropEffect = dropEffect as any;
 | |
|           event.preventDefault();
 | |
|         });
 | |
|         dropTarget.addEventListener('drop', event => {
 | |
|           window['dropped'] = true;
 | |
|         });
 | |
|       }, { effectAllowed, dropEffect });
 | |
|       await page.hover('div');
 | |
|       await page.mouse.down();
 | |
|       await page.hover('drop-target');
 | |
|       await page.mouse.up();
 | |
|       return await page.evaluate('dropped');
 | |
|     }
 | |
|   });
 | |
|   it('should work if the drag is canceled', async ({ page, server }) => {
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     await page.evaluate(() => {
 | |
|       document.body.addEventListener('dragstart', event => {
 | |
|         event.preventDefault();
 | |
|       }, false);
 | |
|     });
 | |
|     await page.hover('#source');
 | |
|     await page.mouse.down();
 | |
|     await page.hover('#target');
 | |
|     await page.mouse.up();
 | |
|     expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(false);
 | |
|   });
 | |
| 
 | |
|   it('should work if the drag event is captured but not canceled', async ({ page, server }) => {
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     await page.evaluate(() => {
 | |
|       document.body.addEventListener('dragstart', event => {
 | |
|         event.stopImmediatePropagation();
 | |
|       }, false);
 | |
|     });
 | |
|     await page.hover('#source');
 | |
|     await page.mouse.down();
 | |
|     await page.hover('#target');
 | |
|     await page.mouse.up();
 | |
|     expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true);
 | |
|   });
 | |
| 
 | |
|   it('should be able to drag the mouse in a frame', async ({ page, server }) => {
 | |
|     await page.goto(server.PREFIX + '/frames/one-frame.html');
 | |
|     const eventsHandle = await trackEvents(await page.frames()[1].$('html'));
 | |
|     await page.mouse.move(30, 30);
 | |
|     await page.mouse.down();
 | |
|     await page.mouse.move(60, 60);
 | |
|     await page.mouse.up();
 | |
|     expect(await eventsHandle.jsonValue()).toEqual(['mousemove at 20;20', 'mousedown at 20;20', 'mousemove at 50;50', 'mouseup at 50;50']);
 | |
|   });
 | |
| 
 | |
|   it('should work if a frame is stalled', async ({ page, server, toImpl }) => {
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     let madeRequest;
 | |
|     const routePromise = new Promise<Route>(x => madeRequest = x);
 | |
|     await page.route('**/empty.html', async (route, request) => {
 | |
|       madeRequest(route);
 | |
|     });
 | |
|     attachFrame(page, 'frame', server.EMPTY_PAGE).catch(() => {});
 | |
|     const route = await routePromise;
 | |
|     await page.hover('#source');
 | |
|     await page.mouse.down();
 | |
|     await page.hover('#target');
 | |
|     await page.mouse.up();
 | |
|     await route.abort();
 | |
|     expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true); // could not find source in target
 | |
|   });
 | |
| 
 | |
|   it('should work with the helper method', async ({ page, server }) => {
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     await page.dragAndDrop('#source', '#target');
 | |
|     expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true); // could not find source in target
 | |
|   });
 | |
| 
 | |
|   it('should allow specifying the position', async ({ page, server }) => {
 | |
|     await page.setContent(`
 | |
|       <div style="width:100px;height:100px;background:red;" id="red">
 | |
|       </div>
 | |
|       <div style="width:100px;height:100px;background:blue;" id="blue">
 | |
|       </div>
 | |
|     `);
 | |
|     const eventsHandle = await page.evaluateHandle(() => {
 | |
|       const events = [];
 | |
|       document.getElementById('red').addEventListener('mousedown', event => {
 | |
|         events.push({
 | |
|           type: 'mousedown',
 | |
|           x: event.offsetX,
 | |
|           y: event.offsetY,
 | |
|         });
 | |
|       });
 | |
|       document.getElementById('blue').addEventListener('mouseup', event => {
 | |
|         events.push({
 | |
|           type: 'mouseup',
 | |
|           x: event.offsetX,
 | |
|           y: event.offsetY,
 | |
|         });
 | |
|       });
 | |
|       return events;
 | |
|     });
 | |
|     await page.dragAndDrop('#red', '#blue', {
 | |
|       sourcePosition: { x: 34, y: 7 },
 | |
|       targetPosition: { x: 10, y: 20 },
 | |
|     });
 | |
|     expect(await eventsHandle.jsonValue()).toEqual([
 | |
|       { type: 'mousedown', x: 34, y: 7 },
 | |
|       { type: 'mouseup', x: 10, y: 20 },
 | |
|     ]);
 | |
|   });
 | |
| 
 | |
|   it('should work with locators', async ({ page, server }) => {
 | |
|     await page.goto(server.PREFIX + '/drag-n-drop.html');
 | |
|     await page.locator('#source').dragTo(page.locator('#target'));
 | |
|     expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true); // could not find source in target
 | |
|   });
 | |
| });
 | |
| 
 | |
| it('should work if not doing a drag', async ({ page, isLinux, headless }) => {
 | |
|   const eventsHandle = await trackEvents(await page.$('html'));
 | |
|   await page.mouse.move(50, 50);
 | |
|   await page.mouse.down();
 | |
|   await page.mouse.move(100, 100);
 | |
|   await page.mouse.up();
 | |
|   expect(await eventsHandle.jsonValue()).toEqual(['mousemove at 50;50', 'mousedown at 50;50', 'mousemove at 100;100', 'mouseup at 100;100']);
 | |
| });
 | |
| 
 | |
| it('should report event.buttons', async ({ page, browserName }) => {
 | |
|   const logsHandle = await page.evaluateHandle(async () => {
 | |
|     const div = document.createElement('div');
 | |
|     document.body.appendChild(div);
 | |
|     div.style.width = '200px';
 | |
|     div.style.height = '200px';
 | |
|     div.style.backgroundColor = 'blue';
 | |
|     div.addEventListener('mousedown', onEvent);
 | |
|     div.addEventListener('mousemove', onEvent, { passive: false });
 | |
|     div.addEventListener('mouseup', onEvent);
 | |
|     const logs = [];
 | |
|     function onEvent(event) {
 | |
|       logs.push({ type: event.type, buttons: event.buttons });
 | |
|     }
 | |
|     await new Promise(window.builtinRequestAnimationFrame);
 | |
|     return logs;
 | |
|   });
 | |
|   await page.mouse.move(20, 20);
 | |
|   await page.mouse.down();
 | |
|   await page.mouse.move(40, 40);
 | |
|   await page.mouse.up();
 | |
|   const logs = await logsHandle.jsonValue();
 | |
|   expect(logs).toEqual([
 | |
|     { type: 'mousemove', buttons: 0 },
 | |
|     { type: 'mousedown', buttons: 1 },
 | |
|     { type: 'mousemove', buttons: 1 },
 | |
|     { type: 'mouseup', buttons: 0 },
 | |
|   ]);
 | |
| });
 | |
| 
 | |
| async function trackEvents(target: ElementHandle) {
 | |
|   const eventsHandle = await target.evaluateHandle(target => {
 | |
|     const events: string[] = [];
 | |
|     for (const event of [
 | |
|       'mousedown', 'mousemove', 'mouseup',
 | |
|       'dragstart', 'dragend', 'dragover', 'dragenter', 'dragleave', 'dragexit',
 | |
|       'drop'
 | |
|     ]) {
 | |
|       target.addEventListener(event, (e: PointerEvent) => {
 | |
|         // Browsers are all over the place with dragend position.
 | |
|         if (event === 'dragend')
 | |
|           events.push('dragend');
 | |
|         else
 | |
|           events.push(`${event} at ${e.clientX};${e.clientY}`);
 | |
|       }, false);
 | |
|     }
 | |
|     return events;
 | |
|   });
 | |
|   return eventsHandle;
 | |
| }
 | |
| 
 | |
| it('should handle custom dataTransfer', async ({ page, browserName, isWindows }) => {
 | |
|   it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/18013' });
 | |
|   it.fixme(browserName === 'webkit' && isWindows);
 | |
|   await page.setContent(`<button draggable="true">Draggable</button>`);
 | |
| 
 | |
|   const resultPromise = page.evaluate(() =>
 | |
|     new Promise(resolve => {
 | |
|       document.addEventListener('dragstart', event => {
 | |
|         event.dataTransfer!.setData('custom-type', 'Hello World');
 | |
|       }, false);
 | |
| 
 | |
|       document.addEventListener('dragenter', event => {
 | |
|         event.preventDefault();
 | |
|       }, false);
 | |
|       document.addEventListener('dragover', event => {
 | |
|         event.preventDefault();
 | |
|       }, false);
 | |
| 
 | |
|       document.addEventListener('drop', event => {
 | |
|         event.preventDefault();
 | |
|         resolve({
 | |
|           types: event.dataTransfer!.types,
 | |
|           data: event.dataTransfer!.getData('custom-type'),
 | |
|         });
 | |
|       }, false);
 | |
|     })
 | |
|   );
 | |
| 
 | |
|   await page.hover('[draggable="true"]');
 | |
|   await page.mouse.down();
 | |
|   await page.mouse.move(100, 100);
 | |
|   await page.mouse.up();
 | |
| 
 | |
|   await expect(resultPromise).resolves.toEqual({
 | |
|     types: ['custom-type'],
 | |
|     data: 'Hello World',
 | |
|   });
 | |
| });
 | |
| 
 | |
| it('what happens when dragging element is destroyed', async ({ page, browserName }) => {
 | |
|   it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/21621' });
 | |
| 
 | |
|   await page.setContent(`
 | |
|     <button draggable="true">Draggable</button>
 | |
|     <div id=target>drop here</div>
 | |
|   `);
 | |
| 
 | |
|   await page.evaluate(() => {
 | |
|     document.querySelector('#target').addEventListener('dragover', event => {
 | |
|       document.querySelector('button')?.remove();
 | |
|     }, false);
 | |
| 
 | |
|     document.querySelector('#target').addEventListener('drop', event => {
 | |
|       document.querySelector('#target').textContent = 'dropped';
 | |
|     }, false);
 | |
|   });
 | |
| 
 | |
|   await page.locator('button').dragTo(page.locator('div'));
 | |
|   await expect(page.locator('div')).toHaveText('drop here');
 | |
| });
 | 
