| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-11 10:52:17 -04:00
										 |  |  | import type { ElementHandle, Route } from 'playwright-core'; | 
					
						
							| 
									
										
										
										
											2021-05-06 07:08:22 -07:00
										 |  |  | import { test as it, expect } from './pageTest'; | 
					
						
							| 
									
										
										
										
											2021-05-05 19:10:28 -07:00
										 |  |  | import { attachFrame } from '../config/utils'; | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 10:20:57 -07:00
										 |  |  | 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.'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  | it.describe('Drag and drop', () => { | 
					
						
							| 
									
										
										
										
											2022-11-07 15:35:21 -08:00
										 |  |  |   it.skip(({ isAndroid }) => isAndroid, 'No drag&drop on Android.'); | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-10 19:42:52 +01:00
										 |  |  |   it('should work @smoke', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  |     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
 | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |   it('should send the right events', async ({ server, page, browserName }) => { | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  |     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(); | 
					
						
							| 
									
										
										
										
											2022-11-04 08:53:50 -07:00
										 |  |  |     expect(await events.jsonValue()).toEqual([ | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'mousemove at 120;86', | 
					
						
							|  |  |  |       'mousedown at 120;86', | 
					
						
							| 
									
										
										
										
											2022-11-04 08:53:50 -07:00
										 |  |  |       browserName === 'firefox' ? 'dragstart at 120;86' : 'mousemove at 240;350', | 
					
						
							|  |  |  |       browserName === 'firefox' ? 'mousemove at 240;350' : 'dragstart at 120;86', | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'dragenter at 240;350', | 
					
						
							|  |  |  |       'dragover at 240;350', | 
					
						
							|  |  |  |       'drop at 240;350', | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  |       'dragend', | 
					
						
							|  |  |  |     ]); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |   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'); | 
					
						
							| 
									
										
										
										
											2022-11-04 08:53:50 -07:00
										 |  |  |     expect(await events.jsonValue()).toEqual([ | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'mousemove at 120;86', | 
					
						
							|  |  |  |       'mousedown at 120;86', | 
					
						
							| 
									
										
										
										
											2022-11-04 08:53:50 -07:00
										 |  |  |       browserName === 'firefox' ? 'dragstart at 120;86' : 'mousemove at 240;350', | 
					
						
							|  |  |  |       browserName === 'firefox' ? 'mousemove at 240;350' : 'dragstart at 120;86', | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'dragenter at 240;350', | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |     ]); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 10:54:00 -08:00
										 |  |  |   it('should work inside iframe', async ({ page, server, browserName, isElectron, isWindows }) => { | 
					
						
							|  |  |  |     it.fixme(isElectron && isWindows, 'Fails on the bots'); | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |     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
 | 
					
						
							| 
									
										
										
										
											2022-11-04 08:53:50 -07:00
										 |  |  |     expect(await frameEvents.jsonValue()).toEqual([ | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'mousemove at 120;86', | 
					
						
							|  |  |  |       'mousedown at 120;86', | 
					
						
							| 
									
										
										
										
											2022-11-04 08:53:50 -07:00
										 |  |  |       browserName === 'firefox' ? 'dragstart at 120;86' : 'mousemove at 240;350', | 
					
						
							|  |  |  |       browserName === 'firefox' ? 'mousemove at 240;350' : 'dragstart at 120;86', | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'dragenter at 240;350', | 
					
						
							|  |  |  |       'dragover at 240;350', | 
					
						
							|  |  |  |       'drop at 240;350', | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |       'dragend', | 
					
						
							|  |  |  |     ]); | 
					
						
							|  |  |  |     expect(await pageEvents.jsonValue()).toEqual([]); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |   it('should cancel on escape', async ({ server, page, browserName }) => { | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  |     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
 | 
					
						
							| 
									
										
										
										
											2022-11-04 08:53:50 -07:00
										 |  |  |     expect(await events.jsonValue()).toEqual([ | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'mousemove at 120;86', | 
					
						
							|  |  |  |       'mousedown at 120;86', | 
					
						
							| 
									
										
										
										
											2022-11-04 08:53:50 -07:00
										 |  |  |       browserName === 'firefox' ? 'dragstart at 120;86' : 'mousemove at 240;350', | 
					
						
							|  |  |  |       browserName === 'firefox' ? 'mousemove at 240;350' : 'dragstart at 120;86', | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'dragenter at 240;350', | 
					
						
							|  |  |  |       browserName === 'chromium' ? null : 'dragover at 240;350', | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  |       'dragend', | 
					
						
							| 
									
										
										
										
											2022-10-03 12:26:36 -07:00
										 |  |  |       'mouseup at 240;350', | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |     ].filter(Boolean)); | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  |   it.describe('iframe', () => { | 
					
						
							| 
									
										
										
										
											2021-05-20 09:51:09 -07:00
										 |  |  |     it.fixme(true, 'implement dragging with iframes'); | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |     it('should drag into an iframe', async ({ server, page, browserName }) => { | 
					
						
							| 
									
										
										
										
											2021-02-23 17:03:28 -08:00
										 |  |  |       await page.goto(server.PREFIX + '/drag-n-drop.html'); | 
					
						
							| 
									
										
										
										
											2022-08-18 20:12:33 +02:00
										 |  |  |       const frame = await attachFrame(page, 'oopif', server.PREFIX + '/drag-n-drop.html'); | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |       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); | 
					
						
							| 
									
										
										
										
											2021-02-23 17:03:28 -08:00
										 |  |  |       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', | 
					
						
							| 
									
										
										
										
											2021-05-13 10:22:23 -07:00
										 |  |  |         browserName === 'firefox' ? 'dragstart' : 'mousemove', | 
					
						
							|  |  |  |         browserName === 'firefox' ? 'mousemove' : 'dragstart', | 
					
						
							| 
									
										
										
										
											2021-02-23 17:03:28 -08:00
										 |  |  |       ]); | 
					
						
							|  |  |  |       expect(await frameEvents.jsonValue()).toEqual([ | 
					
						
							|  |  |  |         'dragenter', | 
					
						
							|  |  |  |         'dragover', | 
					
						
							|  |  |  |         'drop', | 
					
						
							|  |  |  |       ]); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |     it('should drag out of an iframe', async ({ server, page }) => { | 
					
						
							| 
									
										
										
										
											2021-02-23 17:03:28 -08:00
										 |  |  |       await page.goto(server.PREFIX + '/drag-n-drop.html'); | 
					
						
							| 
									
										
										
										
											2022-08-18 20:12:33 +02:00
										 |  |  |       const frame = await attachFrame(page, 'oopif', server.PREFIX + '/drag-n-drop.html'); | 
					
						
							| 
									
										
										
										
											2021-02-23 17:03:28 -08:00
										 |  |  |       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', | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |         'dragend', | 
					
						
							| 
									
										
										
										
											2021-02-23 17:03:28 -08:00
										 |  |  |       ]); | 
					
						
							|  |  |  |       expect(await pageEvents.jsonValue()).toEqual([ | 
					
						
							|  |  |  |         'dragenter', | 
					
						
							|  |  |  |         'dragover', | 
					
						
							|  |  |  |         'drop', | 
					
						
							|  |  |  |       ]); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-13 19:04:09 -07:00
										 |  |  |   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'); | 
					
						
							| 
									
										
										
										
											2021-10-26 12:45:53 -08:00
										 |  |  |     it.slow(trace === 'on'); | 
					
						
							| 
									
										
										
										
											2021-04-04 19:32:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-23 16:25:06 -08:00
										 |  |  |     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> | 
					
						
							|  |  |  |       `);
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |       await page.evaluate(({ effectAllowed, dropEffect }) => { | 
					
						
							| 
									
										
										
										
											2021-02-23 16:25:06 -08:00
										 |  |  |         window['dropped'] = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         document.querySelector('div').addEventListener('dragstart', event => { | 
					
						
							| 
									
										
										
										
											2021-04-30 13:26:13 -07:00
										 |  |  |           event.dataTransfer.effectAllowed = effectAllowed as any; | 
					
						
							| 
									
										
										
										
											2021-02-23 16:25:06 -08:00
										 |  |  |           event.dataTransfer.setData('text/plain', 'drag data'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const dropTarget: HTMLElement = document.querySelector('drop-target'); | 
					
						
							|  |  |  |         dropTarget.addEventListener('dragover', event => { | 
					
						
							| 
									
										
										
										
											2021-04-30 13:26:13 -07:00
										 |  |  |           event.dataTransfer.dropEffect = dropEffect as any; | 
					
						
							| 
									
										
										
										
											2021-02-23 16:25:06 -08:00
										 |  |  |           event.preventDefault(); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         dropTarget.addEventListener('drop', event => { | 
					
						
							|  |  |  |           window['dropped'] = true; | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |       }, { effectAllowed, dropEffect }); | 
					
						
							| 
									
										
										
										
											2021-02-23 16:25:06 -08:00
										 |  |  |       await page.hover('div'); | 
					
						
							|  |  |  |       await page.mouse.down(); | 
					
						
							|  |  |  |       await page.hover('drop-target'); | 
					
						
							|  |  |  |       await page.mouse.up(); | 
					
						
							|  |  |  |       return await page.evaluate('dropped'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |   it('should work if the drag is canceled', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |     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); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |   it('should work if the drag event is captured but not canceled', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |     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); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 10:20:57 -07:00
										 |  |  |   it('should be able to drag the mouse in a frame', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |     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(); | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |     expect(await eventsHandle.jsonValue()).toEqual(['mousemove at 20;20', 'mousedown at 20;20', 'mousemove at 50;50', 'mouseup at 50;50']); | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |   it('should work if a frame is stalled', async ({ page, server, toImpl }) => { | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |     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(); | 
					
						
							| 
									
										
										
										
											2023-06-02 21:59:12 +02:00
										 |  |  |     await route.abort(); | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |     expect(await page.$eval('#target', target => target.contains(document.querySelector('#source')))).toBe(true); // could not find source in target
 | 
					
						
							| 
									
										
										
										
											2021-06-23 11:08:35 +02:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-02-23 16:25:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |   it('should work with the helper method', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											2021-07-19 12:42:55 -05:00
										 |  |  |     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
 | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |   it('should allow specifying the position', async ({ page, server }) => { | 
					
						
							| 
									
										
										
										
											2021-08-03 13:12:34 -04:00
										 |  |  |     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', { | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |       sourcePosition: { x: 34, y: 7 }, | 
					
						
							|  |  |  |       targetPosition: { x: 10, y: 20 }, | 
					
						
							| 
									
										
										
										
											2021-08-03 13:12:34 -04:00
										 |  |  |     }); | 
					
						
							|  |  |  |     expect(await eventsHandle.jsonValue()).toEqual([ | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  |       { type: 'mousedown', x: 34, y: 7 }, | 
					
						
							|  |  |  |       { type: 'mouseup', x: 10, y: 20 }, | 
					
						
							| 
									
										
										
										
											2021-08-03 13:12:34 -04:00
										 |  |  |     ]); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-22 14:27:26 -05:00
										 |  |  |   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
 | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2020-11-25 03:29:20 -08:00
										 |  |  | }); | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-12 15:09:56 -07:00
										 |  |  | it('should work if not doing a drag', async ({ page, isLinux, headless }) => { | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |   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(); | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |   expect(await eventsHandle.jsonValue()).toEqual(['mousemove at 50;50', 'mousedown at 50;50', 'mousemove at 100;100', 'mouseup at 100;100']); | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 18:58:08 +02:00
										 |  |  | it('should report event.buttons', async ({ page, browserName }) => { | 
					
						
							| 
									
										
										
										
											2021-07-16 13:06:00 -05:00
										 |  |  |   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 }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-31 14:44:26 -07:00
										 |  |  |     await new Promise(window.builtinRequestAnimationFrame); | 
					
						
							| 
									
										
										
										
											2021-07-16 13:06:00 -05:00
										 |  |  |     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 }, | 
					
						
							|  |  |  |   ]); | 
					
						
							|  |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  | 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' | 
					
						
							| 
									
										
										
										
											2022-09-30 09:29:19 -07:00
										 |  |  |     ]) { | 
					
						
							|  |  |  |       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); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-07 10:27:34 -07:00
										 |  |  |     return events; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   return eventsHandle; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-02-01 17:18:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-02 10:26:47 -08:00
										 |  |  | it('should handle custom dataTransfer', async ({ page, browserName, isWindows }) => { | 
					
						
							| 
									
										
										
										
											2023-02-01 17:18:58 -08:00
										 |  |  |   it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/18013' }); | 
					
						
							| 
									
										
										
										
											2023-02-02 10:26:47 -08:00
										 |  |  |   it.fixme(browserName === 'webkit' && isWindows); | 
					
						
							| 
									
										
										
										
											2023-02-01 17:18:58 -08:00
										 |  |  |   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', | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
										
										
											2023-04-03 18:42:29 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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'); | 
					
						
							|  |  |  | }); |