| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Copyright 2018 Google Inc. All rights reserved. | 
					
						
							| 
									
										
										
										
											2019-12-10 13:21:51 -08:00
										 |  |  |  * Modifications copyright (c) Microsoft Corporation. | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-22 00:07:02 -07:00
										 |  |  | const {FFOX, CHROMIUM, WEBKIT, MAC, WIN} = require('./utils').testOptions(browserType); | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  | function dimensions() { | 
					
						
							|  |  |  |   const rect = document.querySelector('textarea').getBoundingClientRect(); | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     x: rect.left, | 
					
						
							|  |  |  |     y: rect.top, | 
					
						
							|  |  |  |     width: rect.width, | 
					
						
							|  |  |  |     height: rect.height | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  | describe('Mouse', function() { | 
					
						
							| 
									
										
										
										
											2020-04-22 00:07:02 -07:00
										 |  |  |   // Occasionally times out on FFOX on Windows: https://github.com/microsoft/playwright/pull/1911/checks?check_run_id=607149016
 | 
					
						
							|  |  |  |   it.fail(FFOX && WIN)('should click the document', async({page, server}) => { | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  |     await page.evaluate(() => { | 
					
						
							|  |  |  |       window.clickPromise = new Promise(resolve => { | 
					
						
							|  |  |  |         document.addEventListener('click', event => { | 
					
						
							|  |  |  |           resolve({ | 
					
						
							|  |  |  |             type: event.type, | 
					
						
							|  |  |  |             detail: event.detail, | 
					
						
							|  |  |  |             clientX: event.clientX, | 
					
						
							|  |  |  |             clientY: event.clientY, | 
					
						
							|  |  |  |             isTrusted: event.isTrusted, | 
					
						
							|  |  |  |             button: event.button | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  |           }); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  |     await page.mouse.click(50, 60); | 
					
						
							|  |  |  |     const event = await page.evaluate(() => window.clickPromise); | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   it('should select the text with mouse', async({page, server}) => { | 
					
						
							|  |  |  |     await page.goto(server.PREFIX + '/input/textarea.html'); | 
					
						
							|  |  |  |     await page.focus('textarea'); | 
					
						
							|  |  |  |     const text = 'This is the text that we are going to try to select. Let\'s see how it goes.'; | 
					
						
							|  |  |  |     await page.keyboard.type(text); | 
					
						
							|  |  |  |     // Firefox needs an extra frame here after typing or it will fail to set the scrollTop
 | 
					
						
							|  |  |  |     await page.evaluate(() => new Promise(requestAnimationFrame)); | 
					
						
							|  |  |  |     await page.evaluate(() => document.querySelector('textarea').scrollTop = 0); | 
					
						
							|  |  |  |     const {x, y} = await page.evaluate(dimensions); | 
					
						
							|  |  |  |     await page.mouse.move(x + 2,y + 2); | 
					
						
							|  |  |  |     await page.mouse.down(); | 
					
						
							|  |  |  |     await page.mouse.move(200,200); | 
					
						
							|  |  |  |     await page.mouse.up(); | 
					
						
							|  |  |  |     expect(await page.evaluate(() => { | 
					
						
							|  |  |  |       const textarea = document.querySelector('textarea'); | 
					
						
							|  |  |  |       return textarea.value.substring(textarea.selectionStart, textarea.selectionEnd); | 
					
						
							|  |  |  |     })).toBe(text); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   it('should trigger hover state', async({page, server}) => { | 
					
						
							|  |  |  |     await page.goto(server.PREFIX + '/input/scrollable.html'); | 
					
						
							|  |  |  |     await page.hover('#button-6'); | 
					
						
							|  |  |  |     expect(await page.evaluate(() => document.querySelector('button:hover').id)).toBe('button-6'); | 
					
						
							|  |  |  |     await page.hover('#button-2'); | 
					
						
							|  |  |  |     expect(await page.evaluate(() => document.querySelector('button:hover').id)).toBe('button-2'); | 
					
						
							|  |  |  |     await page.hover('#button-91'); | 
					
						
							|  |  |  |     expect(await page.evaluate(() => document.querySelector('button:hover').id)).toBe('button-91'); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   it('should trigger hover state with removed window.Node', async({page, server}) => { | 
					
						
							|  |  |  |     await page.goto(server.PREFIX + '/input/scrollable.html'); | 
					
						
							|  |  |  |     await page.evaluate(() => delete window.Node); | 
					
						
							|  |  |  |     await page.hover('#button-6'); | 
					
						
							|  |  |  |     expect(await page.evaluate(() => document.querySelector('button:hover').id)).toBe('button-6'); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   it('should set modifier keys on click', async({page, server}) => { | 
					
						
							|  |  |  |     await page.goto(server.PREFIX + '/input/scrollable.html'); | 
					
						
							|  |  |  |     await page.evaluate(() => document.querySelector('#button-3').addEventListener('mousedown', e => window.lastEvent = e, true)); | 
					
						
							|  |  |  |     const modifiers = {'Shift': 'shiftKey', 'Control': 'ctrlKey', 'Alt': 'altKey', 'Meta': 'metaKey'}; | 
					
						
							|  |  |  |     // In Firefox, the Meta modifier only exists on Mac
 | 
					
						
							|  |  |  |     if (FFOX && !MAC) | 
					
						
							|  |  |  |       delete modifiers['Meta']; | 
					
						
							|  |  |  |     for (const modifier in modifiers) { | 
					
						
							|  |  |  |       await page.keyboard.down(modifier); | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  |       await page.click('#button-3'); | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  |       if (!(await page.evaluate(mod => window.lastEvent[mod], modifiers[modifier]))) | 
					
						
							|  |  |  |         throw new Error(modifiers[modifier] + ' should be true'); | 
					
						
							|  |  |  |       await page.keyboard.up(modifier); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     await page.click('#button-3'); | 
					
						
							|  |  |  |     for (const modifier in modifiers) { | 
					
						
							|  |  |  |       if ((await page.evaluate(mod => window.lastEvent[mod], modifiers[modifier]))) | 
					
						
							|  |  |  |         throw new Error(modifiers[modifier] + ' should be false'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   it('should tween mouse movement', async({page, server}) => { | 
					
						
							|  |  |  |     // The test becomes flaky on WebKit without next line.
 | 
					
						
							|  |  |  |     if (WEBKIT) | 
					
						
							|  |  |  |       await page.evaluate(() => new Promise(requestAnimationFrame)); | 
					
						
							|  |  |  |     await page.mouse.move(100, 100); | 
					
						
							|  |  |  |     await page.evaluate(() => { | 
					
						
							|  |  |  |       window.result = []; | 
					
						
							|  |  |  |       document.addEventListener('mousemove', event => { | 
					
						
							|  |  |  |         window.result.push([event.clientX, event.clientY]); | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  |     await page.mouse.move(200, 300, {steps: 5}); | 
					
						
							|  |  |  |     expect(await page.evaluate('result')).toEqual([ | 
					
						
							|  |  |  |       [120, 140], | 
					
						
							|  |  |  |       [140, 180], | 
					
						
							|  |  |  |       [160, 220], | 
					
						
							|  |  |  |       [180, 260], | 
					
						
							|  |  |  |       [200, 300] | 
					
						
							|  |  |  |     ]); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   it.skip(FFOX)('should work with mobile viewports and cross process navigations', async({browser, server}) => { | 
					
						
							|  |  |  |     // @see https://crbug.com/929806
 | 
					
						
							|  |  |  |     const context = await browser.newContext({ viewport: {width: 360, height: 640, isMobile: true} }); | 
					
						
							|  |  |  |     const page = await context.newPage(); | 
					
						
							|  |  |  |     await page.goto(server.EMPTY_PAGE); | 
					
						
							|  |  |  |     await page.goto(server.CROSS_PROCESS_PREFIX + '/mobile.html'); | 
					
						
							|  |  |  |     await page.evaluate(() => { | 
					
						
							|  |  |  |       document.addEventListener('click', event => { | 
					
						
							|  |  |  |         window.result = {x: event.clientX, y: event.clientY}; | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  |     await page.mouse.click(30, 40); | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  |     expect(await page.evaluate('result')).toEqual({x: 30, y: 40}); | 
					
						
							|  |  |  |     await context.close(); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   xdescribe('Drag and Drop', function() { | 
					
						
							|  |  |  |     it('should work', async({server, page}) => { | 
					
						
							|  |  |  |       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'); | 
					
						
							|  |  |  |     }) | 
					
						
							| 
									
										
										
										
											2019-11-18 18:18:28 -08:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2020-04-08 15:19:09 -07:00
										 |  |  | }); |