2024-10-15 18:47:26 -07: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.
|
|
|
|
*/
|
|
|
|
|
2025-03-28 12:46:20 +01:00
|
|
|
import type { Locator, FrameLocator, Page } from '@playwright/test';
|
2025-05-02 13:28:00 -07:00
|
|
|
import { test as it, expect as baseExpect } from './pageTest';
|
|
|
|
|
|
|
|
const expect = baseExpect.extend({
|
|
|
|
toContainYaml(received: string, expected: string) {
|
|
|
|
const trimmed = expected.split('\n').filter(a => !!a.trim());
|
|
|
|
const maxPrefixLength = Math.min(...trimmed.map(line => line.match(/^\s*/)[0].length));
|
|
|
|
const trimmedExpected = trimmed.map(line => line.substring(maxPrefixLength)).join('\n');
|
|
|
|
try {
|
|
|
|
if (this.isNot)
|
|
|
|
expect(received).not.toContain(trimmedExpected);
|
|
|
|
else
|
|
|
|
expect(received).toContain(trimmedExpected);
|
|
|
|
return {
|
|
|
|
pass: !this.isNot,
|
|
|
|
message: () => '',
|
|
|
|
};
|
|
|
|
} catch (e) {
|
|
|
|
return {
|
|
|
|
pass: this.isNot,
|
|
|
|
message: () => e.message,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2024-10-15 18:47:26 -07:00
|
|
|
|
2025-05-08 08:33:10 -07:00
|
|
|
const forAI = { _forAI: true } as any;
|
|
|
|
|
2024-10-17 17:06:18 -07:00
|
|
|
function unshift(snapshot: string): string {
|
|
|
|
const lines = snapshot.split('\n');
|
|
|
|
let whitespacePrefixLength = 100;
|
|
|
|
for (const line of lines) {
|
|
|
|
if (!line.trim())
|
|
|
|
continue;
|
|
|
|
const match = line.match(/^(\s*)/);
|
|
|
|
if (match && match[1].length < whitespacePrefixLength)
|
|
|
|
whitespacePrefixLength = match[1].length;
|
|
|
|
}
|
|
|
|
return lines.filter(t => t.trim()).map(line => line.substring(whitespacePrefixLength)).join('\n');
|
|
|
|
}
|
|
|
|
|
|
|
|
async function checkAndMatchSnapshot(locator: Locator, snapshot: string) {
|
|
|
|
expect.soft(await locator.ariaSnapshot()).toBe(unshift(snapshot));
|
|
|
|
await expect.soft(locator).toMatchAriaSnapshot(snapshot);
|
|
|
|
}
|
|
|
|
|
|
|
|
it('should snapshot', async ({ page }) => {
|
|
|
|
await page.setContent(`<h1>title</h1>`);
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-19 14:23:08 -07:00
|
|
|
- heading "title" [level=1]
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should snapshot list', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<h1>title</h1>
|
|
|
|
<h1>title 2</h1>
|
|
|
|
`);
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-19 14:23:08 -07:00
|
|
|
- heading "title" [level=1]
|
|
|
|
- heading "title 2" [level=1]
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should snapshot list with accessible name', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<ul aria-label="my list">
|
|
|
|
<li>one</li>
|
|
|
|
<li>two</li>
|
|
|
|
</ul>
|
|
|
|
`);
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- list "my list":
|
2024-10-29 16:19:08 -07:00
|
|
|
- listitem: one
|
|
|
|
- listitem: two
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should snapshot complex', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<ul>
|
|
|
|
<li>
|
|
|
|
<a href='about:blank'>link</a>
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
`);
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- list:
|
|
|
|
- listitem:
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "link":
|
|
|
|
- /url: about:blank
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should allow text nodes', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<h1>Microsoft</h1>
|
|
|
|
<div>Open source projects and samples from Microsoft</div>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-19 14:23:08 -07:00
|
|
|
- heading "Microsoft" [level=1]
|
2024-10-29 16:19:08 -07:00
|
|
|
- text: Open source projects and samples from Microsoft
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should snapshot details visibility', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<details>
|
|
|
|
<summary>Summary</summary>
|
|
|
|
<div>Details</div>
|
|
|
|
</details>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-29 16:19:08 -07:00
|
|
|
- group: Summary
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should snapshot integration', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<h1>Microsoft</h1>
|
|
|
|
<div>Open source projects and samples from Microsoft</div>
|
|
|
|
<ul>
|
|
|
|
<li>
|
|
|
|
<details>
|
|
|
|
<summary>
|
|
|
|
Verified
|
|
|
|
</summary>
|
|
|
|
<div>
|
|
|
|
<div>
|
|
|
|
<p>
|
|
|
|
We've verified that the organization <strong>microsoft</strong> controls the domain:
|
|
|
|
</p>
|
|
|
|
<ul>
|
|
|
|
<li class="mb-1">
|
|
|
|
<strong>opensource.microsoft.com</strong>
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
<div>
|
|
|
|
<a href="about: blank">Learn more about verified organizations</a>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</details>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a href="about:blank">
|
|
|
|
<summary title="Label: GitHub Sponsor">Sponsor</summary>
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
</ul>`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-19 14:23:08 -07:00
|
|
|
- heading "Microsoft" [level=1]
|
2024-10-29 16:19:08 -07:00
|
|
|
- text: Open source projects and samples from Microsoft
|
2024-10-17 17:06:18 -07:00
|
|
|
- list:
|
|
|
|
- listitem:
|
2024-10-29 16:19:08 -07:00
|
|
|
- group: Verified
|
2024-10-17 17:06:18 -07:00
|
|
|
- listitem:
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "Sponsor":
|
|
|
|
- /url: about:blank
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support multiline text', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<p>
|
|
|
|
Line 1
|
|
|
|
Line 2
|
|
|
|
Line 3
|
|
|
|
</p>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-29 16:19:08 -07:00
|
|
|
- paragraph: Line 1 Line 2 Line 3
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
await expect(page.locator('body')).toMatchAriaSnapshot(`
|
2024-10-24 16:49:10 -07:00
|
|
|
- paragraph: |
|
2024-10-23 17:34:21 -07:00
|
|
|
Line 1
|
|
|
|
Line 2
|
|
|
|
Line 3
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should concatenate span text', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<span>One</span> <span>Two</span> <span>Three</span>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-29 16:19:08 -07:00
|
|
|
- text: One Two Three
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should concatenate span text 2', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<span>One </span><span>Two </span><span>Three</span>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-29 16:19:08 -07:00
|
|
|
- text: One Two Three
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should concatenate div text with spaces', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<div>One</div><div>Two</div><div>Three</div>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-29 16:19:08 -07:00
|
|
|
- text: One Two Three
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should include pseudo in text', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<style>
|
|
|
|
span:before {
|
|
|
|
content: 'world';
|
|
|
|
}
|
|
|
|
div:after {
|
|
|
|
content: 'bye';
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
<a href="about:blank">
|
|
|
|
<span>hello</span>
|
|
|
|
<div>hello</div>
|
|
|
|
</a>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "worldhello hellobye":
|
|
|
|
- /url: about:blank
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not include hidden pseudo in text', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<style>
|
|
|
|
span:before {
|
|
|
|
content: 'world';
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
div:after {
|
|
|
|
content: 'bye';
|
|
|
|
visibility: hidden;
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
<a href="about:blank">
|
|
|
|
<span>hello</span>
|
|
|
|
<div>hello</div>
|
|
|
|
</a>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "hello hello":
|
|
|
|
- /url: about:blank
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
2024-10-15 18:47:26 -07:00
|
|
|
});
|
|
|
|
|
2024-10-17 17:06:18 -07:00
|
|
|
it('should include new line for block pseudo', async ({ page }) => {
|
2024-10-15 18:47:26 -07:00
|
|
|
await page.setContent(`
|
2024-10-17 17:06:18 -07:00
|
|
|
<style>
|
|
|
|
span:before {
|
|
|
|
content: 'world';
|
|
|
|
display: block;
|
|
|
|
}
|
|
|
|
div:after {
|
|
|
|
content: 'bye';
|
|
|
|
display: block;
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
<a href="about:blank">
|
|
|
|
<span>hello</span>
|
|
|
|
<div>hello</div>
|
|
|
|
</a>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "world hello hello bye":
|
|
|
|
- /url: about:blank
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
2024-10-15 18:47:26 -07:00
|
|
|
});
|
|
|
|
|
2024-10-17 17:06:18 -07:00
|
|
|
it('should work with slots', async ({ page }) => {
|
|
|
|
// Text "foo" is assigned to the slot, should not be used twice.
|
|
|
|
await page.setContent(`
|
|
|
|
<button><div>foo</div></button>
|
|
|
|
<script>
|
|
|
|
(() => {
|
|
|
|
const container = document.querySelector('div');
|
|
|
|
const shadow = container.attachShadow({ mode: 'open' });
|
|
|
|
const slot = document.createElement('slot');
|
|
|
|
shadow.appendChild(slot);
|
|
|
|
})();
|
|
|
|
</script>
|
|
|
|
`);
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- button "foo"
|
|
|
|
`);
|
|
|
|
|
|
|
|
// Text "foo" is assigned to the slot, should be used instead of slot content.
|
|
|
|
await page.setContent(`
|
|
|
|
<div>foo</div>
|
|
|
|
<script>
|
|
|
|
(() => {
|
|
|
|
const container = document.querySelector('div');
|
|
|
|
const shadow = container.attachShadow({ mode: 'open' });
|
|
|
|
const button = document.createElement('button');
|
|
|
|
shadow.appendChild(button);
|
|
|
|
const slot = document.createElement('slot');
|
|
|
|
button.appendChild(slot);
|
|
|
|
const span = document.createElement('span');
|
|
|
|
span.textContent = 'pre';
|
|
|
|
slot.appendChild(span);
|
|
|
|
})();
|
|
|
|
</script>
|
|
|
|
`);
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- button "foo"
|
|
|
|
`);
|
|
|
|
|
|
|
|
// Nothing is assigned to the slot, should use slot content.
|
2024-10-15 18:47:26 -07:00
|
|
|
await page.setContent(`
|
2024-10-17 17:06:18 -07:00
|
|
|
<div></div>
|
|
|
|
<script>
|
|
|
|
(() => {
|
|
|
|
const container = document.querySelector('div');
|
|
|
|
const shadow = container.attachShadow({ mode: 'open' });
|
|
|
|
const button = document.createElement('button');
|
|
|
|
shadow.appendChild(button);
|
|
|
|
const slot = document.createElement('slot');
|
|
|
|
button.appendChild(slot);
|
|
|
|
const span = document.createElement('span');
|
|
|
|
span.textContent = 'pre';
|
|
|
|
slot.appendChild(span);
|
|
|
|
})();
|
|
|
|
</script>
|
|
|
|
`);
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- button "pre"
|
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should snapshot inner text', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<div role="listitem">
|
|
|
|
<div>
|
|
|
|
<div>
|
|
|
|
<span title="a.test.ts">a.test.ts</span>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<button title="Run"></button>
|
|
|
|
<button title="Show source"></button>
|
|
|
|
<button title="Watch"></button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div role="listitem">
|
|
|
|
<div>
|
|
|
|
<div>
|
|
|
|
<span title="snapshot">snapshot</span>
|
|
|
|
</div>
|
|
|
|
<div class="ui-mode-list-item-time">30ms</div>
|
|
|
|
<div>
|
|
|
|
<button title="Run"></button>
|
|
|
|
<button title="Show source"></button>
|
|
|
|
<button title="Watch"></button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- listitem:
|
2024-10-29 16:19:08 -07:00
|
|
|
- text: a.test.ts
|
2024-10-17 17:06:18 -07:00
|
|
|
- button "Run"
|
|
|
|
- button "Show source"
|
|
|
|
- button "Watch"
|
|
|
|
- listitem:
|
2024-10-29 16:19:08 -07:00
|
|
|
- text: snapshot 30ms
|
2024-10-17 17:06:18 -07:00
|
|
|
- button "Run"
|
|
|
|
- button "Show source"
|
|
|
|
- button "Watch"
|
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should include pseudo codepoints', async ({ page, server }) => {
|
|
|
|
await page.goto(server.EMPTY_PAGE);
|
|
|
|
await page.setContent(`
|
|
|
|
<link href="codicon.css" rel="stylesheet" />
|
|
|
|
<p class='codicon codicon-check'>hello</p>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-29 16:19:08 -07:00
|
|
|
- paragraph: \ueab2hello
|
2024-10-23 17:34:21 -07:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
2024-10-31 17:14:11 -07:00
|
|
|
it('check aria-hidden text', async ({ page }) => {
|
2024-10-23 17:34:21 -07:00
|
|
|
await page.setContent(`
|
|
|
|
<p>
|
|
|
|
<span>hello</span>
|
|
|
|
<span aria-hidden="true">world</span>
|
|
|
|
</p>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-29 16:19:08 -07:00
|
|
|
- paragraph: hello
|
2024-10-17 17:06:18 -07:00
|
|
|
`);
|
2024-10-15 18:47:26 -07:00
|
|
|
});
|
2024-10-25 16:13:38 -07:00
|
|
|
|
2024-10-31 17:14:11 -07:00
|
|
|
it('should ignore presentation and none roles', async ({ page }) => {
|
2024-10-25 16:13:38 -07:00
|
|
|
await page.setContent(`
|
|
|
|
<ul>
|
|
|
|
<li role='presentation'>hello</li>
|
|
|
|
<li role='none'>world</li>
|
|
|
|
</ul>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-10-29 16:19:08 -07:00
|
|
|
- list: hello world
|
2024-10-25 16:13:38 -07:00
|
|
|
`);
|
|
|
|
});
|
2024-10-31 20:41:52 -07:00
|
|
|
|
2025-04-08 08:02:19 +00:00
|
|
|
it('should treat input value as text in templates, but not for checkbox/radio/file', async ({ page }) => {
|
2024-10-31 20:41:52 -07:00
|
|
|
await page.setContent(`
|
|
|
|
<input value='hello world'>
|
2025-04-08 08:02:19 +00:00
|
|
|
<input type=file>
|
|
|
|
<input type=checkbox checked>
|
|
|
|
<input type=radio checked>
|
2024-10-31 20:41:52 -07:00
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- textbox: hello world
|
2025-04-08 08:02:19 +00:00
|
|
|
- button "Choose File"
|
|
|
|
- checkbox [checked]
|
|
|
|
- radio [checked]
|
2024-10-31 20:41:52 -07:00
|
|
|
`);
|
|
|
|
});
|
2024-11-01 15:25:38 -07:00
|
|
|
|
2024-12-10 14:04:18 -08:00
|
|
|
it('should not use on as checkbox value', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<input type='checkbox'>
|
|
|
|
<input type='radio'>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- checkbox
|
|
|
|
- radio
|
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
2024-11-01 15:25:38 -07:00
|
|
|
it('should respect aria-owns', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<a href='about:blank' aria-owns='input p'>
|
|
|
|
<div role='region'>Link 1</div>
|
|
|
|
</a>
|
|
|
|
<a href='about:blank' aria-owns='input p'>
|
|
|
|
<div role='region'>Link 2</div>
|
|
|
|
</a>
|
|
|
|
<input id='input' value='Value'>
|
|
|
|
<p id='p'>Paragraph</p>
|
|
|
|
`);
|
|
|
|
|
|
|
|
// - Different from Chrome DevTools which attributes ownership to the last element.
|
|
|
|
// - CDT also does not include non-owned children in accessible name.
|
|
|
|
// - Disregarding these as aria-owns can't suggest multiple parts by spec.
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- link "Link 1 Value Paragraph":
|
2025-03-11 10:04:52 -07:00
|
|
|
- /url: about:blank
|
2024-11-01 15:25:38 -07:00
|
|
|
- region: Link 1
|
|
|
|
- textbox: Value
|
|
|
|
- paragraph: Paragraph
|
|
|
|
- link "Link 2 Value Paragraph":
|
2025-03-11 10:04:52 -07:00
|
|
|
- /url: about:blank
|
2024-11-01 15:25:38 -07:00
|
|
|
- region: Link 2
|
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should be ok with circular ownership', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<a href='about:blank' id='parent'>
|
|
|
|
<div role='region' aria-owns='parent'>Hello</div>
|
|
|
|
</a>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- link "Hello":
|
2025-03-11 10:04:52 -07:00
|
|
|
- /url: about:blank
|
2024-11-01 15:25:38 -07:00
|
|
|
- region: Hello
|
|
|
|
`);
|
|
|
|
});
|
2024-11-08 10:25:05 -08:00
|
|
|
|
|
|
|
it('should escape yaml text in text nodes', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<details>
|
|
|
|
<summary>one: <a href="#">link1</a> "two <a href="#">link2</a> 'three <a href="#">link3</a> \`four</summary>
|
|
|
|
</details>
|
2024-11-19 17:09:49 -08:00
|
|
|
<ul>
|
|
|
|
<a href="#">one</a>,<a href="#">two</a>
|
|
|
|
(<a href="#">three</a>)
|
|
|
|
{<a href="#">four</a>}
|
|
|
|
[<a href="#">five</a>]
|
|
|
|
</ul>
|
2024-12-19 12:46:54 -08:00
|
|
|
<div>[Select all]</div>
|
2024-11-08 10:25:05 -08:00
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- group:
|
|
|
|
- text: "one:"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "link1":
|
|
|
|
- /url: "#"
|
2024-11-08 10:25:05 -08:00
|
|
|
- text: "\\\"two"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "link2":
|
|
|
|
- /url: "#"
|
2024-11-08 10:25:05 -08:00
|
|
|
- text: "'three"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "link3":
|
|
|
|
- /url: "#"
|
2024-11-08 10:25:05 -08:00
|
|
|
- text: "\`four"
|
2024-11-19 17:09:49 -08:00
|
|
|
- list:
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "one":
|
|
|
|
- /url: "#"
|
2024-11-19 17:09:49 -08:00
|
|
|
- text: ","
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "two":
|
|
|
|
- /url: "#"
|
2024-11-19 17:09:49 -08:00
|
|
|
- text: (
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "three":
|
|
|
|
- /url: "#"
|
2024-11-19 17:09:49 -08:00
|
|
|
- text: ") {"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "four":
|
|
|
|
- /url: "#"
|
2024-11-19 17:09:49 -08:00
|
|
|
- text: "} ["
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "five":
|
|
|
|
- /url: "#"
|
2024-11-19 17:09:49 -08:00
|
|
|
- text: "]"
|
2024-12-19 12:46:54 -08:00
|
|
|
- text: "[Select all]"
|
2024-11-08 10:25:05 -08:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
2025-01-10 19:31:47 +00:00
|
|
|
it('should normalize whitespace', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<details>
|
|
|
|
<summary> one \n two <a href="#"> link \n 1 </a> </summary>
|
|
|
|
</details>
|
|
|
|
<input value=' hello world '>
|
2025-02-25 16:54:02 +00:00
|
|
|
<button>hello\u00ad\u200bworld</button>
|
2025-01-10 19:31:47 +00:00
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- group:
|
|
|
|
- text: one two
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "link 1":
|
|
|
|
- /url: "#"
|
2025-01-10 19:31:47 +00:00
|
|
|
- textbox: hello world
|
2025-02-25 16:54:02 +00:00
|
|
|
- button "helloworld"
|
2025-01-10 19:31:47 +00:00
|
|
|
`);
|
|
|
|
|
|
|
|
// Weird whitespace in the template should be normalized.
|
|
|
|
await expect(page.locator('body')).toMatchAriaSnapshot(`
|
|
|
|
- group:
|
|
|
|
- text: |
|
|
|
|
one
|
|
|
|
two
|
2025-03-11 10:04:52 -07:00
|
|
|
- link " link 1 ":
|
|
|
|
- /url: "#"
|
2025-01-10 19:31:47 +00:00
|
|
|
- textbox: hello world
|
2025-02-25 16:54:02 +00:00
|
|
|
- button "he\u00adlloworld\u200b"
|
2025-01-10 19:31:47 +00:00
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
2024-11-12 02:26:54 -08:00
|
|
|
it('should handle long strings', async ({ page }) => {
|
|
|
|
const s = 'a'.repeat(10000);
|
2024-11-08 10:25:05 -08:00
|
|
|
await page.setContent(`
|
|
|
|
<a href='about:blank'>
|
2024-11-12 02:26:54 -08:00
|
|
|
<div role='region'>${s}</div>
|
2024-11-08 10:25:05 -08:00
|
|
|
</a>
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2024-11-12 02:26:54 -08:00
|
|
|
- link:
|
2025-03-11 10:04:52 -07:00
|
|
|
- /url: about:blank
|
2024-11-12 02:26:54 -08:00
|
|
|
- region: ${s}
|
2024-11-08 10:25:05 -08:00
|
|
|
`);
|
|
|
|
});
|
2024-11-28 11:21:52 +00:00
|
|
|
|
|
|
|
it('should escape special yaml characters', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<a href="#">@hello</a>@hello
|
|
|
|
<a href="#">]hello</a>]hello
|
|
|
|
<a href="#">hello\n</a>
|
|
|
|
hello\n<a href="#">\n hello</a>\n hello
|
|
|
|
<a href="#">#hello</a>#hello
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "@hello":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "@hello"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "]hello":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "]hello"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "hello":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: hello
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "hello":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: hello
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "#hello":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "#hello"
|
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should escape special yaml values', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<a href="#">true</a>False
|
|
|
|
<a href="#">NO</a>yes
|
|
|
|
<a href="#">y</a>N
|
|
|
|
<a href="#">on</a>Off
|
|
|
|
<a href="#">null</a>NULL
|
|
|
|
<a href="#">123</a>123
|
|
|
|
<a href="#">-1.2</a>-1.2
|
2025-01-07 11:14:45 -08:00
|
|
|
<a href="#">-</a>-
|
2024-11-28 11:21:52 +00:00
|
|
|
<input type=text value="555">
|
|
|
|
`);
|
|
|
|
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "true":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "False"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "NO":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "yes"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "y":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "N"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "on":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "Off"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "null":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "NULL"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "123":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "123"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "-1.2":
|
|
|
|
- /url: "#"
|
2024-11-28 11:21:52 +00:00
|
|
|
- text: "-1.2"
|
2025-03-11 10:04:52 -07:00
|
|
|
- link "-":
|
|
|
|
- /url: "#"
|
2025-01-07 11:14:45 -08:00
|
|
|
- text: "-"
|
2024-11-28 11:21:52 +00:00
|
|
|
- textbox: "555"
|
|
|
|
`);
|
|
|
|
});
|
2025-01-29 16:06:07 -08:00
|
|
|
|
|
|
|
it('should not report textarea textContent', async ({ page }) => {
|
|
|
|
await page.setContent(`<textarea>Before</textarea>`);
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- textbox: Before
|
|
|
|
`);
|
|
|
|
await page.evaluate(() => {
|
|
|
|
document.querySelector('textarea').value = 'After';
|
|
|
|
});
|
|
|
|
await checkAndMatchSnapshot(page.locator('body'), `
|
|
|
|
- textbox: After
|
|
|
|
`);
|
|
|
|
});
|
2025-03-10 14:39:37 -07:00
|
|
|
|
|
|
|
it('should generate refs', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<button>One</button>
|
|
|
|
<button>Two</button>
|
|
|
|
<button>Three</button>
|
|
|
|
`);
|
|
|
|
|
2025-05-08 08:33:10 -07:00
|
|
|
const snapshot1 = await page.locator('body').ariaSnapshot(forAI);
|
2025-03-10 14:39:37 -07:00
|
|
|
expect(snapshot1).toContain('- button "One" [ref=s1e3]');
|
|
|
|
expect(snapshot1).toContain('- button "Two" [ref=s1e4]');
|
|
|
|
expect(snapshot1).toContain('- button "Three" [ref=s1e5]');
|
|
|
|
|
|
|
|
await expect(page.locator('aria-ref=s1e3')).toHaveText('One');
|
|
|
|
await expect(page.locator('aria-ref=s1e4')).toHaveText('Two');
|
|
|
|
await expect(page.locator('aria-ref=s1e5')).toHaveText('Three');
|
|
|
|
|
2025-05-08 08:33:10 -07:00
|
|
|
const snapshot2 = await page.locator('body').ariaSnapshot(forAI);
|
2025-03-10 14:39:37 -07:00
|
|
|
expect(snapshot2).toContain('- button "One" [ref=s2e3]');
|
|
|
|
await expect(page.locator('aria-ref=s2e3')).toHaveText('One');
|
|
|
|
|
|
|
|
const e = await expect(page.locator('aria-ref=s1e3')).toHaveText('One').catch(e => e);
|
|
|
|
expect(e.message).toContain('Error: Stale aria-ref, expected s2e{number}, got s1e3');
|
|
|
|
});
|
2025-03-28 12:46:20 +01:00
|
|
|
|
2025-04-08 11:32:59 +02:00
|
|
|
it('should list iframes', async ({ page }) => {
|
2025-03-28 12:46:20 +01:00
|
|
|
await page.setContent(`
|
|
|
|
<h1>Hello</h1>
|
|
|
|
<iframe name="foo" src="data:text/html,<h1>World</h1>">
|
|
|
|
`);
|
|
|
|
|
2025-05-08 08:33:10 -07:00
|
|
|
const snapshot1 = await page.locator('body').ariaSnapshot(forAI);
|
2025-04-08 11:32:59 +02:00
|
|
|
expect(snapshot1).toContain('- iframe');
|
2025-03-28 12:46:20 +01:00
|
|
|
|
2025-04-08 11:32:59 +02:00
|
|
|
const frameSnapshot = await page.frameLocator(`iframe`).locator('body').ariaSnapshot();
|
|
|
|
expect(frameSnapshot).toEqual('- heading "World" [level=1]');
|
2025-03-28 12:46:20 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it('ref mode can be used to stitch all frame snapshots', async ({ page, server }) => {
|
|
|
|
await page.goto(server.PREFIX + '/frames/nested-frames.html');
|
|
|
|
|
|
|
|
async function allFrameSnapshot(frame: Page | FrameLocator): Promise<string> {
|
2025-05-08 08:33:10 -07:00
|
|
|
const snapshot = await frame.locator('body').ariaSnapshot(forAI);
|
2025-03-28 12:46:20 +01:00
|
|
|
const lines = snapshot.split('\n');
|
|
|
|
const result = [];
|
|
|
|
for (const line of lines) {
|
|
|
|
const match = line.match(/^(\s*)- iframe \[ref=(.*)\]/);
|
|
|
|
if (!match) {
|
|
|
|
result.push(line);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const leadingSpace = match[1];
|
|
|
|
const ref = match[2];
|
|
|
|
const childFrame = frame.frameLocator(`aria-ref=${ref}`);
|
|
|
|
const childSnapshot = await allFrameSnapshot(childFrame);
|
|
|
|
result.push(line + ':', childSnapshot.split('\n').map(l => leadingSpace + ' ' + l).join('\n'));
|
|
|
|
}
|
|
|
|
return result.join('\n');
|
|
|
|
}
|
|
|
|
|
2025-05-02 13:28:00 -07:00
|
|
|
expect(await allFrameSnapshot(page)).toContainYaml(`
|
|
|
|
- iframe [ref=s1e3]:
|
|
|
|
- iframe [ref=s1e3]:
|
2025-05-08 08:33:10 -07:00
|
|
|
- generic [ref=s1e3]: Hi, I'm frame
|
2025-05-02 13:28:00 -07:00
|
|
|
- iframe [ref=s1e4]:
|
2025-05-08 08:33:10 -07:00
|
|
|
- generic [ref=s1e3]: Hi, I'm frame
|
2025-05-02 13:28:00 -07:00
|
|
|
- iframe [ref=s1e4]:
|
2025-05-08 08:33:10 -07:00
|
|
|
- generic [ref=s1e3]: Hi, I'm frame
|
2025-05-02 13:28:00 -07:00
|
|
|
`);
|
2025-03-28 12:46:20 +01:00
|
|
|
});
|
2025-04-21 16:07:14 -07:00
|
|
|
|
2025-04-30 17:25:06 -07:00
|
|
|
it('should not generate refs for hidden elements', async ({ page }) => {
|
2025-04-22 14:23:17 -07:00
|
|
|
await page.setContent(`
|
|
|
|
<button>One</button>
|
|
|
|
<button style="width: 0; height: 0; appearance: none; border: 0; padding: 0;">Two</button>
|
|
|
|
<button>Three</button>
|
|
|
|
`);
|
|
|
|
|
2025-05-08 08:33:10 -07:00
|
|
|
const snapshot = await page.locator('body').ariaSnapshot(forAI);
|
2025-05-02 13:28:00 -07:00
|
|
|
expect(snapshot).toContainYaml(`
|
2025-05-08 08:33:10 -07:00
|
|
|
- generic [ref=s1e2]:
|
|
|
|
- button "One" [ref=s1e3]
|
|
|
|
- button "Two"
|
|
|
|
- button "Three" [ref=s1e5]
|
2025-05-02 13:28:00 -07:00
|
|
|
`);
|
2025-04-22 14:23:17 -07:00
|
|
|
});
|
|
|
|
|
2025-04-30 17:25:06 -07:00
|
|
|
it('should not generate refs for elements with pointer-events:none', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<button style="pointer-events: none">no-ref</button>
|
|
|
|
<div style="pointer-events: none">
|
|
|
|
<button style="pointer-events: auto">with-ref</button>
|
|
|
|
</div>
|
|
|
|
<div style="pointer-events: none">
|
|
|
|
<div style="pointer-events: initial">
|
|
|
|
<button>with-ref</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div style="pointer-events: none">
|
|
|
|
<div style="pointer-events: auto">
|
|
|
|
<button>with-ref</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div style="pointer-events: auto">
|
|
|
|
<div style="pointer-events: none">
|
|
|
|
<button>no-ref</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
`);
|
|
|
|
|
2025-05-08 08:33:10 -07:00
|
|
|
const snapshot = await page.locator('body').ariaSnapshot(forAI);
|
2025-05-02 13:28:00 -07:00
|
|
|
expect(snapshot).toContainYaml(`
|
2025-05-08 08:33:10 -07:00
|
|
|
- generic [ref=s1e2]:
|
|
|
|
- button "no-ref"
|
|
|
|
- button "with-ref" [ref=s1e5]
|
|
|
|
- button "with-ref" [ref=s1e8]
|
|
|
|
- button "with-ref" [ref=s1e11]
|
|
|
|
- generic [ref=s1e12]:
|
|
|
|
- generic:
|
|
|
|
- button "no-ref"
|
2025-05-02 13:28:00 -07:00
|
|
|
`);
|
2025-04-30 17:25:06 -07:00
|
|
|
});
|
|
|
|
|
2025-04-21 17:36:10 -07:00
|
|
|
it('emit generic roles for nodes w/o roles', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<style>
|
|
|
|
input {
|
|
|
|
width: 0;
|
|
|
|
height: 0;
|
|
|
|
opacity: 0;
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
<div>
|
|
|
|
<label>
|
2025-04-22 14:23:17 -07:00
|
|
|
<span>
|
|
|
|
<input type="radio" value="Apple" checked="">
|
|
|
|
</span>
|
2025-04-21 17:36:10 -07:00
|
|
|
<span>Apple</span>
|
|
|
|
</label>
|
|
|
|
<label>
|
2025-04-22 14:23:17 -07:00
|
|
|
<span>
|
|
|
|
<input type="radio" value="Pear">
|
|
|
|
</span>
|
2025-04-21 17:36:10 -07:00
|
|
|
<span>Pear</span>
|
|
|
|
</label>
|
|
|
|
<label>
|
2025-04-22 14:23:17 -07:00
|
|
|
<span>
|
|
|
|
<input type="radio" value="Orange">
|
|
|
|
</span>
|
2025-04-21 17:36:10 -07:00
|
|
|
<span>Orange</span>
|
|
|
|
</label>
|
|
|
|
</div>
|
|
|
|
`);
|
|
|
|
|
2025-05-08 08:33:10 -07:00
|
|
|
const snapshot = await page.locator('body').ariaSnapshot(forAI);
|
2025-04-21 17:36:10 -07:00
|
|
|
|
2025-05-02 13:28:00 -07:00
|
|
|
expect(snapshot).toContainYaml(`
|
|
|
|
- generic [ref=s1e5]:
|
|
|
|
- radio "Apple" [checked]
|
|
|
|
- generic [ref=s1e7]: Apple
|
|
|
|
- generic [ref=s1e9]:
|
|
|
|
- radio "Pear"
|
|
|
|
- generic [ref=s1e11]: Pear
|
|
|
|
- generic [ref=s1e13]:
|
|
|
|
- radio "Orange"
|
|
|
|
- generic [ref=s1e15]: Orange
|
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should collapse generic nodes', async ({ page }) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<div>
|
|
|
|
<div>
|
|
|
|
<div>
|
|
|
|
<button>Button</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
`);
|
|
|
|
|
2025-05-08 08:33:10 -07:00
|
|
|
const snapshot = await page.locator('body').ariaSnapshot(forAI);
|
2025-05-02 13:28:00 -07:00
|
|
|
expect(snapshot).toContainYaml(`
|
|
|
|
- button \"Button\" [ref=s1e6]
|
|
|
|
`);
|
2025-04-21 17:36:10 -07:00
|
|
|
});
|