diff --git a/e2e-tests/code-editing-and-ast-interaction.test.ts b/e2e-tests/code-editing-and-ast-interaction.test.ts index 3d683bd..dd3f9b8 100644 --- a/e2e-tests/code-editing-and-ast-interaction.test.ts +++ b/e2e-tests/code-editing-and-ast-interaction.test.ts @@ -127,3 +127,1230 @@ test(`should keep ESQuery highlights aligned while typing before a matching lite ).toBe(true); await expect(highlight).toHaveText(["42"]); }); + +test.describe("AST node expansion", () => { + test.beforeEach(async ({ page }) => { + await page.goto("/"); + await page.getByRole("button", { name: "Language Options" }).click(); + }); + + test.describe("Language: JavaScript", () => { + test.beforeEach(async ({ page }) => { + // `Language`: select `JavaScript` + await page + .getByRole("combobox", { exact: true, name: "Language" }) + .click(); + await page + .getByRole("option", { exact: true, name: "JavaScript" }) + .click(); + }); + + test("SourceType: Module", async ({ page }) => { + // `Source Type`: `Module` + const sourceTypeSelect = page.getByRole("combobox", { + exact: true, + name: "Source Type", + }); + await sourceTypeSelect.click(); + await page + .getByRole("option", { exact: true, name: "Module" }) + .click(); + await expect(sourceTypeSelect).toHaveText("Module"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a JavaScript module sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill('import x from "x";'); + + // Verify that the AST structure matches expectations for module source type. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. ImportDeclaration" }), + ).toBeVisible(); + }); + + test("SourceType: CommonJS", async ({ page }) => { + // `Source Type`: `CommonJS` + const sourceTypeSelect = page.getByRole("combobox", { + exact: true, + name: "Source Type", + }); + await sourceTypeSelect.click(); + await page + .getByRole("option", { exact: true, name: "CommonJS" }) + .click(); + await expect(sourceTypeSelect).toHaveText("CommonJS"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a CommonJS sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("return;"); + + // Verify that the AST structure matches expectations for CommonJS source type. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. ReturnStatement" }), + ).toBeVisible(); + }); + + test("SourceType: Script", async ({ page }) => { + // `Source Type`: `Script` + const sourceTypeSelect = page.getByRole("combobox", { + exact: true, + name: "Source Type", + }); + await sourceTypeSelect.click(); + await page + .getByRole("option", { exact: true, name: "Script" }) + .click(); + await expect(sourceTypeSelect).toHaveText("Script"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a script sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("with (x) {}"); + + // Verify that the AST structure matches expectations for script source type. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. WithStatement" }), + ).toBeVisible(); + }); + + test("ECMAScript Version: 2026", async ({ page }) => { + // `ECMAScript Version`: `2026` + const esVersionSelect = page.getByRole("combobox", { + exact: true, + name: "ECMAScript Version", + }); + await esVersionSelect.click(); + await page + .getByRole("option", { exact: true, name: "2026" }) + .click(); + await expect(esVersionSelect).toHaveText("2026"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a JavaScript sample with using syntax into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("using x = y;"); + + // Verify that the AST structure matches expectations for using syntax. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page + .getByRole("button", { name: "0. VariableDeclaration" }) + .click(); + + await expect( + page + .getByRole("region", { name: "0. VariableDeclaration" }) + .getByRole("listitem") + .filter({ hasText: "kindusing" }), + ).toBeVisible(); + }); + + test("JSX: true", async ({ page }) => { + // `JSX`: `true` + const jsxSwitch = page.getByRole("switch", { + exact: true, + name: "JSX", + }); + await expect(jsxSwitch).toHaveAttribute("aria-checked", "true"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a JSX sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill(";"); + + // Verify that the AST structure matches expectations for JSX. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page + .getByRole("button", { name: "0. ExpressionStatement" }) + .click(); + + await expect( + page + .getByRole("region", { name: "0. ExpressionStatement" }) + .getByRole("listitem") + .filter({ + hasText: "expressionJSXElement{type, start, end, ...}", + }), + ).toBeVisible(); + }); + }); + + test.describe("Language: JSON", () => { + test.beforeEach(async ({ page }) => { + // `Language`: select `JSON` + await page + .getByRole("combobox", { exact: true, name: "Language" }) + .click(); + await page + .getByRole("option", { exact: true, name: "JSON" }) + .click(); + }); + + test("Mode: JSON", async ({ page }) => { + // `Mode`: `JSON` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "json" }) + .click(); + await expect(modeSelect).toHaveText("json"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a JSON sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill('{"foo":"bar"}'); + + // Verify that the AST structure matches expectations for JSON. + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "bodyObject{type, members, loc, ...}" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "membersArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Member" }), + ).toBeVisible(); + + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "tokensArray[5 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. LBrace" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. String" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "2. Colon" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "3. String" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "4. RBrace" }), + ).toBeVisible(); + }); + + test("Mode: JSONC", async ({ page }) => { + // `Mode`: `JSONC` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "jsonc" }) + .click(); + await expect(modeSelect).toHaveText("jsonc"); + + // `Allow Trailing Commas`: `false` + const allowTrailingCommasSwitch = page.getByRole("switch", { + exact: true, + name: "Allow Trailing Commas", + }); + await expect(allowTrailingCommasSwitch).toHaveAttribute( + "aria-checked", + "false", + ); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a JSONC sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill('{\n// comment\n"foo":"bar"\n}'); + + // Verify that the AST structure matches expectations for JSONC. + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "bodyObject{type, members, loc, ...}" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "membersArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Member" }), + ).toBeVisible(); + + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "tokensArray[6 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. LBrace" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. LineComment" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "2. String" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "3. Colon" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "4. String" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "5. RBrace" }), + ).toBeVisible(); + }); + + test("Mode: JSON5", async ({ page }) => { + // `Mode`: `JSON5` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "json5" }) + .click(); + await expect(modeSelect).toHaveText("json5"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a JSON5 sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill('{foo:"bar",}'); + + // Verify that the AST structure matches expectations for JSON5. + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "bodyObject{type, members, loc, ...}" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "membersArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Member" }), + ).toBeVisible(); + + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "tokensArray[6 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. LBrace" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. Identifier" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "2. Colon" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "3. String" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "4. Comma" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "5. RBrace" }), + ).toBeVisible(); + }); + + test("Allow Trailing Commas: true", async ({ page }) => { + // `Mode`: `JSONC` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "jsonc" }) + .click(); + await expect(modeSelect).toHaveText("jsonc"); + + // `Allow Trailing Commas`: `true` + const allowTrailingCommasSwitch = page.getByRole("switch", { + exact: true, + name: "Allow Trailing Commas", + }); + await allowTrailingCommasSwitch.click(); + await expect(allowTrailingCommasSwitch).toHaveAttribute( + "aria-checked", + "true", + ); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a JSONC sample with a trailing comma into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill('{\n// comment\n"foo":"bar",\n}'); + + // Verify that the AST structure matches expectations for JSONC with trailing commas. + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "bodyObject{type, members, loc, ...}" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "membersArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Member" }), + ).toBeVisible(); + + await page + .getByRole("region", { name: "Document" }) + .getByRole("listitem") + .filter({ hasText: "tokensArray[7 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. LBrace" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. LineComment" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "2. String" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "3. Colon" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "4. String" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "5. Comma" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "6. RBrace" }), + ).toBeVisible(); + }); + }); + + test.describe("Language: Markdown", () => { + test.beforeEach(async ({ page }) => { + // `Language`: select `Markdown` + await page + .getByRole("combobox", { exact: true, name: "Language" }) + .click(); + await page + .getByRole("option", { exact: true, name: "Markdown" }) + .click(); + }); + + test("Mode: CommonMark", async ({ page }) => { + // `Mode`: `CommonMark` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "CommonMark" }) + .click(); + await expect(modeSelect).toHaveText("CommonMark"); + + // `Front Matter`: `Off` + const frontMatterSelect = page.getByRole("combobox", { + exact: true, + name: "Front Matter", + }); + await frontMatterSelect.click(); + await page + .getByRole("option", { exact: true, name: "Off" }) + .click(); + await expect(frontMatterSelect).toHaveText("Off"); + + // `Math`: `false` + const mathSwitch = page.getByRole("switch", { + exact: true, + name: "Math", + }); + await expect(mathSwitch).toHaveAttribute("aria-checked", "false"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a CommonMark sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("text, *emphasis*, **strong**"); + + // Verify that the AST structure matches expectations for CommonMark. + await page + .getByRole("region", { name: "root" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. paragraph" }).click(); + + await page + .getByRole("region", { name: "0. paragraph" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[4 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. text" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. emphasis" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "2. text" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "3. strong" }), + ).toBeVisible(); + }); + + test("Mode: GFM", async ({ page }) => { + // `Mode`: `GFM` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "GitHub-Flavored" }) + .click(); + await expect(modeSelect).toHaveText("GitHub-Flavored"); + + // `Front Matter`: `Off` + const frontMatterSelect = page.getByRole("combobox", { + exact: true, + name: "Front Matter", + }); + await frontMatterSelect.click(); + await page + .getByRole("option", { exact: true, name: "Off" }) + .click(); + await expect(frontMatterSelect).toHaveText("Off"); + + // `Math`: `false` + const mathSwitch = page.getByRole("switch", { + exact: true, + name: "Math", + }); + await expect(mathSwitch).toHaveAttribute("aria-checked", "false"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a GFM sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("text, ~~delete~~, *emphasis*"); + + // Verify that the AST structure matches expectations for GFM. + await page + .getByRole("region", { name: "root" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. paragraph" }).click(); + + await page + .getByRole("region", { name: "0. paragraph" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[4 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. text" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. delete" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "2. text" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "3. emphasis" }), + ).toBeVisible(); + }); + + test("Frontmatter: YAML", async ({ page }) => { + // `Mode`: `CommonMark` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "CommonMark" }) + .click(); + await expect(modeSelect).toHaveText("CommonMark"); + + // `Front Matter`: `YAML` + const frontMatterSelect = page.getByRole("combobox", { + exact: true, + name: "Front Matter", + }); + await frontMatterSelect.click(); + await page + .getByRole("option", { exact: true, name: "YAML" }) + .click(); + await expect(frontMatterSelect).toHaveText("YAML"); + + // `Math`: `false` + const mathSwitch = page.getByRole("switch", { + exact: true, + name: "Math", + }); + await expect(mathSwitch).toHaveAttribute("aria-checked", "false"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a Markdown sample with YAML frontmatter into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("---\ntitle: Test\n---\n\ntext"); + + // Verify that the AST structure matches expectations for YAML frontmatter. + await page + .getByRole("region", { name: "root" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[2 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. yaml" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. paragraph" }), + ).toBeVisible(); + }); + + test("Frontmatter: TOML", async ({ page }) => { + // `Mode`: `CommonMark` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "CommonMark" }) + .click(); + await expect(modeSelect).toHaveText("CommonMark"); + + // `Front Matter`: `TOML` + const frontMatterSelect = page.getByRole("combobox", { + exact: true, + name: "Front Matter", + }); + await frontMatterSelect.click(); + await page + .getByRole("option", { exact: true, name: "TOML" }) + .click(); + await expect(frontMatterSelect).toHaveText("TOML"); + + // `Math`: `false` + const mathSwitch = page.getByRole("switch", { + exact: true, + name: "Math", + }); + await expect(mathSwitch).toHaveAttribute("aria-checked", "false"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a Markdown sample with TOML frontmatter into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("+++\ncount = 1\n+++\n\ntext"); + + // Verify that the AST structure matches expectations for TOML frontmatter. + await page + .getByRole("region", { name: "root" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[2 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. toml" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. paragraph" }), + ).toBeVisible(); + }); + + test("Frontmatter: JSON", async ({ page }) => { + // `Mode`: `CommonMark` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "CommonMark" }) + .click(); + await expect(modeSelect).toHaveText("CommonMark"); + + // `Front Matter`: `JSON` + const frontMatterSelect = page.getByRole("combobox", { + exact: true, + name: "Front Matter", + }); + await frontMatterSelect.click(); + await page + .getByRole("option", { exact: true, name: "JSON" }) + .click(); + await expect(frontMatterSelect).toHaveText("JSON"); + + // `Math`: `false` + const mathSwitch = page.getByRole("switch", { + exact: true, + name: "Math", + }); + await expect(mathSwitch).toHaveAttribute("aria-checked", "false"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a Markdown sample with JSON frontmatter into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill('---\n{"count":1}\n---\n\ntext'); + + // Verify that the AST structure matches expectations for JSON frontmatter. + await page + .getByRole("region", { name: "root" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[2 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. json" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. paragraph" }), + ).toBeVisible(); + }); + + test("Math: true", async ({ page }) => { + // `Mode`: `CommonMark` + const modeSelect = page.getByRole("combobox", { + exact: true, + name: "Mode", + }); + await modeSelect.click(); + await page + .getByRole("option", { exact: true, name: "CommonMark" }) + .click(); + await expect(modeSelect).toHaveText("CommonMark"); + + // `Front Matter`: `Off` + const frontMatterSelect = page.getByRole("combobox", { + exact: true, + name: "Front Matter", + }); + await frontMatterSelect.click(); + await page + .getByRole("option", { exact: true, name: "Off" }) + .click(); + await expect(frontMatterSelect).toHaveText("Off"); + + // `Math`: `true` + const mathSwitch = page.getByRole("switch", { + exact: true, + name: "Math", + }); + await mathSwitch.click(); + await expect(mathSwitch).toHaveAttribute("aria-checked", "true"); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a Markdown sample with inline and block math into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("inline $x + y$ math\n\n$$\nx + y\n$$"); + + // Verify that the AST structure matches expectations for math. + await page + .getByRole("region", { name: "root" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[2 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. paragraph" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. math" }), + ).toBeVisible(); + + await page.getByRole("button", { name: "0. paragraph" }).click(); + + await page + .getByRole("region", { name: "0. paragraph" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[3 elements]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. text" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "1. inlineMath" }), + ).toBeVisible(); + await expect( + page.getByRole("button", { name: "2. text" }), + ).toBeVisible(); + }); + }); + + test.describe("Language: CSS", () => { + test.beforeEach(async ({ page }) => { + // `Language`: select `CSS` + await page + .getByRole("combobox", { exact: true, name: "Language" }) + .click(); + await page + .getByRole("option", { exact: true, name: "CSS" }) + .click(); + }); + + test("Tolerant Parsing: false", async ({ page }) => { + // `Tolerant Parsing`: `false` + const tolerantSwitch = page.getByRole("switch", { + exact: true, + name: "Tolerant Parsing", + }); + await expect(tolerantSwitch).toHaveAttribute( + "aria-checked", + "false", + ); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a CSS sample into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("a{}"); + + // Verify that the AST structure matches expectations for CSS. + await page + .getByRole("region", { name: "StyleSheet" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Rule" }), + ).toBeVisible(); + }); + + test("Tolerant Parsing: true", async ({ page }) => { + // `Tolerant Parsing`: `true` + const tolerantSwitch = page.getByRole("switch", { + exact: true, + name: "Tolerant Parsing", + }); + await tolerantSwitch.click(); + await expect(tolerantSwitch).toHaveAttribute( + "aria-checked", + "true", + ); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill a CSS sample that requires tolerant parsing into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("a{"); + + // Verify that the AST structure matches expectations for tolerant CSS. + await page + .getByRole("region", { name: "StyleSheet" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Rule" }), + ).toBeVisible(); + }); + }); + + test.describe("Language: HTML", () => { + test.beforeEach(async ({ page }) => { + // `Language`: select `HTML` + await page + .getByRole("combobox", { exact: true, name: "Language" }) + .click(); + await page + .getByRole("option", { exact: true, name: "HTML" }) + .click(); + }); + + test("Template Engine Syntax: Handlebars", async ({ page }) => { + // `Template Engine Syntax`: `Handlebars` + const templateEngineSyntaxSelect = page.getByRole("combobox", { + exact: true, + name: "Template Engine Syntax", + }); + await templateEngineSyntaxSelect.click(); + await page + .getByRole("option", { exact: true, name: "Handlebars" }) + .click(); + await expect(templateEngineSyntaxSelect).toHaveText("Handlebars"); + + // `Front Matter`: `false` + const frontMatterSwitch = page.getByRole("switch", { + exact: true, + name: "Front Matter", + }); + await expect(frontMatterSwitch).toHaveAttribute( + "aria-checked", + "false", + ); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill an HTML sample with Handlebars syntax into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("

{{x}}

"); + + // Verify that the AST structure matches expectations for Handlebars syntax. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Document" }).click(); + + await page + .getByRole("region", { name: "0. Document" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Tag" }).click(); + + await page + .getByRole("region", { name: "0. Tag" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Text" }).click(); + + await page + .getByRole("region", { name: "0. Text" }) + .getByRole("listitem") + .filter({ hasText: "partsArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Template" }), + ).toBeVisible(); + }); + + test("Template Engine Syntax: Twig", async ({ page }) => { + // `Template Engine Syntax`: `Twig` + const templateEngineSyntaxSelect = page.getByRole("combobox", { + exact: true, + name: "Template Engine Syntax", + }); + await templateEngineSyntaxSelect.click(); + await page + .getByRole("option", { exact: true, name: "Twig" }) + .click(); + await expect(templateEngineSyntaxSelect).toHaveText("Twig"); + + // `Front Matter`: `false` + const frontMatterSwitch = page.getByRole("switch", { + exact: true, + name: "Front Matter", + }); + await expect(frontMatterSwitch).toHaveAttribute( + "aria-checked", + "false", + ); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill an HTML sample with Twig syntax into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("

{%x%}

"); + + // Verify that the AST structure matches expectations for Twig syntax. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Document" }).click(); + + await page + .getByRole("region", { name: "0. Document" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Tag" }).click(); + + await page + .getByRole("region", { name: "0. Tag" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Text" }).click(); + + await page + .getByRole("region", { name: "0. Text" }) + .getByRole("listitem") + .filter({ hasText: "partsArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Template" }), + ).toBeVisible(); + }); + + test("Template Engine Syntax: ERB", async ({ page }) => { + // `Template Engine Syntax`: `ERB` + const templateEngineSyntaxSelect = page.getByRole("combobox", { + exact: true, + name: "Template Engine Syntax", + }); + await templateEngineSyntaxSelect.click(); + await page + .getByRole("option", { exact: true, name: "ERB" }) + .click(); + await expect(templateEngineSyntaxSelect).toHaveText("ERB"); + + // `Front Matter`: `false` + const frontMatterSwitch = page.getByRole("switch", { + exact: true, + name: "Front Matter", + }); + await expect(frontMatterSwitch).toHaveAttribute( + "aria-checked", + "false", + ); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill an HTML sample with ERB syntax into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("

<%x%>

"); + + // Verify that the AST structure matches expectations for ERB syntax. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Document" }).click(); + + await page + .getByRole("region", { name: "0. Document" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Tag" }).click(); + + await page + .getByRole("region", { name: "0. Tag" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Text" }).click(); + + await page + .getByRole("region", { name: "0. Text" }) + .getByRole("listitem") + .filter({ hasText: "partsArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Template" }), + ).toBeVisible(); + }); + + test("Front Matter: true", async ({ page }) => { + // `Template Engine Syntax`: `None` + const templateEngineSyntaxSelect = page.getByRole("combobox", { + exact: true, + name: "Template Engine Syntax", + }); + await expect(templateEngineSyntaxSelect).toHaveText("None"); + + // `Front Matter`: `true` + const frontMatterSwitch = page.getByRole("switch", { + exact: true, + name: "Front Matter", + }); + await frontMatterSwitch.click(); + await expect(frontMatterSwitch).toHaveAttribute( + "aria-checked", + "true", + ); + + // Hide the settings menu to ensure it doesn't interfere with the test. + await page.keyboard.press("Escape"); + + // Fill an HTML sample with front matter into the editor. + await page + .getByRole("textbox", { exact: true, name: "Code Editor" }) + .fill("---\ntitle: Test\n---\n

x

"); + + // Verify that the AST structure matches expectations for HTML front matter. + await page + .getByRole("region", { name: "Program" }) + .getByRole("listitem") + .filter({ hasText: "bodyArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await page.getByRole("button", { name: "0. Document" }).click(); + + await page + .getByRole("region", { name: "0. Document" }) + .getByRole("listitem") + .filter({ hasText: "childrenArray[1 element]" }) + .getByRole("button", { name: "Toggle Property" }) + .click(); + + await expect( + page.getByRole("button", { name: "0. Tag" }), + ).toBeVisible(); + }); + }); +});