2021-11-08 20:58:24 +03:00
|
|
|
/**
|
|
|
|
* Copyright 2018 Google Inc. All rights reserved.
|
|
|
|
* Modifications 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.
|
|
|
|
*/
|
|
|
|
|
2022-04-07 00:57:14 +03:00
|
|
|
import type { Page } from 'playwright-core';
|
2021-11-08 20:58:24 +03:00
|
|
|
import { test as it, expect } from './pageTest';
|
|
|
|
|
|
|
|
async function routeIframe(page: Page) {
|
|
|
|
await page.route('**/empty.html', route => {
|
|
|
|
route.fulfill({
|
2024-03-18 23:42:08 +03:00
|
|
|
body: '<iframe src="iframe.html" name="frame1"></iframe>',
|
2021-11-08 20:58:24 +03:00
|
|
|
contentType: 'text/html'
|
|
|
|
}).catch(() => {});
|
|
|
|
});
|
|
|
|
await page.route('**/iframe.html', route => {
|
|
|
|
route.fulfill({
|
|
|
|
body: `
|
|
|
|
<html>
|
|
|
|
<div>
|
2022-09-28 07:06:07 +03:00
|
|
|
<button data-testid="buttonId">Hello iframe</button>
|
2021-11-08 20:58:24 +03:00
|
|
|
<iframe src="iframe-2.html"></iframe>
|
|
|
|
</div>
|
|
|
|
<span>1</span>
|
|
|
|
<span>2</span>
|
2022-09-30 07:45:44 +03:00
|
|
|
<label for=target>Name</label><input id=target type=text placeholder=Placeholder title=Title alt=Alternative>
|
2021-11-08 20:58:24 +03:00
|
|
|
</html>`,
|
|
|
|
contentType: 'text/html'
|
|
|
|
}).catch(() => {});
|
|
|
|
});
|
|
|
|
await page.route('**/iframe-2.html', route => {
|
|
|
|
route.fulfill({
|
|
|
|
body: '<html><button>Hello nested iframe</button></html>',
|
|
|
|
contentType: 'text/html'
|
|
|
|
}).catch(() => {});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-11-10 01:14:20 +03:00
|
|
|
async function routeAmbiguous(page: Page) {
|
|
|
|
await page.route('**/empty.html', route => {
|
|
|
|
route.fulfill({
|
|
|
|
body: `<iframe src="iframe-1.html"></iframe>
|
|
|
|
<iframe src="iframe-2.html"></iframe>
|
|
|
|
<iframe src="iframe-3.html"></iframe>`,
|
|
|
|
contentType: 'text/html'
|
|
|
|
}).catch(() => {});
|
|
|
|
});
|
|
|
|
await page.route('**/iframe-*', route => {
|
|
|
|
const path = new URL(route.request().url()).pathname.slice(1);
|
|
|
|
route.fulfill({
|
|
|
|
body: `<html><button>Hello from ${path}</button></html>`,
|
|
|
|
contentType: 'text/html'
|
|
|
|
}).catch(() => {});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-03-10 21:42:52 +03:00
|
|
|
it('should work for iframe @smoke', async ({ page, server }) => {
|
2021-11-08 20:58:24 +03:00
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
2022-09-29 22:50:52 +03:00
|
|
|
const button = page.frameLocator('iframe').locator('button');
|
2021-11-08 20:58:24 +03:00
|
|
|
await button.waitFor();
|
|
|
|
expect(await button.innerText()).toBe('Hello iframe');
|
|
|
|
await expect(button).toHaveText('Hello iframe');
|
|
|
|
await button.click();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work for nested iframe', async ({ page, server }) => {
|
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
2022-09-29 22:50:52 +03:00
|
|
|
const button = page.frameLocator('iframe').frameLocator('iframe').locator('button');
|
2021-11-08 20:58:24 +03:00
|
|
|
await button.waitFor();
|
|
|
|
expect(await button.innerText()).toBe('Hello nested iframe');
|
|
|
|
await expect(button).toHaveText('Hello nested iframe');
|
|
|
|
await button.click();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work for $ and $$', async ({ page, server }) => {
|
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
2022-09-29 22:50:52 +03:00
|
|
|
const locator = page.frameLocator('iframe').locator('button');
|
2021-11-08 20:58:24 +03:00
|
|
|
await expect(locator).toHaveText('Hello iframe');
|
2022-09-29 22:50:52 +03:00
|
|
|
const spans = page.frameLocator('iframe').locator('span');
|
2021-11-08 20:58:24 +03:00
|
|
|
await expect(spans).toHaveCount(2);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should wait for frame', async ({ page, server }) => {
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
2022-11-15 21:50:46 +03:00
|
|
|
const error = await page.locator('body').frameLocator('iframe').locator('span').click({ timeout: 1000 }).catch(e => e);
|
2024-09-19 06:15:01 +03:00
|
|
|
expect(error.message).toContain(`waiting for locator('body').locator('iframe').contentFrame()`);
|
2021-11-08 20:58:24 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should wait for frame 2', async ({ page, server }) => {
|
|
|
|
await routeIframe(page);
|
|
|
|
setTimeout(() => page.goto(server.EMPTY_PAGE).catch(() => {}), 300);
|
|
|
|
await page.frameLocator('iframe').locator('button').click();
|
|
|
|
});
|
|
|
|
|
2022-08-03 11:51:45 +03:00
|
|
|
it('should wait for frame to go', async ({ page, server, isAndroid }) => {
|
|
|
|
it.fixme(isAndroid);
|
|
|
|
|
2021-11-08 20:58:24 +03:00
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
setTimeout(() => page.$eval('iframe', e => e.remove()).catch(() => {}), 300);
|
|
|
|
await expect(page.frameLocator('iframe').locator('button')).toBeHidden();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not wait for frame', async ({ page, server }) => {
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await expect(page.frameLocator('iframe').locator('span')).toBeHidden();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not wait for frame 2', async ({ page, server }) => {
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await expect(page.frameLocator('iframe').locator('span')).not.toBeVisible();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not wait for frame 3', async ({ page, server }) => {
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await expect(page.frameLocator('iframe').locator('span')).toHaveCount(0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should click in lazy iframe', async ({ page, server }) => {
|
|
|
|
await page.route('**/iframe.html', route => {
|
|
|
|
route.fulfill({
|
|
|
|
body: '<html><button>Hello iframe</button></html>',
|
|
|
|
contentType: 'text/html'
|
|
|
|
}).catch(() => {});
|
|
|
|
});
|
|
|
|
|
|
|
|
// empty pge
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
|
|
|
|
// add blank iframe
|
|
|
|
setTimeout(() => {
|
2023-06-02 22:59:12 +03:00
|
|
|
void page.evaluate(() => {
|
2021-11-08 20:58:24 +03:00
|
|
|
const iframe = document.createElement('iframe');
|
|
|
|
document.body.appendChild(iframe);
|
|
|
|
});
|
|
|
|
// navigate iframe
|
|
|
|
setTimeout(() => {
|
2023-06-02 22:59:12 +03:00
|
|
|
void page.evaluate(() => document.querySelector('iframe').src = 'iframe.html');
|
2021-11-08 20:58:24 +03:00
|
|
|
}, 500);
|
|
|
|
}, 500);
|
|
|
|
|
|
|
|
// Click in iframe
|
|
|
|
const button = page.frameLocator('iframe').locator('button');
|
|
|
|
const [, text] = await Promise.all([
|
|
|
|
button.click(),
|
|
|
|
button.innerText(),
|
|
|
|
expect(button).toHaveText('Hello iframe')
|
|
|
|
]);
|
|
|
|
expect(text).toBe('Hello iframe');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('waitFor should survive frame reattach', async ({ page, server }) => {
|
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const button = page.frameLocator('iframe').locator('button:has-text("Hello nested iframe")');
|
|
|
|
const promise = button.waitFor();
|
|
|
|
await page.locator('iframe').evaluate(e => e.remove());
|
|
|
|
await page.evaluate(() => {
|
|
|
|
const iframe = document.createElement('iframe');
|
|
|
|
iframe.src = 'iframe-2.html';
|
|
|
|
document.body.appendChild(iframe);
|
|
|
|
});
|
|
|
|
await promise;
|
|
|
|
});
|
|
|
|
|
|
|
|
it('click should survive frame reattach', async ({ page, server }) => {
|
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const button = page.frameLocator('iframe').locator('button:has-text("Hello nested iframe")');
|
|
|
|
const promise = button.click();
|
|
|
|
await page.locator('iframe').evaluate(e => e.remove());
|
|
|
|
await page.evaluate(() => {
|
|
|
|
const iframe = document.createElement('iframe');
|
|
|
|
iframe.src = 'iframe-2.html';
|
|
|
|
document.body.appendChild(iframe);
|
|
|
|
});
|
|
|
|
await promise;
|
|
|
|
});
|
|
|
|
|
|
|
|
it('click should survive iframe navigation', async ({ page, server }) => {
|
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const button = page.frameLocator('iframe').locator('button:has-text("Hello nested iframe")');
|
|
|
|
const promise = button.click();
|
2023-06-02 22:59:12 +03:00
|
|
|
void page.locator('iframe').evaluate(e => (e as HTMLIFrameElement).src = 'iframe-2.html');
|
2021-11-08 20:58:24 +03:00
|
|
|
await promise;
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should non work for non-frame', async ({ page, server }) => {
|
|
|
|
await routeIframe(page);
|
|
|
|
await page.setContent('<div></div>');
|
|
|
|
const button = page.frameLocator('div').locator('button');
|
|
|
|
const error = await button.waitFor().catch(e => e);
|
|
|
|
expect(error.message).toContain('<div></div>');
|
|
|
|
expect(error.message).toContain('<iframe> was expected');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('locator.frameLocator should work for iframe', async ({ page, server }) => {
|
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const button = page.locator('body').frameLocator('iframe').locator('button');
|
|
|
|
await button.waitFor();
|
|
|
|
expect(await button.innerText()).toBe('Hello iframe');
|
|
|
|
await expect(button).toHaveText('Hello iframe');
|
|
|
|
await button.click();
|
|
|
|
});
|
2021-11-10 01:14:20 +03:00
|
|
|
|
|
|
|
it('locator.frameLocator should throw on ambiguity', async ({ page, server }) => {
|
|
|
|
await routeAmbiguous(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const button = page.locator('body').frameLocator('iframe').locator('button');
|
|
|
|
const error = await button.waitFor().catch(e => e);
|
2022-11-05 01:19:16 +03:00
|
|
|
expect(error.message).toContain(`Error: strict mode violation: locator('body').locator('iframe') resolved to 3 elements`);
|
2021-11-10 01:14:20 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
it('locator.frameLocator should not throw on first/last/nth', async ({ page, server }) => {
|
|
|
|
await routeAmbiguous(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const button1 = page.locator('body').frameLocator('iframe').first().locator('button');
|
|
|
|
await expect(button1).toHaveText('Hello from iframe-1.html');
|
|
|
|
const button2 = page.locator('body').frameLocator('iframe').nth(1).locator('button');
|
|
|
|
await expect(button2).toHaveText('Hello from iframe-2.html');
|
|
|
|
const button3 = page.locator('body').frameLocator('iframe').last().locator('button');
|
|
|
|
await expect(button3).toHaveText('Hello from iframe-3.html');
|
|
|
|
});
|
2022-09-28 02:13:56 +03:00
|
|
|
|
2022-09-30 04:12:49 +03:00
|
|
|
it('getBy coverage', async ({ page, server }) => {
|
2022-09-28 02:13:56 +03:00
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const button1 = page.frameLocator('iframe').getByRole('button');
|
|
|
|
const button2 = page.frameLocator('iframe').getByText('Hello');
|
2022-09-28 07:06:07 +03:00
|
|
|
const button3 = page.frameLocator('iframe').getByTestId('buttonId');
|
2022-09-28 02:13:56 +03:00
|
|
|
await expect(button1).toHaveText('Hello iframe');
|
|
|
|
await expect(button2).toHaveText('Hello iframe');
|
2022-09-28 07:06:07 +03:00
|
|
|
await expect(button3).toHaveText('Hello iframe');
|
2022-10-04 20:29:26 +03:00
|
|
|
const input1 = page.frameLocator('iframe').getByLabel('Name');
|
2022-09-30 04:12:49 +03:00
|
|
|
await expect(input1).toHaveValue('');
|
2022-10-04 20:29:26 +03:00
|
|
|
const input2 = page.frameLocator('iframe').getByPlaceholder('Placeholder');
|
2022-09-30 04:12:49 +03:00
|
|
|
await expect(input2).toHaveValue('');
|
2022-09-30 07:45:44 +03:00
|
|
|
const input3 = page.frameLocator('iframe').getByAltText('Alternative');
|
|
|
|
await expect(input3).toHaveValue('');
|
|
|
|
const input4 = page.frameLocator('iframe').getByTitle('Title');
|
|
|
|
await expect(input4).toHaveValue('');
|
2022-09-28 02:13:56 +03:00
|
|
|
});
|
2023-03-22 18:56:50 +03:00
|
|
|
|
|
|
|
it('wait for hidden should succeed when frame is not in dom', async ({ page }) => {
|
|
|
|
it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/21879' });
|
|
|
|
await page.goto('about:blank');
|
|
|
|
const button = page.frameLocator('iframe1').locator('button');
|
|
|
|
expect(await button.isHidden()).toBeTruthy();
|
|
|
|
await button.waitFor({ state: 'hidden', timeout: 1000 });
|
2023-03-31 20:54:07 +03:00
|
|
|
await button.waitFor({ state: 'detached', timeout: 1000 });
|
|
|
|
const error = await button.waitFor({ state: 'attached', timeout: 1000 }).catch(e => e);
|
|
|
|
expect(error.message).toContain('Timeout 1000ms exceeded');
|
2023-03-22 18:56:50 +03:00
|
|
|
});
|
2023-11-10 18:44:02 +03:00
|
|
|
|
|
|
|
it('should work with COEP/COOP/CORP isolated iframe', async ({ page, server, browserName }) => {
|
|
|
|
it.info().annotations.push({ type: 'issue', description: 'https://github.com/microsoft/playwright/issues/28082' });
|
|
|
|
it.fixme(browserName === 'firefox');
|
|
|
|
await page.route('**/empty.html', route => {
|
|
|
|
return route.fulfill({
|
|
|
|
body: `<iframe src="https://${server.CROSS_PROCESS_PREFIX}/btn.html" allow="cross-origin-isolated; fullscreen" sandbox="allow-same-origin allow-scripts allow-popups" ></iframe>`,
|
|
|
|
contentType: 'text/html',
|
|
|
|
headers: {
|
|
|
|
'cross-origin-embedder-policy': 'require-corp',
|
|
|
|
'cross-origin-opener-policy': 'same-origin',
|
|
|
|
'cross-origin-resource-policy': 'cross-origin',
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
await page.route('**/btn.html', route => {
|
|
|
|
return route.fulfill({
|
|
|
|
body: '<button onclick="window.__clicked=true">Click target</button>',
|
|
|
|
contentType: 'text/html',
|
|
|
|
headers: {
|
|
|
|
'cross-origin-embedder-policy': 'require-corp',
|
|
|
|
'cross-origin-opener-policy': 'same-origin',
|
|
|
|
'cross-origin-resource-policy': 'cross-origin',
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await page.frameLocator('iframe').getByRole('button').click();
|
|
|
|
expect(await page.frames()[1].evaluate(() => window['__clicked'])).toBe(true);
|
|
|
|
});
|
2024-03-18 23:42:08 +03:00
|
|
|
|
2024-03-25 17:42:13 +03:00
|
|
|
it('locator.contentFrame should work', async ({ page, server }) => {
|
2024-03-18 23:42:08 +03:00
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const locator = page.locator('iframe');
|
2024-03-25 17:42:13 +03:00
|
|
|
const frameLocator = locator.contentFrame();
|
2024-03-18 23:42:08 +03:00
|
|
|
const button = frameLocator.locator('button');
|
|
|
|
expect(await button.innerText()).toBe('Hello iframe');
|
|
|
|
await expect(button).toHaveText('Hello iframe');
|
|
|
|
await button.click();
|
|
|
|
});
|
|
|
|
|
2024-03-25 17:42:13 +03:00
|
|
|
it('frameLocator.owner should work', async ({ page, server }) => {
|
2024-03-18 23:42:08 +03:00
|
|
|
await routeIframe(page);
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
const frameLocator = page.frameLocator('iframe');
|
2024-03-25 17:42:13 +03:00
|
|
|
const locator = frameLocator.owner();
|
2024-03-18 23:42:08 +03:00
|
|
|
await expect(locator).toBeVisible();
|
|
|
|
expect(await locator.getAttribute('name')).toBe('frame1');
|
|
|
|
});
|