mirror of
https://github.com/microsoft/playwright.git
synced 2024-12-14 21:53:35 +03:00
747 lines
27 KiB
TypeScript
747 lines
27 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 { test, expect } from './inspectorTest';
|
|
|
|
test.describe('cli codegen', () => {
|
|
test.skip(({ mode }) => mode !== 'default');
|
|
test.fixme(({ browserName, headless }) => browserName === 'firefox' && !headless, 'Focus is off');
|
|
|
|
test('should click', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<button onclick="console.log('click')">Submit</button>`);
|
|
|
|
const selector = await recorder.hoverOverElement('button');
|
|
expect(selector).toBe('text=Submit');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'click'),
|
|
page.dispatchEvent('button', 'click', { detail: 1 })
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Click text=Submit
|
|
await page.locator('text=Submit').click();`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Click text=Submit
|
|
page.locator("text=Submit").click()`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Click text=Submit
|
|
await page.locator("text=Submit").click()`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Click text=Submit
|
|
page.locator("text=Submit").click();`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Click text=Submit
|
|
await page.Locator("text=Submit").ClickAsync();`);
|
|
|
|
expect(message.text()).toBe('click');
|
|
});
|
|
|
|
test('should click after same-document navigation', async ({ page, openRecorder, server }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
server.setRoute('/foo.html', (req, res) => {
|
|
res.setHeader('Content-Type', 'text/html; charset=utf-8');
|
|
res.end('');
|
|
});
|
|
await recorder.setContentAndWait(`<button onclick="console.log('click')">Submit</button>`, server.PREFIX + '/foo.html');
|
|
await Promise.all([
|
|
page.waitForNavigation(),
|
|
page.evaluate(() => history.pushState({}, '', '/url.html')),
|
|
]);
|
|
// This is the only way to give recorder a chance to install
|
|
// the second unnecessary copy of the recorder script.
|
|
await page.waitForTimeout(1000);
|
|
|
|
const selector = await recorder.hoverOverElement('button');
|
|
expect(selector).toBe('text=Submit');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'click'),
|
|
page.dispatchEvent('button', 'click', { detail: 1 })
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Click text=Submit
|
|
await page.locator('text=Submit').click();`);
|
|
expect(message.text()).toBe('click');
|
|
});
|
|
|
|
test('should make a positioned click on a canvas', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`
|
|
<canvas width="500" height="500" style="margin: 42px"/>
|
|
<script>
|
|
document.querySelector("canvas").addEventListener("click", event => {
|
|
const rect = event.target.getBoundingClientRect();
|
|
console.log("click", event.clientX - rect.left, event.clientY - rect.top);
|
|
})
|
|
</script>
|
|
`);
|
|
|
|
const selector = await recorder.waitForHighlight(() => recorder.page.hover('canvas', {
|
|
position: { x: 250, y: 250 },
|
|
}));
|
|
expect(selector).toBe('canvas');
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'click'),
|
|
recorder.page.click('canvas', {
|
|
position: { x: 250, y: 250 },
|
|
})
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Click canvas
|
|
await page.locator('canvas').click({
|
|
position: {
|
|
x: 250,
|
|
y: 250
|
|
}
|
|
});`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Click canvas
|
|
page.locator("canvas").click(position={"x":250,"y":250})`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Click canvas
|
|
await page.locator("canvas").click(position={"x":250,"y":250})`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Click canvas
|
|
page.locator("canvas").click(new Locator.ClickOptions()
|
|
.setPosition(250, 250));`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Click canvas
|
|
await page.Locator("canvas").ClickAsync(new LocatorClickOptions
|
|
{
|
|
Position = new Position
|
|
{
|
|
X = 250,
|
|
Y = 250,
|
|
},
|
|
});`);
|
|
expect(message.text()).toBe('click 250 250');
|
|
});
|
|
|
|
test('should work with TrustedTypes', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`
|
|
<head>
|
|
<meta http-equiv="Content-Security-Policy" content="trusted-types unsafe escape; require-trusted-types-for 'script'">
|
|
</head>
|
|
<body>
|
|
<button onclick="console.log('click')">Submit</button>
|
|
</body>`);
|
|
|
|
const selector = await recorder.hoverOverElement('button');
|
|
expect(selector).toBe('text=Submit');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'click'),
|
|
page.dispatchEvent('button', 'click', { detail: 1 })
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Click text=Submit
|
|
await page.locator('text=Submit').click();`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Click text=Submit
|
|
page.locator("text=Submit").click()`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Click text=Submit
|
|
await page.locator("text=Submit").click()`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Click text=Submit
|
|
page.locator("text=Submit").click();`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Click text=Submit
|
|
await page.Locator("text=Submit").ClickAsync();`);
|
|
|
|
expect(message.text()).toBe('click');
|
|
});
|
|
|
|
test('should not target selector preview by text regexp', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<span>dummy</span>`);
|
|
|
|
// Force highlight.
|
|
await recorder.hoverOverElement('span');
|
|
|
|
// Append text after highlight.
|
|
await page.evaluate(() => {
|
|
const div = document.createElement('div');
|
|
div.setAttribute('onclick', "console.log('click')");
|
|
div.textContent = ' Some long text here ';
|
|
document.documentElement.appendChild(div);
|
|
});
|
|
|
|
const selector = await recorder.hoverOverElement('div');
|
|
expect(selector).toBe('text=Some long text here');
|
|
|
|
// Sanity check that selector does not match our highlight.
|
|
const divContents = await page.$eval(selector, div => div.outerHTML);
|
|
expect(divContents.replace(/\s__playwright_target__="[^"]+"/, '')).toBe(`<div onclick="console.log('click')"> Some long text here </div>`);
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'click'),
|
|
page.dispatchEvent('div', 'click', { detail: 1 })
|
|
]);
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Click text=Some long text here
|
|
await page.locator('text=Some long text here').click();`);
|
|
expect(message.text()).toBe('click');
|
|
});
|
|
|
|
test('should fill', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<input id="input" name="name" oninput="console.log(input.value)"></input>`);
|
|
const selector = await recorder.focusElement('input');
|
|
expect(selector).toBe('input[name="name"]');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'fill'),
|
|
page.fill('input', 'John')
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Fill input[name="name"]
|
|
await page.locator('input[name="name"]').fill('John');`);
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Fill input[name="name"]
|
|
page.locator("input[name=\\\"name\\\"]").fill("John");`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Fill input[name="name"]
|
|
page.locator(\"input[name=\\\"name\\\"]\").fill(\"John\")`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Fill input[name="name"]
|
|
await page.locator(\"input[name=\\\"name\\\"]\").fill(\"John\")`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Fill input[name="name"]
|
|
await page.Locator(\"input[name=\\\"name\\\"]\").FillAsync(\"John\");`);
|
|
|
|
expect(message.text()).toBe('John');
|
|
});
|
|
|
|
test('should fill textarea', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<textarea id="textarea" name="name" oninput="console.log(textarea.value)"></textarea>`);
|
|
const selector = await recorder.focusElement('textarea');
|
|
expect(selector).toBe('textarea[name="name"]');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'fill'),
|
|
page.fill('textarea', 'John')
|
|
]);
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Fill textarea[name="name"]
|
|
await page.locator('textarea[name="name"]').fill('John');`);
|
|
expect(message.text()).toBe('John');
|
|
});
|
|
|
|
test('should press', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<input name="name" onkeypress="console.log('press')"></input>`);
|
|
|
|
const selector = await recorder.focusElement('input');
|
|
expect(selector).toBe('input[name="name"]');
|
|
|
|
const messages: any[] = [];
|
|
page.on('console', message => messages.push(message));
|
|
const [, sources] = await Promise.all([
|
|
recorder.waitForActionPerformed(),
|
|
recorder.waitForOutput('JavaScript', 'press'),
|
|
page.press('input', 'Shift+Enter')
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Press Enter with modifiers
|
|
await page.locator('input[name="name"]').press('Shift+Enter');`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Press Enter with modifiers
|
|
page.locator("input[name=\\\"name\\\"]").press("Shift+Enter");`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Press Enter with modifiers
|
|
page.locator(\"input[name=\\\"name\\\"]\").press(\"Shift+Enter\")`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Press Enter with modifiers
|
|
await page.locator(\"input[name=\\\"name\\\"]\").press(\"Shift+Enter\")`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Press Enter with modifiers
|
|
await page.Locator(\"input[name=\\\"name\\\"]\").PressAsync(\"Shift+Enter\");`);
|
|
|
|
expect(messages[0].text()).toBe('press');
|
|
});
|
|
|
|
test('should update selected element after pressing Tab', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`
|
|
<input name="one"></input>
|
|
<input name="two"></input>
|
|
`);
|
|
|
|
await page.click('input[name="one"]');
|
|
await recorder.waitForOutput('JavaScript', 'click');
|
|
await page.keyboard.type('foobar123');
|
|
await recorder.waitForOutput('JavaScript', 'foobar123');
|
|
|
|
await page.keyboard.press('Tab');
|
|
await recorder.waitForOutput('JavaScript', 'Tab');
|
|
await page.keyboard.type('barfoo321');
|
|
await recorder.waitForOutput('JavaScript', 'barfoo321');
|
|
|
|
const text = recorder.sources().get('JavaScript').text;
|
|
expect(text).toContain(`
|
|
// Fill input[name="one"]
|
|
await page.locator('input[name="one"]').fill('foobar123');`);
|
|
|
|
expect(text).toContain(`
|
|
// Press Tab
|
|
await page.locator('input[name="one"]').press('Tab');`);
|
|
|
|
expect(text).toContain(`
|
|
// Fill input[name="two"]
|
|
await page.locator('input[name="two"]').fill('barfoo321');`);
|
|
});
|
|
|
|
test('should record ArrowDown', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<input name="name" onkeydown="console.log('press:' + event.key)"></input>`);
|
|
|
|
const selector = await recorder.focusElement('input');
|
|
expect(selector).toBe('input[name="name"]');
|
|
|
|
const messages: any[] = [];
|
|
page.on('console', message => {
|
|
messages.push(message);
|
|
});
|
|
const [, sources] = await Promise.all([
|
|
recorder.waitForActionPerformed(),
|
|
recorder.waitForOutput('JavaScript', 'press'),
|
|
page.press('input', 'ArrowDown')
|
|
]);
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Press ArrowDown
|
|
await page.locator('input[name="name"]').press('ArrowDown');`);
|
|
expect(messages[0].text()).toBe('press:ArrowDown');
|
|
});
|
|
|
|
test('should emit single keyup on ArrowDown', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<input name="name" onkeydown="console.log('down:' + event.key)" onkeyup="console.log('up:' + event.key)"></input>`);
|
|
|
|
const selector = await recorder.focusElement('input');
|
|
expect(selector).toBe('input[name="name"]');
|
|
|
|
const messages: any[] = [];
|
|
page.on('console', message => {
|
|
if (message.type() !== 'error')
|
|
messages.push(message);
|
|
});
|
|
const [, sources] = await Promise.all([
|
|
recorder.waitForActionPerformed(),
|
|
recorder.waitForOutput('JavaScript', 'press'),
|
|
page.press('input', 'ArrowDown')
|
|
]);
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Press ArrowDown
|
|
await page.locator('input[name="name"]').press('ArrowDown');`);
|
|
expect(messages.length).toBe(2);
|
|
expect(messages[0].text()).toBe('down:ArrowDown');
|
|
expect(messages[1].text()).toBe('up:ArrowDown');
|
|
});
|
|
|
|
test('should check', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<input id="checkbox" type="checkbox" name="accept" onchange="console.log(checkbox.checked)"></input>`);
|
|
|
|
const selector = await recorder.focusElement('input');
|
|
expect(selector).toBe('input[name="accept"]');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'check'),
|
|
page.click('input')
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Check input[name="accept"]
|
|
await page.locator('input[name="accept"]').check();`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Check input[name="accept"]
|
|
page.locator("input[name=\\\"accept\\\"]").check();`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Check input[name="accept"]
|
|
page.locator(\"input[name=\\\"accept\\\"]\").check()`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Check input[name="accept"]
|
|
await page.locator(\"input[name=\\\"accept\\\"]\").check()`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Check input[name="accept"]
|
|
await page.Locator(\"input[name=\\\"accept\\\"]\").CheckAsync();`);
|
|
|
|
expect(message.text()).toBe('true');
|
|
});
|
|
|
|
test('should check a radio button', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<input id="checkbox" type="radio" name="accept" onchange="console.log(checkbox.checked)"></input>`);
|
|
|
|
const selector = await recorder.focusElement('input');
|
|
expect(selector).toBe('input[name="accept"]');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'check'),
|
|
page.click('input')
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Check input[name="accept"]
|
|
await page.locator('input[name="accept"]').check();`);
|
|
expect(message.text()).toBe('true');
|
|
});
|
|
|
|
test('should check with keyboard', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<input id="checkbox" type="checkbox" name="accept" onchange="console.log(checkbox.checked)"></input>`);
|
|
|
|
const selector = await recorder.focusElement('input');
|
|
expect(selector).toBe('input[name="accept"]');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'check'),
|
|
page.keyboard.press('Space')
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Check input[name="accept"]
|
|
await page.locator('input[name="accept"]').check();`);
|
|
expect(message.text()).toBe('true');
|
|
});
|
|
|
|
test('should uncheck', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<input id="checkbox" type="checkbox" checked name="accept" onchange="console.log(checkbox.checked)"></input>`);
|
|
|
|
const selector = await recorder.focusElement('input');
|
|
expect(selector).toBe('input[name="accept"]');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'uncheck'),
|
|
page.click('input')
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Uncheck input[name="accept"]
|
|
await page.locator('input[name="accept"]').uncheck();`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Uncheck input[name="accept"]
|
|
page.locator("input[name=\\\"accept\\\"]").uncheck();`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Uncheck input[name="accept"]
|
|
page.locator(\"input[name=\\\"accept\\\"]\").uncheck()`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Uncheck input[name="accept"]
|
|
await page.locator(\"input[name=\\\"accept\\\"]\").uncheck()`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Uncheck input[name="accept"]
|
|
await page.Locator(\"input[name=\\\"accept\\\"]\").UncheckAsync();`);
|
|
|
|
expect(message.text()).toBe('false');
|
|
});
|
|
|
|
test('should select', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait('<select id="age" onchange="console.log(age.selectedOptions[0].value)"><option value="1"><option value="2"></select>');
|
|
|
|
const selector = await recorder.hoverOverElement('select');
|
|
expect(selector).toBe('select');
|
|
|
|
const [message, sources] = await Promise.all([
|
|
page.waitForEvent('console', msg => msg.type() !== 'error'),
|
|
recorder.waitForOutput('JavaScript', 'select'),
|
|
page.selectOption('select', '2')
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Select 2
|
|
await page.locator('select').selectOption('2');`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Select 2
|
|
page.locator("select").selectOption("2");`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Select 2
|
|
page.locator(\"select\").select_option(\"2\")`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Select 2
|
|
await page.locator(\"select\").select_option(\"2\")`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Select 2
|
|
await page.Locator(\"select\").SelectOptionAsync(new[] { \"2\" });`);
|
|
|
|
expect(message.text()).toBe('2');
|
|
});
|
|
|
|
test('should await popup', async ({ page, openRecorder, browserName, headless }) => {
|
|
test.fixme(browserName === 'webkit' && !headless, 'Middle click does not open a popup in our webkit embedder');
|
|
|
|
const recorder = await openRecorder();
|
|
await recorder.setContentAndWait('<a target=_blank rel=noopener href="about:blank">link</a>');
|
|
|
|
const selector = await recorder.hoverOverElement('a');
|
|
expect(selector).toBe('text=link');
|
|
|
|
const [popup, sources] = await Promise.all([
|
|
page.context().waitForEvent('page'),
|
|
recorder.waitForOutput('JavaScript', 'waitForEvent'),
|
|
page.dispatchEvent('a', 'click', { detail: 1 })
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Click text=link
|
|
const [page1] = await Promise.all([
|
|
page.waitForEvent('popup'),
|
|
page.locator('text=link').click()
|
|
]);`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Click text=link
|
|
Page page1 = page.waitForPopup(() -> {
|
|
page.locator("text=link").click();
|
|
});`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Click text=link
|
|
with page.expect_popup() as popup_info:
|
|
page.locator(\"text=link\").click()
|
|
page1 = popup_info.value`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Click text=link
|
|
async with page.expect_popup() as popup_info:
|
|
await page.locator(\"text=link\").click()
|
|
page1 = await popup_info.value`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
var page1 = await page.RunAndWaitForPopupAsync(async () =>
|
|
{
|
|
await page.Locator(\"text=link\").ClickAsync();
|
|
});`);
|
|
|
|
expect(popup.url()).toBe('about:blank');
|
|
});
|
|
|
|
test('should assert navigation', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<a onclick="window.location.href='about:blank#foo'">link</a>`);
|
|
|
|
const selector = await recorder.hoverOverElement('a');
|
|
expect(selector).toBe('text=link');
|
|
const [, sources] = await Promise.all([
|
|
page.waitForNavigation(),
|
|
recorder.waitForOutput('JavaScript', 'assert'),
|
|
page.dispatchEvent('a', 'click', { detail: 1 })
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Click text=link
|
|
await page.locator('text=link').click();
|
|
// assert.equal(page.url(), 'about:blank#foo');`);
|
|
|
|
expect(sources.get('Playwright Test').text).toContain(`
|
|
// Click text=link
|
|
await page.locator('text=link').click();
|
|
await expect(page).toHaveURL('about:blank#foo');`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Click text=link
|
|
page.locator("text=link").click();
|
|
// assertThat(page).hasURL("about:blank#foo");`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Click text=link
|
|
page.locator(\"text=link\").click()
|
|
# expect(page).to_have_url(\"about:blank#foo\")`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Click text=link
|
|
await page.locator(\"text=link\").click()
|
|
# await expect(page).to_have_url(\"about:blank#foo\")`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Click text=link
|
|
await page.Locator(\"text=link\").ClickAsync();
|
|
// Assert.AreEqual(\"about:blank#foo\", page.Url);`);
|
|
|
|
expect(page.url()).toContain('about:blank#foo');
|
|
});
|
|
|
|
|
|
test('should await navigation', async ({ page, openRecorder }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<a onclick="setTimeout(() => window.location.href='about:blank#foo', 1000)">link</a>`);
|
|
|
|
const selector = await recorder.hoverOverElement('a');
|
|
expect(selector).toBe('text=link');
|
|
|
|
const [, sources] = await Promise.all([
|
|
page.waitForNavigation(),
|
|
recorder.waitForOutput('JavaScript', 'waitForNavigation'),
|
|
page.dispatchEvent('a', 'click', { detail: 1 })
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
// Click text=link
|
|
await Promise.all([
|
|
page.waitForNavigation(/*{ url: 'about:blank#foo' }*/),
|
|
page.locator('text=link').click()
|
|
]);`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
// Click text=link
|
|
// page.waitForNavigation(new Page.WaitForNavigationOptions().setUrl("about:blank#foo"), () ->
|
|
page.waitForNavigation(() -> {
|
|
page.locator("text=link").click();
|
|
});`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
# Click text=link
|
|
# with page.expect_navigation(url=\"about:blank#foo\"):
|
|
with page.expect_navigation():
|
|
page.locator(\"text=link\").click()`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
# Click text=link
|
|
# async with page.expect_navigation(url=\"about:blank#foo\"):
|
|
async with page.expect_navigation():
|
|
await page.locator(\"text=link\").click()`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
// Click text=link
|
|
await page.RunAndWaitForNavigationAsync(async () =>
|
|
{
|
|
await page.Locator(\"text=link\").ClickAsync();
|
|
}/*, new PageWaitForNavigationOptions
|
|
{
|
|
UrlString = \"about:blank#foo\"
|
|
}*/);`);
|
|
|
|
expect(page.url()).toContain('about:blank#foo');
|
|
});
|
|
|
|
test('should ignore AltGraph', async ({ openRecorder, browserName }) => {
|
|
test.skip(browserName === 'firefox', 'The TextInputProcessor in Firefox does not work with AltGraph.');
|
|
const recorder = await openRecorder();
|
|
await recorder.setContentAndWait(`<input></input>`);
|
|
|
|
await recorder.page.type('input', 'playwright');
|
|
await recorder.page.keyboard.press('AltGraph');
|
|
await recorder.page.keyboard.insertText('@');
|
|
await recorder.page.keyboard.type('example.com');
|
|
await recorder.waitForOutput('JavaScript', 'example.com');
|
|
expect(recorder.sources().get('JavaScript').text).not.toContain(`await page.locator('input').press('AltGraph');`);
|
|
expect(recorder.sources().get('JavaScript').text).toContain(`await page.locator('input').fill('playwright@example.com');`);
|
|
});
|
|
|
|
test('should middle click', async ({ page, openRecorder, server }) => {
|
|
const recorder = await openRecorder();
|
|
|
|
await recorder.setContentAndWait(`<a href${JSON.stringify(server.EMPTY_PAGE)}>Click me</a>`);
|
|
|
|
const [sources] = await Promise.all([
|
|
recorder.waitForOutput('JavaScript', 'click'),
|
|
page.click('a', { button: 'middle' }),
|
|
]);
|
|
|
|
expect(sources.get('JavaScript').text).toContain(`
|
|
await page.locator('text=Click me').click({
|
|
button: 'middle'
|
|
});`);
|
|
|
|
expect(sources.get('Python').text).toContain(`
|
|
page.locator("text=Click me").click(button="middle")`);
|
|
|
|
expect(sources.get('Python Async').text).toContain(`
|
|
await page.locator("text=Click me").click(button="middle")`);
|
|
|
|
expect(sources.get('Java').text).toContain(`
|
|
page.locator("text=Click me").click(new Locator.ClickOptions()
|
|
.setButton(MouseButton.MIDDLE));`);
|
|
|
|
expect(sources.get('C#').text).toContain(`
|
|
await page.Locator("text=Click me").ClickAsync(new LocatorClickOptions
|
|
{
|
|
Button = MouseButton.Middle,
|
|
});`);
|
|
});
|
|
});
|