diff --git a/cypress/e2e/left-panel/left-panel.spec.ts b/cypress/e2e/left-panel/left-panel.spec.ts deleted file mode 100644 index 232cffe9cc..0000000000 --- a/cypress/e2e/left-panel/left-panel.spec.ts +++ /dev/null @@ -1,42 +0,0 @@ -/* -Copyright 2023 Suguru Hirahara - -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 { HomeserverInstance } from "../../plugins/utils/homeserver"; - -describe("LeftPanel", () => { - let homeserver: HomeserverInstance; - - beforeEach(() => { - cy.startHomeserver("default").then((data) => { - homeserver = data; - - cy.initTestUser(homeserver, "Hanako"); - }); - }); - - afterEach(() => { - cy.stopHomeserver(homeserver); - }); - - it("should render the Rooms list", () => { - // create rooms and check room names are correct - cy.createRoom({ name: "Apple" }).then(() => cy.findByRole("treeitem", { name: "Apple" })); - cy.createRoom({ name: "Pineapple" }).then(() => cy.findByRole("treeitem", { name: "Pineapple" })); - cy.createRoom({ name: "Orange" }).then(() => cy.findByRole("treeitem", { name: "Orange" })); - }); -}); diff --git a/cypress/e2e/login/consent.spec.ts b/cypress/e2e/login/consent.spec.ts deleted file mode 100644 index 32f4e0f033..0000000000 --- a/cypress/e2e/login/consent.spec.ts +++ /dev/null @@ -1,75 +0,0 @@ -/* -Copyright 2022 The Matrix.org Foundation C.I.C. - -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 { SinonStub } from "cypress/types/sinon"; - -import { HomeserverInstance } from "../../plugins/utils/homeserver"; - -describe("Consent", () => { - let homeserver: HomeserverInstance; - - beforeEach(() => { - cy.startHomeserver("consent").then((data) => { - homeserver = data; - - cy.initTestUser(homeserver, "Bob"); - }); - }); - - afterEach(() => { - cy.stopHomeserver(homeserver); - }); - - it("should prompt the user to consent to terms when server deems it necessary", () => { - // Attempt to create a room using the js-sdk which should return an error with `M_CONSENT_NOT_GIVEN` - cy.window().then((win) => { - win.mxMatrixClientPeg.matrixClient.createRoom({}).catch(() => {}); - - // Stub `window.open` - clicking the primary button below will call it - cy.stub(win, "open").as("windowOpen").returns({}); - }); - - // Accept terms & conditions - cy.get(".mx_QuestionDialog").within(() => { - cy.get("#mx_BaseDialog_title").within(() => { - cy.findByText("Terms and Conditions"); - }); - cy.findByRole("button", { name: "Review terms and conditions" }).click(); - }); - - cy.get("@windowOpen").then((stub) => { - const url = stub.getCall(0).args[0]; - - // Go to Homeserver's consent page and accept it - cy.origin(homeserver.baseUrl, { args: { url } }, ({ url }) => { - cy.visit(url); - - cy.get('[type="submit"]').click(); - cy.contains("p", "Danke schoen"); - }); - }); - - // go back to the app - cy.visit("/"); - // wait for the app to re-load - cy.get(".mx_MatrixChat", { timeout: 15000 }); - - // attempt to perform the same action again and expect it to not fail - cy.createRoom({}); - }); -}); diff --git a/cypress/e2e/user-onboarding/user-onboarding-new.spec.ts b/cypress/e2e/user-onboarding/user-onboarding-new.spec.ts deleted file mode 100644 index 5d79c9c58e..0000000000 --- a/cypress/e2e/user-onboarding/user-onboarding-new.spec.ts +++ /dev/null @@ -1,96 +0,0 @@ -/* -Copyright 2022 The Matrix.org Foundation C.I.C. - -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 { MatrixClient } from "../../global"; -import { HomeserverInstance } from "../../plugins/utils/homeserver"; - -describe("User Onboarding (new user)", () => { - let homeserver: HomeserverInstance; - - const bot1Name = "BotBob"; - let bot1: MatrixClient; - - beforeEach(() => { - cy.startHomeserver("default").then((data) => { - homeserver = data; - cy.initTestUser(homeserver, "Jane Doe"); - cy.window({ log: false }).then((win) => { - win.localStorage.setItem("mx_registration_time", "1656633601"); - }); - cy.reload().then(() => { - // wait for the app to load - return cy.get(".mx_MatrixChat", { timeout: 15000 }); - }); - cy.getBot(homeserver, { displayName: bot1Name }).then((_bot1) => { - bot1 = _bot1; - }); - cy.get(".mx_UserOnboardingPage").should("exist"); - cy.findByRole("button", { name: "Welcome" }).should("exist"); - cy.get(".mx_UserOnboardingList") - .should("exist") - .should(($list) => { - const list = $list.get(0); - expect(getComputedStyle(list).opacity).to.be.eq("1"); - }); - }); - }); - - afterEach(() => { - cy.stopHomeserver(homeserver); - }); - - it("page is shown and preference exists", () => { - cy.get(".mx_UserOnboardingPage").percySnapshotElement("User onboarding page"); - cy.openUserSettings("Preferences"); - cy.findByText("Show shortcut to welcome checklist above the room list").should("exist"); - }); - - it("app download dialog", () => { - cy.findByRole("button", { name: "Download apps" }).click(); - cy.get("[role=dialog]").get("#mx_BaseDialog_title").findByText("Download Element").should("exist"); - cy.get("[role=dialog]").percySnapshotElement("App download dialog", { - widths: [640], - }); - }); - - it("using find friends action should increase progress", () => { - cy.get(".mx_ProgressBar") - .invoke("val") - .then((oldProgress) => { - const findPeopleAction = cy.findByRole("button", { name: "Find friends" }); - expect(findPeopleAction).to.exist; - findPeopleAction.click(); - cy.get(".mx_InviteDialog_editor").findByRole("textbox").type(bot1.getUserId()); - cy.findByRole("button", { name: "Go" }).click(); - cy.get(".mx_InviteDialog_buttonAndSpinner").should("not.exist"); - const message = "Hi!"; - cy.findByRole("textbox", { name: "Send a messageā€¦" }).type(`${message}{enter}`); - cy.get(".mx_MTextBody.mx_EventTile_content").findByText(message); - cy.visit("/#/home"); - cy.get(".mx_UserOnboardingPage").should("exist"); - cy.findByRole("button", { name: "Welcome" }).should("exist"); - cy.get(".mx_UserOnboardingList") - .should("exist") - .should(($list) => { - const list = $list.get(0); - expect(getComputedStyle(list).opacity).to.be.eq("1"); - }); - cy.get(".mx_ProgressBar").invoke("val").should("be.greaterThan", oldProgress); - }); - }); -}); diff --git a/cypress/e2e/user-onboarding/user-onboarding-old.spec.ts b/cypress/e2e/user-onboarding/user-onboarding-old.spec.ts deleted file mode 100644 index 502bdd2a0a..0000000000 --- a/cypress/e2e/user-onboarding/user-onboarding-old.spec.ts +++ /dev/null @@ -1,49 +0,0 @@ -/* -Copyright 2022 The Matrix.org Foundation C.I.C. - -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 { HomeserverInstance } from "../../plugins/utils/homeserver"; - -describe("User Onboarding (old user)", () => { - let homeserver: HomeserverInstance; - - beforeEach(() => { - cy.startHomeserver("default").then((data) => { - homeserver = data; - cy.initTestUser(homeserver, "Jane Doe"); - cy.window({ log: false }).then((win) => { - win.localStorage.setItem("mx_registration_time", "2"); - }); - cy.reload().then(() => { - // wait for the app to load - return cy.get(".mx_MatrixChat", { timeout: 15000 }); - }); - }); - }); - - afterEach(() => { - cy.visit("/#/home"); - cy.stopHomeserver(homeserver); - }); - - it("page and preference are hidden", () => { - cy.get(".mx_UserOnboardingPage").should("not.exist"); - cy.get(".mx_UserOnboardingButton").should("not.exist"); - cy.openUserSettings("Preferences"); - cy.findByText(/Show shortcut to welcome page above the room list/).should("not.exist"); - }); -}); diff --git a/package.json b/package.json index 0cab33910a..dd468c2a32 100644 --- a/package.json +++ b/package.json @@ -46,9 +46,9 @@ "start:all": "echo THIS IS FOR LEGACY PURPOSES ONLY. && yarn start:build", "start:build": "babel src -w -s -d lib --verbose --extensions \".ts,.js\"", "lint": "yarn lint:types && yarn lint:js && yarn lint:style && yarn lint:workflows", - "lint:js": "eslint --max-warnings 0 src test cypress && prettier --check .", - "lint:js-fix": "eslint --fix src test cypress && prettier --loglevel=warn --write .", - "lint:types": "tsc --noEmit --jsx react && tsc --noEmit --jsx react -p cypress", + "lint:js": "eslint --max-warnings 0 src test cypress playwright && prettier --check .", + "lint:js-fix": "eslint --fix src test cypress playwright && prettier --loglevel=warn --write .", + "lint:types": "tsc --noEmit --jsx react && tsc --noEmit --jsx react -p cypress && tsc --noEmit --jsx react -p playwright", "lint:style": "stylelint \"res/css/**/*.pcss\"", "test": "jest", "test:cypress": "cypress run", @@ -226,7 +226,7 @@ "stylelint-config-standard": "^34.0.0", "stylelint-scss": "^5.0.0", "ts-node": "^10.9.1", - "typescript": "5.1.6", + "typescript": "5.2.2", "@axe-core/playwright": "^4.8.1", "@action-validator/core": "^0.5.3", "@action-validator/cli": "^0.5.3" diff --git a/playwright/e2e/left-panel/left-panel.spec.ts b/playwright/e2e/left-panel/left-panel.spec.ts new file mode 100644 index 0000000000..ae0efcad0c --- /dev/null +++ b/playwright/e2e/left-panel/left-panel.spec.ts @@ -0,0 +1,31 @@ +/* +Copyright 2023 Suguru Hirahara + +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 "../../element-web-test"; + +test.describe("LeftPanel", () => { + test.use({ + displayName: "Hanako", + }); + + test("should render the Rooms list", async ({ page, app, user }) => { + // create rooms and check room names are correct + for (const name of ["Apple", "Pineapple", "Orange"]) { + await app.createRoom({ name }); + await expect(page.getByRole("treeitem", { name })).toBeVisible(); + } + }); +}); diff --git a/playwright/e2e/login/consent.spec.ts b/playwright/e2e/login/consent.spec.ts new file mode 100644 index 0000000000..eb966d7141 --- /dev/null +++ b/playwright/e2e/login/consent.spec.ts @@ -0,0 +1,55 @@ +/* +Copyright 2022 The Matrix.org Foundation C.I.C. + +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 { Page } from "@playwright/test"; + +import { test, expect } from "../../element-web-test"; + +test.describe("Consent", () => { + test.use({ + startHomeserverOpts: "consent", + displayName: "Bob", + }); + + test("should prompt the user to consent to terms when server deems it necessary", async ({ + context, + page, + user, + app, + }) => { + // Attempt to create a room using the js-sdk which should return an error with `M_CONSENT_NOT_GIVEN` + await app.createRoom({}).catch(() => {}); + const newPagePromise = new Promise((resolve) => context.once("page", resolve)); + + const dialog = page.locator(".mx_QuestionDialog"); + // Accept terms & conditions + await expect(dialog.getByRole("heading", { name: "Terms and Conditions" })).toBeVisible(); + await page.getByRole("button", { name: "Review terms and conditions" }).click(); + + const newPage = await newPagePromise; + await newPage.locator('[type="submit"]').click(); + await expect(newPage.getByText("Danke schoen")).toBeVisible(); + + // go back to the app + await page.goto("/"); + // wait for the app to re-load + await expect(page.locator(".mx_MatrixChat")).toBeVisible(); + + // attempt to perform the same action again and expect it to not fail + await app.createRoom({ name: "Test Room" }); + await expect(page.getByText("Test Room")).toBeVisible(); + }); +}); diff --git a/playwright/e2e/login/utils.ts b/playwright/e2e/login/utils.ts index 856e3f610b..4bb3023269 100644 --- a/playwright/e2e/login/utils.ts +++ b/playwright/e2e/login/utils.ts @@ -43,7 +43,7 @@ export async function doTokenRegistration( // Synapse prompts us to pick a user ID await expect(page.getByRole("heading", { name: "Create your account" })).toBeVisible(); - await page.getByRole("textbox", { name: "Username (required)" }).type("alice"); + await page.getByRole("textbox", { name: "Username (required)" }).fill("alice"); // wait for username validation to start, and complete await expect(page.locator("#field-username-output")).toHaveText(""); diff --git a/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts new file mode 100644 index 0000000000..106706e965 --- /dev/null +++ b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts @@ -0,0 +1,75 @@ +/* +Copyright 2022 The Matrix.org Foundation C.I.C. + +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 "../../element-web-test"; + +test.describe("User Onboarding (new user)", () => { + test.use({ + displayName: "Jane Doe", + }); + + // This first beforeEach happens before the `user` fixture runs + test.beforeEach(async ({ page }) => { + await page.addInitScript(() => { + window.localStorage.setItem("mx_registration_time", "1656633601"); + }); + }); + + test.beforeEach(async ({ page, user }) => { + await expect(page.locator(".mx_UserOnboardingPage")).toBeVisible(); + await expect(page.getByRole("button", { name: "Welcome" })).toBeVisible(); + await expect(page.locator(".mx_UserOnboardingList")).toBeVisible(); + }); + + test("page is shown and preference exists", async ({ page, app }) => { + await expect(page.locator(".mx_UserOnboardingPage")).toHaveScreenshot(); + await app.openUserSettings("Preferences"); + await expect(page.getByText("Show shortcut to welcome checklist above the room list")).toBeVisible(); + }); + + test("app download dialog", async ({ page }) => { + await page.getByRole("button", { name: "Download apps" }).click(); + await expect( + page.getByRole("dialog").getByRole("heading", { level: 2, name: "Download Element" }), + ).toBeVisible(); + await expect(page.getByRole("dialog")).toHaveScreenshot(); + }); + + test("using find friends action should increase progress", async ({ page, homeserver }) => { + const bot = await homeserver.registerUser("botbob", "password", "BotBob"); + + const oldProgress = parseFloat(await page.getByRole("progressbar").getAttribute("value")); + await page.getByRole("button", { name: "Find friends" }).click(); + await page.locator(".mx_InviteDialog_editor").getByRole("textbox").fill(bot.userId); + await page.getByRole("button", { name: "Go" }).click(); + await expect(page.locator(".mx_InviteDialog_buttonAndSpinner")).not.toBeVisible(); + + const message = "Hi!"; + const composer = page.getByRole("textbox", { name: "Send a messageā€¦" }); + await composer.fill(`${message}`); + await composer.press("Enter"); + await expect(page.locator(".mx_MTextBody.mx_EventTile_content", { hasText: message })).toBeVisible(); + + await page.goto("/#/home"); + await expect(page.locator(".mx_UserOnboardingPage")).toBeVisible(); + await expect(page.getByRole("button", { name: "Welcome" })).toBeVisible(); + await expect(page.locator(".mx_UserOnboardingList")).toBeVisible(); + + await page.waitForTimeout(500); // await progress bar animation + const progress = parseFloat(await page.getByRole("progressbar").getAttribute("value")); + expect(progress).toBeGreaterThan(oldProgress); + }); +}); diff --git a/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-app-download-dialog-1-Legacy-Crypto-linux.png b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-app-download-dialog-1-Legacy-Crypto-linux.png new file mode 100644 index 0000000000..932bfc6c3a Binary files /dev/null and b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-app-download-dialog-1-Legacy-Crypto-linux.png differ diff --git a/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-app-download-dialog-1-Rust-Crypto-linux.png b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-app-download-dialog-1-Rust-Crypto-linux.png new file mode 100644 index 0000000000..932bfc6c3a Binary files /dev/null and b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-app-download-dialog-1-Rust-Crypto-linux.png differ diff --git a/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-page-is-shown-and-preference-exists-1-Legacy-Crypto-linux.png b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-page-is-shown-and-preference-exists-1-Legacy-Crypto-linux.png new file mode 100644 index 0000000000..29f7adc4dd Binary files /dev/null and b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-page-is-shown-and-preference-exists-1-Legacy-Crypto-linux.png differ diff --git a/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-page-is-shown-and-preference-exists-1-Rust-Crypto-linux.png b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-page-is-shown-and-preference-exists-1-Rust-Crypto-linux.png new file mode 100644 index 0000000000..29f7adc4dd Binary files /dev/null and b/playwright/e2e/user-onboarding/user-onboarding-new.spec.ts-snapshots/User-Onboarding-new-user-page-is-shown-and-preference-exists-1-Rust-Crypto-linux.png differ diff --git a/playwright/e2e/user-onboarding/user-onboarding-old.spec.ts b/playwright/e2e/user-onboarding/user-onboarding-old.spec.ts new file mode 100644 index 0000000000..99bbcedd98 --- /dev/null +++ b/playwright/e2e/user-onboarding/user-onboarding-old.spec.ts @@ -0,0 +1,36 @@ +/* +Copyright 2022 The Matrix.org Foundation C.I.C. + +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 "../../element-web-test"; + +test.describe("User Onboarding (old user)", () => { + test.use({ + displayName: "Jane Doe", + }); + + test.beforeEach(async ({ page }) => { + await page.addInitScript(() => { + window.localStorage.setItem("mx_registration_time", "2"); + }); + }); + + test("page and preference are hidden", async ({ page, user, app }) => { + await expect(page.locator(".mx_UserOnboardingPage")).not.toBeVisible(); + await expect(page.locator(".mx_UserOnboardingButton")).not.toBeVisible(); + await app.openUserSettings("Preferences"); + await expect(page.getByText("Show shortcut to welcome checklist above the room list")).not.toBeVisible(); + }); +}); diff --git a/playwright/element-web-test.ts b/playwright/element-web-test.ts index ee1ae67f3a..9592a2d1eb 100644 --- a/playwright/element-web-test.ts +++ b/playwright/element-web-test.ts @@ -23,6 +23,7 @@ import type { IConfigOptions } from "../src/IConfigOptions"; import { Credentials, HomeserverInstance, StartHomeserverOpts } from "./plugins/utils/homeserver"; import { Synapse } from "./plugins/synapse"; import { Instance } from "./plugins/mailhog"; +import { ElementAppPage } from "./pages/ElementAppPage"; import { OAuthServer } from "./plugins/oauth_server"; import { Toasts } from "./pages/toasts"; @@ -60,6 +61,7 @@ export const test = base.extend< displayName: string; }; displayName?: string; + app: ElementAppPage; mailhog?: { api: mailhog.API; instance: Instance }; toasts: Toasts; } @@ -150,6 +152,9 @@ export const test = base.extend< expect(results.violations).toEqual([]); }), + app: async ({ page }, use) => { + await use(new ElementAppPage(page)); + }, toasts: async ({ page }, use) => { await use(new Toasts(page)); }, diff --git a/playwright/global.d.ts b/playwright/global.d.ts new file mode 100644 index 0000000000..784c09cee4 --- /dev/null +++ b/playwright/global.d.ts @@ -0,0 +1,25 @@ +/* +Copyright 2023 The Matrix.org Foundation C.I.C. + +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 { type MatrixClient } from "matrix-js-sdk/src/matrix"; + +declare global { + interface Window { + mxMatrixClientPeg: { + get(): MatrixClient; + }; + } +} diff --git a/playwright/pages/ElementAppPage.ts b/playwright/pages/ElementAppPage.ts new file mode 100644 index 0000000000..a913675f0c --- /dev/null +++ b/playwright/pages/ElementAppPage.ts @@ -0,0 +1,68 @@ +/* +Copyright 2023 The Matrix.org Foundation C.I.C. + +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 { type Locator, type Page } from "@playwright/test"; +import { type ICreateRoomOpts } from "matrix-js-sdk/src/matrix"; + +export class ElementAppPage { + public constructor(private readonly page: Page) {} + + /** + * Open the top left user menu, returning a Locator to the resulting context menu. + */ + public async openUserMenu(): Promise { + await this.page.getByRole("button", { name: "User menu" }).click(); + const locator = this.page.locator(".mx_ContextualMenu"); + await locator.waitFor(); + return locator; + } + + /** + * Switch settings tab to the one by the given name + * @param tab the name of the tab to switch to. + */ + public async switchTab(tab: string): Promise { + await this.page + .locator(".mx_TabbedView_tabLabels") + .locator(".mx_TabbedView_tabLabel", { hasText: tab }) + .click(); + } + + /** + * Open user settings (via user menu), returns a locator to the dialog + * @param tab the name of the tab to switch to after opening, optional. + */ + public async openUserSettings(tab?: string): Promise { + const locator = await this.openUserMenu(); + await locator.getByRole("menuitem", { name: "All settings", exact: true }).click(); + if (tab) await this.switchTab(tab); + return this.page.locator(".mx_UserSettingsDialog"); + } + + /** + * Create a room with given options. + * @param options the options to apply when creating the room + * @return the ID of the newly created room + */ + public async createRoom(options: ICreateRoomOpts): Promise { + return this.page.evaluate, ICreateRoomOpts>(async (options) => { + return window.mxMatrixClientPeg + .get() + .createRoom(options) + .then((res) => res.room_id); + }, options); + } +} diff --git a/playwright/plugins/synapse/index.ts b/playwright/plugins/synapse/index.ts index 432fe006d7..3ac861cd18 100644 --- a/playwright/plugins/synapse/index.ts +++ b/playwright/plugins/synapse/index.ts @@ -193,6 +193,10 @@ export class Synapse implements Homeserver, HomeserverInstance { }, }); + if (!res.ok()) { + throw await res.json(); + } + const data = await res.json(); return { homeServer: data.home_server, diff --git a/playwright/tsconfig.json b/playwright/tsconfig.json index 7651d24e97..cea25ebab7 100644 --- a/playwright/tsconfig.json +++ b/playwright/tsconfig.json @@ -2,11 +2,15 @@ "compilerOptions": { "target": "es2016", "jsx": "react", - "lib": ["es2021", "dom", "dom.iterable"], + "lib": ["ESNext", "es2021", "dom", "dom.iterable"], "resolveJsonModule": true, "esModuleInterop": true, "moduleResolution": "node", "module": "es2022" }, - "include": ["**/*.ts", "../src/@types/global.d.ts"] + "include": [ + "**/*.ts", + "../node_modules/matrix-js-sdk/src/@types/*.d.ts", + "../node_modules/matrix-js-sdk/node_modules/@matrix-org/olm/index.d.ts" + ] } diff --git a/yarn.lock b/yarn.lock index d0dd55aeda..20f71f9443 100644 --- a/yarn.lock +++ b/yarn.lock @@ -9986,10 +9986,10 @@ typed-array-length@^1.0.4: for-each "^0.3.3" is-typed-array "^1.1.9" -typescript@5.1.6: - version "5.1.6" - resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.1.6.tgz#02f8ac202b6dad2c0dd5e0913745b47a37998274" - integrity sha512-zaWCozRZ6DLEWAWFrVDz1H6FVXzUSfTy5FUMWsQlU8Ym5JP9eO4xkTIROFCQvhQf61z6O/G6ugw3SgAnvvm+HA== +typescript@5.2.2: + version "5.2.2" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.2.2.tgz#5ebb5e5a5b75f085f22bc3f8460fba308310fa78" + integrity sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w== ua-parser-js@^1.0.2: version "1.0.37"