227 lines
7.2 KiB
TypeScript
227 lines
7.2 KiB
TypeScript
|
/*
|
||
|
* Copyright 2024 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 as base, expect } from "../../element-web-test";
|
||
|
import { Client } from "../../pages/client";
|
||
|
import { ElementAppPage } from "../../pages/ElementAppPage";
|
||
|
import { Bot } from "../../pages/bot";
|
||
|
|
||
|
/**
|
||
|
* Set up for pinned message tests.
|
||
|
*/
|
||
|
export const test = base.extend<{
|
||
|
room1Name?: string;
|
||
|
room1: { name: string; roomId: string };
|
||
|
util: Helpers;
|
||
|
}>({
|
||
|
displayName: "Alice",
|
||
|
botCreateOpts: { displayName: "Other User" },
|
||
|
|
||
|
room1Name: "Room 1",
|
||
|
room1: async ({ room1Name: name, app, user, bot }, use) => {
|
||
|
const roomId = await app.client.createRoom({ name, invite: [bot.credentials.userId] });
|
||
|
await use({ name, roomId });
|
||
|
},
|
||
|
|
||
|
util: async ({ page, app, bot }, use) => {
|
||
|
await use(new Helpers(page, app, bot));
|
||
|
},
|
||
|
});
|
||
|
|
||
|
export class Helpers {
|
||
|
constructor(
|
||
|
private page: Page,
|
||
|
private app: ElementAppPage,
|
||
|
private bot: Bot,
|
||
|
) {}
|
||
|
|
||
|
/**
|
||
|
* Sends messages into given room as a bot
|
||
|
* @param room - the name of the room to send messages into
|
||
|
* @param messages - the list of messages to send, these can be strings or implementations of MessageSpec like `editOf`
|
||
|
*/
|
||
|
async receiveMessages(room: string | { name: string }, messages: string[]) {
|
||
|
await this.sendMessageAsClient(this.bot, room, messages);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Use the supplied client to send messages or perform actions as specified by
|
||
|
* the supplied {@link Message} items.
|
||
|
*/
|
||
|
private async sendMessageAsClient(cli: Client, roomName: string | { name: string }, messages: string[]) {
|
||
|
const room = await this.findRoomByName(typeof roomName === "string" ? roomName : roomName.name);
|
||
|
const roomId = await room.evaluate((room) => room.roomId);
|
||
|
|
||
|
for (const message of messages) {
|
||
|
await cli.sendMessage(roomId, { body: message, msgtype: "m.text" });
|
||
|
|
||
|
// TODO: without this wait, some tests that send lots of messages flake
|
||
|
// from time to time. I (andyb) have done some investigation, but it
|
||
|
// needs more work to figure out. The messages do arrive over sync, but
|
||
|
// they never appear in the timeline, and they never fire a
|
||
|
// Room.timeline event. I think this only happens with events that refer
|
||
|
// to other events (e.g. replies), so it might be caused by the
|
||
|
// referring event arriving before the referred-to event.
|
||
|
await this.page.waitForTimeout(100);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Find a room by its name
|
||
|
* @param roomName
|
||
|
* @private
|
||
|
*/
|
||
|
private async findRoomByName(roomName: string) {
|
||
|
return this.app.client.evaluateHandle((cli, roomName) => {
|
||
|
return cli.getRooms().find((r) => r.name === roomName);
|
||
|
}, roomName);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Open the room with the supplied name.
|
||
|
*/
|
||
|
async goTo(room: string | { name: string }) {
|
||
|
await this.app.viewRoomByName(typeof room === "string" ? room : room.name);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Pin the given message
|
||
|
* @param message
|
||
|
*/
|
||
|
async pinMessage(message: string) {
|
||
|
const timelineMessage = this.page.locator(".mx_MTextBody", { hasText: message });
|
||
|
await timelineMessage.click({ button: "right" });
|
||
|
await this.page.getByRole("menuitem", { name: "Pin" }).click();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Pin the given messages
|
||
|
* @param messages
|
||
|
*/
|
||
|
async pinMessages(messages: string[]) {
|
||
|
for (const message of messages) {
|
||
|
await this.pinMessage(message);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Open the room info panel
|
||
|
*/
|
||
|
async openRoomInfo() {
|
||
|
await this.page.getByRole("button", { name: "Room info" }).nth(1).click();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Assert that the pinned count in the room info is correct
|
||
|
* Open the room info and check the pinned count
|
||
|
* @param count
|
||
|
*/
|
||
|
async assertPinnedCountInRoomInfo(count: number) {
|
||
|
await expect(this.page.getByRole("menuitem", { name: "Pinned messages" })).toHaveText(
|
||
|
`Pinned messages${count}`,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Open the pinned messages list
|
||
|
*/
|
||
|
async openPinnedMessagesList() {
|
||
|
await this.page.getByRole("menuitem", { name: "Pinned messages" }).click();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return the right panel
|
||
|
* @private
|
||
|
*/
|
||
|
private getRightPanel() {
|
||
|
return this.page.locator("#mx_RightPanel");
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Assert that the pinned message list contains the given messages
|
||
|
* @param messages
|
||
|
*/
|
||
|
async assertPinnedMessagesList(messages: string[]) {
|
||
|
const rightPanel = this.getRightPanel();
|
||
|
await expect(rightPanel.getByRole("heading", { name: "Pinned messages" })).toHaveText(
|
||
|
`${messages.length} Pinned messages`,
|
||
|
);
|
||
|
await expect(rightPanel).toMatchScreenshot(`pinned-messages-list-messages-${messages.length}.png`);
|
||
|
|
||
|
const list = rightPanel.getByRole("list");
|
||
|
await expect(list.getByRole("listitem")).toHaveCount(messages.length);
|
||
|
|
||
|
for (const message of messages) {
|
||
|
await expect(list.getByText(message)).toBeVisible();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Assert that the pinned message list is empty
|
||
|
*/
|
||
|
async assertEmptyPinnedMessagesList() {
|
||
|
const rightPanel = this.getRightPanel();
|
||
|
await expect(rightPanel).toMatchScreenshot(`pinned-messages-list-empty.png`);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Open the unpin all dialog
|
||
|
*/
|
||
|
async openUnpinAllDialog() {
|
||
|
await this.openRoomInfo();
|
||
|
await this.openPinnedMessagesList();
|
||
|
await this.page.getByRole("button", { name: "Unpin all" }).click();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return the unpin all dialog
|
||
|
*/
|
||
|
getUnpinAllDialog() {
|
||
|
return this.page.locator(".mx_Dialog", { hasText: "Unpin all messages?" });
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Click on the Continue button of the unoin all dialog
|
||
|
*/
|
||
|
async confirmUnpinAllDialog() {
|
||
|
await this.getUnpinAllDialog().getByRole("button", { name: "Continue" }).click();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Go back from the pinned messages list
|
||
|
*/
|
||
|
async backPinnedMessagesList() {
|
||
|
await this.page.locator("#mx_RightPanel").getByTestId("base-card-back-button").click();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Open the contextual menu of a message in the pin message list and click on unpin
|
||
|
* @param message
|
||
|
*/
|
||
|
async unpinMessageFromMessageList(message: string) {
|
||
|
const item = this.getRightPanel().getByRole("list").getByRole("listitem").filter({
|
||
|
hasText: message,
|
||
|
});
|
||
|
|
||
|
await item.getByRole("button").click();
|
||
|
await this.page.getByRole("menu", { name: "Open menu" }).getByRole("menuitem", { name: "Unpin" }).click();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
export { expect };
|