2020-01-22 11:56:27 +00:00
|
|
|
/*
|
2024-09-09 13:57:16 +00:00
|
|
|
Copyright 2024 New Vector Ltd.
|
2020-01-22 11:56:27 +00:00
|
|
|
Copyright 2020 The Matrix.org Foundation C.I.C.
|
|
|
|
|
2024-09-09 13:57:16 +00:00
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
|
|
Please see LICENSE files in the repository root for full details.
|
2020-01-22 11:56:27 +00:00
|
|
|
*/
|
|
|
|
|
2020-10-06 23:10:40 +00:00
|
|
|
import React from "react";
|
2024-10-14 16:11:58 +00:00
|
|
|
import { fireEvent, render, waitFor } from "jest-matrix-react";
|
2023-03-23 11:47:40 +00:00
|
|
|
import { IContent, MatrixClient, MsgType } from "matrix-js-sdk/src/matrix";
|
2022-07-13 05:56:36 +00:00
|
|
|
import { mocked } from "jest-mock";
|
2023-05-09 10:56:12 +00:00
|
|
|
import userEvent from "@testing-library/user-event";
|
2021-07-01 21:55:27 +00:00
|
|
|
|
2020-11-17 22:36:58 +00:00
|
|
|
import SendMessageComposer, {
|
2023-03-23 11:47:40 +00:00
|
|
|
attachMentions,
|
2020-11-17 22:36:58 +00:00
|
|
|
createMessageContent,
|
|
|
|
isQuickReaction,
|
2024-10-15 13:57:26 +00:00
|
|
|
} from "../../../../../src/components/views/rooms/SendMessageComposer";
|
|
|
|
import MatrixClientContext from "../../../../../src/contexts/MatrixClientContext";
|
|
|
|
import RoomContext, { TimelineRenderingType } from "../../../../../src/contexts/RoomContext";
|
|
|
|
import EditorModel from "../../../../../src/editor/model";
|
2023-03-09 12:15:48 +00:00
|
|
|
import { createPartCreator } from "../../../editor/mock";
|
2024-10-15 13:57:26 +00:00
|
|
|
import { createTestClient, mkEvent, mkStubRoom, stubClient } from "../../../../test-utils";
|
|
|
|
import { MatrixClientPeg } from "../../../../../src/MatrixClientPeg";
|
|
|
|
import defaultDispatcher from "../../../../../src/dispatcher/dispatcher";
|
|
|
|
import DocumentOffset from "../../../../../src/editor/offset";
|
|
|
|
import { Layout } from "../../../../../src/settings/enums/Layout";
|
|
|
|
import { IRoomState, MainSplitContentType } from "../../../../../src/components/structures/RoomView";
|
|
|
|
import { mockPlatformPeg } from "../../../../test-utils/platform";
|
|
|
|
import { doMaybeLocalRoomAction } from "../../../../../src/utils/local-room";
|
|
|
|
import { addTextToComposer } from "../../../../test-utils/composer";
|
2022-07-13 05:56:36 +00:00
|
|
|
|
2024-10-15 13:57:26 +00:00
|
|
|
jest.mock("../../../../../src/utils/local-room", () => ({
|
2022-07-13 05:56:36 +00:00
|
|
|
doMaybeLocalRoomAction: jest.fn(),
|
|
|
|
}));
|
2022-03-18 10:07:33 +00:00
|
|
|
|
2022-12-12 11:24:14 +00:00
|
|
|
describe("<SendMessageComposer/>", () => {
|
2022-06-10 18:41:05 +00:00
|
|
|
const defaultRoomContext: IRoomState = {
|
2021-10-01 13:35:54 +00:00
|
|
|
roomLoading: true,
|
|
|
|
peekLoading: false,
|
|
|
|
shouldPeek: true,
|
|
|
|
membersLoaded: false,
|
|
|
|
numUnreadMessages: 0,
|
|
|
|
canPeek: false,
|
|
|
|
showApps: false,
|
|
|
|
isPeeking: false,
|
|
|
|
showRightPanel: true,
|
|
|
|
joining: false,
|
|
|
|
atEndOfLiveTimeline: true,
|
|
|
|
showTopUnreadMessagesBar: false,
|
|
|
|
statusBarVisible: false,
|
|
|
|
canReact: false,
|
2022-03-04 15:53:22 +00:00
|
|
|
canSendMessages: false,
|
2021-10-01 13:35:54 +00:00
|
|
|
layout: Layout.Group,
|
|
|
|
lowBandwidth: false,
|
|
|
|
alwaysShowTimestamps: false,
|
|
|
|
showTwelveHourTimestamps: false,
|
2024-09-02 09:07:07 +00:00
|
|
|
userTimezone: undefined,
|
2021-10-01 13:35:54 +00:00
|
|
|
readMarkerInViewThresholdMs: 3000,
|
|
|
|
readMarkerOutOfViewThresholdMs: 30000,
|
2022-04-14 23:23:22 +00:00
|
|
|
showHiddenEvents: false,
|
2021-10-01 13:35:54 +00:00
|
|
|
showReadReceipts: true,
|
|
|
|
showRedactions: true,
|
|
|
|
showJoinLeaves: true,
|
|
|
|
showAvatarChanges: true,
|
|
|
|
showDisplaynameChanges: true,
|
|
|
|
matrixClientIsReady: false,
|
|
|
|
timelineRenderingType: TimelineRenderingType.Room,
|
2024-10-04 09:41:00 +00:00
|
|
|
mainSplitContentType: MainSplitContentType.Timeline,
|
2021-10-01 13:35:54 +00:00
|
|
|
liveTimeline: undefined,
|
2022-06-10 18:41:05 +00:00
|
|
|
canSelfRedact: false,
|
|
|
|
resizing: false,
|
|
|
|
narrow: false,
|
2022-10-07 02:27:28 +00:00
|
|
|
activeCall: null,
|
2023-02-10 07:40:38 +00:00
|
|
|
msc3946ProcessDynamicPredecessor: false,
|
2023-08-07 06:27:09 +00:00
|
|
|
canAskToJoin: false,
|
|
|
|
promptAskToJoin: false,
|
2023-11-01 12:03:10 +00:00
|
|
|
viewRoomOpts: { buttons: [] },
|
2021-10-01 13:35:54 +00:00
|
|
|
};
|
2020-01-22 11:56:27 +00:00
|
|
|
describe("createMessageContent", () => {
|
|
|
|
it("sends plaintext messages correctly", () => {
|
2023-03-09 12:15:48 +00:00
|
|
|
const model = new EditorModel([], createPartCreator());
|
2021-10-01 13:35:54 +00:00
|
|
|
const documentOffset = new DocumentOffset(11, true);
|
|
|
|
model.update("hello world", "insertText", documentOffset);
|
2020-01-22 11:56:27 +00:00
|
|
|
|
2024-11-13 14:11:20 +00:00
|
|
|
const content = createMessageContent("@alice:test", model, undefined, undefined);
|
2020-01-22 11:56:27 +00:00
|
|
|
|
|
|
|
expect(content).toEqual({
|
2023-09-14 11:36:15 +00:00
|
|
|
"body": "hello world",
|
|
|
|
"msgtype": "m.text",
|
|
|
|
"m.mentions": {},
|
2020-01-22 11:56:27 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("sends markdown messages correctly", () => {
|
2023-03-09 12:15:48 +00:00
|
|
|
const model = new EditorModel([], createPartCreator());
|
2021-10-01 13:35:54 +00:00
|
|
|
const documentOffset = new DocumentOffset(13, true);
|
|
|
|
model.update("hello *world*", "insertText", documentOffset);
|
2020-01-22 11:56:27 +00:00
|
|
|
|
2024-11-13 14:11:20 +00:00
|
|
|
const content = createMessageContent("@alice:test", model, undefined, undefined);
|
2020-01-22 11:56:27 +00:00
|
|
|
|
|
|
|
expect(content).toEqual({
|
2023-09-14 11:36:15 +00:00
|
|
|
"body": "hello *world*",
|
|
|
|
"msgtype": "m.text",
|
|
|
|
"format": "org.matrix.custom.html",
|
|
|
|
"formatted_body": "hello <em>world</em>",
|
|
|
|
"m.mentions": {},
|
2020-01-22 11:56:27 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("strips /me from messages and marks them as m.emote accordingly", () => {
|
2023-03-09 12:15:48 +00:00
|
|
|
const model = new EditorModel([], createPartCreator());
|
2021-10-01 13:35:54 +00:00
|
|
|
const documentOffset = new DocumentOffset(22, true);
|
|
|
|
model.update("/me blinks __quickly__", "insertText", documentOffset);
|
2020-01-22 11:56:27 +00:00
|
|
|
|
2024-11-13 14:11:20 +00:00
|
|
|
const content = createMessageContent("@alice:test", model, undefined, undefined);
|
2020-01-22 11:56:27 +00:00
|
|
|
|
|
|
|
expect(content).toEqual({
|
2023-09-14 11:36:15 +00:00
|
|
|
"body": "blinks __quickly__",
|
|
|
|
"msgtype": "m.emote",
|
|
|
|
"format": "org.matrix.custom.html",
|
|
|
|
"formatted_body": "blinks <strong>quickly</strong>",
|
|
|
|
"m.mentions": {},
|
2020-01-22 11:56:27 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-03-21 19:09:43 +00:00
|
|
|
it("allows emoting with non-text parts", () => {
|
2023-03-09 12:15:48 +00:00
|
|
|
const model = new EditorModel([], createPartCreator());
|
2022-03-21 19:09:43 +00:00
|
|
|
const documentOffset = new DocumentOffset(16, true);
|
|
|
|
model.update("/me ✨sparkles✨", "insertText", documentOffset);
|
|
|
|
expect(model.parts.length).toEqual(4); // Emoji count as non-text
|
|
|
|
|
2024-11-13 14:11:20 +00:00
|
|
|
const content = createMessageContent("@alice:test", model, undefined, undefined);
|
2022-03-21 19:09:43 +00:00
|
|
|
|
|
|
|
expect(content).toEqual({
|
2023-09-14 11:36:15 +00:00
|
|
|
"body": "✨sparkles✨",
|
|
|
|
"msgtype": "m.emote",
|
|
|
|
"m.mentions": {},
|
2022-03-21 19:09:43 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-01-22 11:56:27 +00:00
|
|
|
it("allows sending double-slash escaped slash commands correctly", () => {
|
2023-03-09 12:15:48 +00:00
|
|
|
const model = new EditorModel([], createPartCreator());
|
2021-10-01 13:35:54 +00:00
|
|
|
const documentOffset = new DocumentOffset(32, true);
|
|
|
|
|
|
|
|
model.update("//dev/null is my favourite place", "insertText", documentOffset);
|
2020-01-22 11:56:27 +00:00
|
|
|
|
2024-11-13 14:11:20 +00:00
|
|
|
const content = createMessageContent("@alice:test", model, undefined, undefined);
|
2020-01-22 11:56:27 +00:00
|
|
|
|
|
|
|
expect(content).toEqual({
|
2023-09-14 11:36:15 +00:00
|
|
|
"body": "/dev/null is my favourite place",
|
|
|
|
"msgtype": "m.text",
|
|
|
|
"m.mentions": {},
|
2020-01-22 11:56:27 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2023-03-23 11:47:40 +00:00
|
|
|
describe("attachMentions", () => {
|
|
|
|
const partsCreator = createPartCreator();
|
|
|
|
|
|
|
|
it("no mentions", () => {
|
|
|
|
const model = new EditorModel([], partsCreator);
|
|
|
|
const content: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, undefined);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": {},
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test user mentions", () => {
|
|
|
|
const model = new EditorModel([partsCreator.userPill("Bob", "@bob:test")], partsCreator);
|
|
|
|
const content: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, undefined);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": { user_ids: ["@bob:test"] },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test reply", () => {
|
|
|
|
// Replying to an event adds the sender to the list of mentioned users.
|
|
|
|
const model = new EditorModel([], partsCreator);
|
|
|
|
let replyToEvent = mkEvent({
|
|
|
|
type: "m.room.message",
|
|
|
|
user: "@bob:test",
|
|
|
|
room: "!abc:test",
|
2023-07-11 22:29:54 +00:00
|
|
|
content: { "m.mentions": {} },
|
2023-03-23 11:47:40 +00:00
|
|
|
event: true,
|
|
|
|
});
|
|
|
|
let content: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, replyToEvent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": { user_ids: ["@bob:test"] },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// It also adds any other mentioned users, but removes yourself.
|
|
|
|
replyToEvent = mkEvent({
|
|
|
|
type: "m.room.message",
|
|
|
|
user: "@bob:test",
|
|
|
|
room: "!abc:test",
|
2023-07-11 22:29:54 +00:00
|
|
|
content: { "m.mentions": { user_ids: ["@alice:test", "@charlie:test"] } },
|
2023-03-23 11:47:40 +00:00
|
|
|
event: true,
|
|
|
|
});
|
|
|
|
content = {};
|
|
|
|
attachMentions("@alice:test", content, model, replyToEvent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": { user_ids: ["@bob:test", "@charlie:test"] },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test room mention", () => {
|
|
|
|
const model = new EditorModel([partsCreator.atRoomPill("@room")], partsCreator);
|
|
|
|
const content: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, undefined);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": { room: true },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test reply to room mention", () => {
|
|
|
|
// Replying to a room mention shouldn't automatically be a room mention.
|
|
|
|
const model = new EditorModel([], partsCreator);
|
|
|
|
const replyToEvent = mkEvent({
|
|
|
|
type: "m.room.message",
|
|
|
|
user: "@alice:test",
|
|
|
|
room: "!abc:test",
|
2023-07-11 22:29:54 +00:00
|
|
|
content: { "m.mentions": { room: true } },
|
2023-03-23 11:47:40 +00:00
|
|
|
event: true,
|
|
|
|
});
|
|
|
|
const content: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, replyToEvent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": {},
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test broken mentions", () => {
|
|
|
|
// Replying to a room mention shouldn't automatically be a room mention.
|
|
|
|
const model = new EditorModel([], partsCreator);
|
|
|
|
const replyToEvent = mkEvent({
|
|
|
|
type: "m.room.message",
|
|
|
|
user: "@alice:test",
|
|
|
|
room: "!abc:test",
|
|
|
|
// @ts-ignore - Purposefully testing invalid data.
|
2023-07-11 22:29:54 +00:00
|
|
|
content: { "m.mentions": { user_ids: "@bob:test" } },
|
2023-03-23 11:47:40 +00:00
|
|
|
event: true,
|
|
|
|
});
|
|
|
|
const content: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, replyToEvent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": {},
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("attachMentions with edit", () => {
|
|
|
|
it("no mentions", () => {
|
|
|
|
const model = new EditorModel([], partsCreator);
|
|
|
|
const content: IContent = { "m.new_content": {} };
|
|
|
|
const prevContent: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, undefined, prevContent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": {},
|
|
|
|
"m.new_content": { "m.mentions": {} },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("mentions do not propagate", () => {
|
|
|
|
const model = new EditorModel([], partsCreator);
|
|
|
|
const content: IContent = { "m.new_content": {} };
|
|
|
|
const prevContent: IContent = {
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": { user_ids: ["@bob:test"], room: true },
|
2023-03-23 11:47:40 +00:00
|
|
|
};
|
|
|
|
attachMentions("@alice:test", content, model, undefined, prevContent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": {},
|
|
|
|
"m.new_content": { "m.mentions": {} },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test user mentions", () => {
|
|
|
|
const model = new EditorModel([partsCreator.userPill("Bob", "@bob:test")], partsCreator);
|
|
|
|
const content: IContent = { "m.new_content": {} };
|
|
|
|
const prevContent: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, undefined, prevContent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": { user_ids: ["@bob:test"] },
|
|
|
|
"m.new_content": { "m.mentions": { user_ids: ["@bob:test"] } },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test prev user mentions", () => {
|
|
|
|
const model = new EditorModel([partsCreator.userPill("Bob", "@bob:test")], partsCreator);
|
|
|
|
const content: IContent = { "m.new_content": {} };
|
2023-07-11 22:29:54 +00:00
|
|
|
const prevContent: IContent = { "m.mentions": { user_ids: ["@bob:test"] } };
|
2023-03-23 11:47:40 +00:00
|
|
|
attachMentions("@alice:test", content, model, undefined, prevContent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": {},
|
|
|
|
"m.new_content": { "m.mentions": { user_ids: ["@bob:test"] } },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test room mention", () => {
|
|
|
|
const model = new EditorModel([partsCreator.atRoomPill("@room")], partsCreator);
|
|
|
|
const content: IContent = { "m.new_content": {} };
|
|
|
|
const prevContent: IContent = {};
|
|
|
|
attachMentions("@alice:test", content, model, undefined, prevContent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": { room: true },
|
|
|
|
"m.new_content": { "m.mentions": { room: true } },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test prev room mention", () => {
|
|
|
|
const model = new EditorModel([partsCreator.atRoomPill("@room")], partsCreator);
|
|
|
|
const content: IContent = { "m.new_content": {} };
|
2023-07-11 22:29:54 +00:00
|
|
|
const prevContent: IContent = { "m.mentions": { room: true } };
|
2023-03-23 11:47:40 +00:00
|
|
|
attachMentions("@alice:test", content, model, undefined, prevContent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": {},
|
|
|
|
"m.new_content": { "m.mentions": { room: true } },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("test broken mentions", () => {
|
|
|
|
// Replying to a room mention shouldn't automatically be a room mention.
|
|
|
|
const model = new EditorModel([], partsCreator);
|
|
|
|
const content: IContent = { "m.new_content": {} };
|
|
|
|
// @ts-ignore - Purposefully testing invalid data.
|
2023-07-11 22:29:54 +00:00
|
|
|
const prevContent: IContent = { "m.mentions": { user_ids: "@bob:test" } };
|
2023-03-23 11:47:40 +00:00
|
|
|
attachMentions("@alice:test", content, model, undefined, prevContent);
|
|
|
|
expect(content).toEqual({
|
2023-07-11 22:29:54 +00:00
|
|
|
"m.mentions": {},
|
|
|
|
"m.new_content": { "m.mentions": {} },
|
2023-03-23 11:47:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-10-06 23:10:40 +00:00
|
|
|
describe("functions correctly mounted", () => {
|
2022-03-18 10:07:33 +00:00
|
|
|
const mockClient = createTestClient();
|
2022-12-12 11:24:14 +00:00
|
|
|
jest.spyOn(MatrixClientPeg, "get").mockReturnValue(mockClient);
|
|
|
|
const mockRoom = mkStubRoom("myfakeroom", "myfakeroom", mockClient) as any;
|
2020-10-06 23:10:40 +00:00
|
|
|
const mockEvent = mkEvent({
|
|
|
|
type: "m.room.message",
|
2022-12-12 11:24:14 +00:00
|
|
|
room: "myfakeroom",
|
|
|
|
user: "myfakeuser",
|
|
|
|
content: { msgtype: "m.text", body: "Replying to this" },
|
2020-10-06 23:10:40 +00:00
|
|
|
event: true,
|
|
|
|
});
|
2022-12-12 11:24:14 +00:00
|
|
|
mockRoom.findEventById = jest.fn((eventId) => {
|
2020-10-06 23:10:40 +00:00
|
|
|
return eventId === mockEvent.getId() ? mockEvent : null;
|
|
|
|
});
|
|
|
|
|
|
|
|
const spyDispatcher = jest.spyOn(defaultDispatcher, "dispatch");
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
localStorage.clear();
|
|
|
|
spyDispatcher.mockReset();
|
|
|
|
});
|
|
|
|
|
2022-03-18 10:07:33 +00:00
|
|
|
const defaultProps = {
|
|
|
|
room: mockRoom,
|
|
|
|
toggleStickerPickerOpen: jest.fn(),
|
|
|
|
};
|
2022-11-18 09:22:43 +00:00
|
|
|
const getRawComponent = (props = {}, roomContext = defaultRoomContext, client = mockClient) => (
|
|
|
|
<MatrixClientContext.Provider value={client}>
|
|
|
|
<RoomContext.Provider value={roomContext}>
|
|
|
|
<SendMessageComposer {...defaultProps} {...props} />
|
|
|
|
</RoomContext.Provider>
|
|
|
|
</MatrixClientContext.Provider>
|
|
|
|
);
|
2022-03-18 10:07:33 +00:00
|
|
|
const getComponent = (props = {}, roomContext = defaultRoomContext, client = mockClient) => {
|
2022-11-18 09:22:43 +00:00
|
|
|
return render(getRawComponent(props, roomContext, client));
|
2022-03-18 10:07:33 +00:00
|
|
|
};
|
|
|
|
|
2022-03-21 19:09:43 +00:00
|
|
|
it("renders text and placeholder correctly", () => {
|
2022-11-18 09:22:43 +00:00
|
|
|
const { container } = getComponent({ placeholder: "placeholder string" });
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
expect(container.querySelectorAll('[aria-label="placeholder string"]')).toHaveLength(1);
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
addTextToComposer(container, "Test Text");
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
expect(container.textContent).toBe("Test Text");
|
2020-10-06 23:10:40 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("correctly persists state to and from localStorage", () => {
|
2022-11-18 09:22:43 +00:00
|
|
|
const props = { replyToEvent: mockEvent };
|
|
|
|
const { container, unmount, rerender } = getComponent(props);
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
addTextToComposer(container, "Test Text");
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
const key = "mx_cider_state_myfakeroom";
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
expect(container.textContent).toBe("Test Text");
|
2020-10-06 23:10:40 +00:00
|
|
|
expect(localStorage.getItem(key)).toBeNull();
|
|
|
|
|
|
|
|
// ensure the right state was persisted to localStorage
|
2022-11-18 09:22:43 +00:00
|
|
|
unmount();
|
2023-02-15 13:36:22 +00:00
|
|
|
expect(JSON.parse(localStorage.getItem(key)!)).toStrictEqual({
|
2022-12-12 11:24:14 +00:00
|
|
|
parts: [{ type: "plain", text: "Test Text" }],
|
2020-10-06 23:10:40 +00:00
|
|
|
replyEventId: mockEvent.getId(),
|
|
|
|
});
|
|
|
|
|
|
|
|
// ensure the correct model is re-loaded
|
2022-11-18 09:22:43 +00:00
|
|
|
rerender(getRawComponent(props));
|
|
|
|
expect(container.textContent).toBe("Test Text");
|
2020-10-06 23:10:40 +00:00
|
|
|
expect(spyDispatcher).toHaveBeenCalledWith({
|
|
|
|
action: "reply_to_event",
|
|
|
|
event: mockEvent,
|
2021-10-19 15:05:34 +00:00
|
|
|
context: TimelineRenderingType.Room,
|
2020-10-06 23:10:40 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// now try with localStorage wiped out
|
2022-11-18 09:22:43 +00:00
|
|
|
unmount();
|
2020-10-06 23:10:40 +00:00
|
|
|
localStorage.removeItem(key);
|
2022-11-18 09:22:43 +00:00
|
|
|
rerender(getRawComponent(props));
|
|
|
|
expect(container.textContent).toBe("");
|
2020-10-06 23:10:40 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("persists state correctly without replyToEvent onbeforeunload", () => {
|
2022-11-18 09:22:43 +00:00
|
|
|
const { container } = getComponent();
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
addTextToComposer(container, "Hello World");
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
const key = "mx_cider_state_myfakeroom";
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
expect(container.textContent).toBe("Hello World");
|
2020-10-06 23:10:40 +00:00
|
|
|
expect(localStorage.getItem(key)).toBeNull();
|
|
|
|
|
|
|
|
// ensure the right state was persisted to localStorage
|
2022-12-12 11:24:14 +00:00
|
|
|
window.dispatchEvent(new Event("beforeunload"));
|
2023-02-15 13:36:22 +00:00
|
|
|
expect(JSON.parse(localStorage.getItem(key)!)).toStrictEqual({
|
2022-12-12 11:24:14 +00:00
|
|
|
parts: [{ type: "plain", text: "Hello World" }],
|
2020-10-06 23:10:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("persists to session history upon sending", async () => {
|
2022-03-23 10:27:28 +00:00
|
|
|
mockPlatformPeg({ overrideBrowserShortcuts: jest.fn().mockReturnValue(false) });
|
2021-10-01 13:35:54 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
const { container } = getComponent({ replyToEvent: mockEvent });
|
2020-10-06 23:10:40 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
addTextToComposer(container, "This is a message");
|
2023-02-15 13:36:22 +00:00
|
|
|
fireEvent.keyDown(container.querySelector(".mx_SendMessageComposer")!, { key: "Enter" });
|
2022-11-18 09:22:43 +00:00
|
|
|
|
|
|
|
await waitFor(() => {
|
|
|
|
expect(spyDispatcher).toHaveBeenCalledWith({
|
|
|
|
action: "reply_to_event",
|
|
|
|
event: null,
|
|
|
|
context: TimelineRenderingType.Room,
|
|
|
|
});
|
2020-10-06 23:10:40 +00:00
|
|
|
});
|
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
expect(container.textContent).toBe("");
|
2023-02-15 13:36:22 +00:00
|
|
|
const str = sessionStorage.getItem(`mx_cider_history_${mockRoom.roomId}[0]`)!;
|
2020-10-06 23:10:40 +00:00
|
|
|
expect(JSON.parse(str)).toStrictEqual({
|
2022-12-12 11:24:14 +00:00
|
|
|
parts: [{ type: "plain", text: "This is a message" }],
|
2020-10-06 23:10:40 +00:00
|
|
|
replyEventId: mockEvent.getId(),
|
|
|
|
});
|
|
|
|
});
|
2021-10-01 13:35:54 +00:00
|
|
|
|
2022-07-13 05:56:36 +00:00
|
|
|
it("correctly sends a message", () => {
|
2022-12-12 11:24:14 +00:00
|
|
|
mocked(doMaybeLocalRoomAction).mockImplementation(
|
2023-02-24 15:28:40 +00:00
|
|
|
<T,>(roomId: string, fn: (actualRoomId: string) => Promise<T>, _client?: MatrixClient) => {
|
2022-12-12 11:24:14 +00:00
|
|
|
return fn(roomId);
|
|
|
|
},
|
|
|
|
);
|
2022-07-13 05:56:36 +00:00
|
|
|
|
|
|
|
mockPlatformPeg({ overrideBrowserShortcuts: jest.fn().mockReturnValue(false) });
|
2022-11-18 09:22:43 +00:00
|
|
|
const { container } = getComponent();
|
2022-07-13 05:56:36 +00:00
|
|
|
|
2022-11-18 09:22:43 +00:00
|
|
|
addTextToComposer(container, "test message");
|
2023-02-15 13:36:22 +00:00
|
|
|
fireEvent.keyDown(container.querySelector(".mx_SendMessageComposer")!, { key: "Enter" });
|
2022-07-13 05:56:36 +00:00
|
|
|
|
2022-12-12 11:24:14 +00:00
|
|
|
expect(mockClient.sendMessage).toHaveBeenCalledWith("myfakeroom", null, {
|
2023-09-14 11:36:15 +00:00
|
|
|
"body": "test message",
|
|
|
|
"msgtype": MsgType.Text,
|
|
|
|
"m.mentions": {},
|
2022-12-12 11:24:14 +00:00
|
|
|
});
|
2022-07-13 05:56:36 +00:00
|
|
|
});
|
2023-02-20 14:46:07 +00:00
|
|
|
|
2024-11-13 14:11:20 +00:00
|
|
|
it("correctly sends a reply using a slash command", async () => {
|
|
|
|
stubClient();
|
|
|
|
mocked(doMaybeLocalRoomAction).mockImplementation(
|
|
|
|
<T,>(roomId: string, fn: (actualRoomId: string) => Promise<T>, _client?: MatrixClient) => {
|
|
|
|
return fn(roomId);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
const replyToEvent = mkEvent({
|
|
|
|
type: "m.room.message",
|
|
|
|
user: "@bob:test",
|
|
|
|
room: "!abc:test",
|
|
|
|
content: { "m.mentions": {} },
|
|
|
|
event: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
mockPlatformPeg({ overrideBrowserShortcuts: jest.fn().mockReturnValue(false) });
|
|
|
|
const { container } = getComponent({ replyToEvent });
|
|
|
|
|
|
|
|
addTextToComposer(container, "/tableflip");
|
|
|
|
fireEvent.keyDown(container.querySelector(".mx_SendMessageComposer")!, { key: "Enter" });
|
|
|
|
|
|
|
|
await waitFor(() =>
|
|
|
|
expect(mockClient.sendMessage).toHaveBeenCalledWith("myfakeroom", null, {
|
|
|
|
"body": "(╯°□°)╯︵ ┻━┻",
|
|
|
|
"msgtype": MsgType.Text,
|
|
|
|
"m.mentions": {
|
|
|
|
user_ids: ["@bob:test"],
|
|
|
|
},
|
|
|
|
"m.relates_to": {
|
|
|
|
"m.in_reply_to": {
|
|
|
|
event_id: replyToEvent.getId(),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-02-20 14:46:07 +00:00
|
|
|
it("shows chat effects on message sending", () => {
|
|
|
|
mocked(doMaybeLocalRoomAction).mockImplementation(
|
2023-02-24 15:28:40 +00:00
|
|
|
<T,>(roomId: string, fn: (actualRoomId: string) => Promise<T>, _client?: MatrixClient) => {
|
2023-02-20 14:46:07 +00:00
|
|
|
return fn(roomId);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
mockPlatformPeg({ overrideBrowserShortcuts: jest.fn().mockReturnValue(false) });
|
|
|
|
const { container } = getComponent();
|
|
|
|
|
|
|
|
addTextToComposer(container, "🎉");
|
|
|
|
fireEvent.keyDown(container.querySelector(".mx_SendMessageComposer")!, { key: "Enter" });
|
|
|
|
|
|
|
|
expect(mockClient.sendMessage).toHaveBeenCalledWith("myfakeroom", null, {
|
2023-09-14 11:36:15 +00:00
|
|
|
"body": "test message",
|
|
|
|
"msgtype": MsgType.Text,
|
|
|
|
"m.mentions": {},
|
2023-02-20 14:46:07 +00:00
|
|
|
});
|
|
|
|
|
2023-08-03 12:56:30 +00:00
|
|
|
expect(defaultDispatcher.dispatch).toHaveBeenCalledWith({ action: `effects.confetti` });
|
2023-02-20 14:46:07 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("not to send chat effects on message sending for threads", () => {
|
|
|
|
mocked(doMaybeLocalRoomAction).mockImplementation(
|
2023-02-24 15:28:40 +00:00
|
|
|
<T,>(roomId: string, fn: (actualRoomId: string) => Promise<T>, _client?: MatrixClient) => {
|
2023-02-20 14:46:07 +00:00
|
|
|
return fn(roomId);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
mockPlatformPeg({ overrideBrowserShortcuts: jest.fn().mockReturnValue(false) });
|
|
|
|
const { container } = getComponent({
|
|
|
|
relation: {
|
|
|
|
rel_type: "m.thread",
|
|
|
|
event_id: "$yolo",
|
|
|
|
is_falling_back: true,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
addTextToComposer(container, "🎉");
|
|
|
|
fireEvent.keyDown(container.querySelector(".mx_SendMessageComposer")!, { key: "Enter" });
|
|
|
|
|
|
|
|
expect(mockClient.sendMessage).toHaveBeenCalledWith("myfakeroom", null, {
|
2023-09-14 11:36:15 +00:00
|
|
|
"body": "test message",
|
|
|
|
"msgtype": MsgType.Text,
|
|
|
|
"m.mentions": {},
|
2023-02-20 14:46:07 +00:00
|
|
|
});
|
|
|
|
|
2023-08-03 12:56:30 +00:00
|
|
|
expect(defaultDispatcher.dispatch).not.toHaveBeenCalledWith({ action: `effects.confetti` });
|
2023-02-20 14:46:07 +00:00
|
|
|
});
|
2020-10-06 23:10:40 +00:00
|
|
|
});
|
2020-11-17 22:36:58 +00:00
|
|
|
|
|
|
|
describe("isQuickReaction", () => {
|
|
|
|
it("correctly detects quick reaction", () => {
|
2023-03-09 12:15:48 +00:00
|
|
|
const model = new EditorModel([], createPartCreator());
|
2021-10-01 13:35:54 +00:00
|
|
|
model.update("+😊", "insertText", new DocumentOffset(3, true));
|
2020-11-17 22:36:58 +00:00
|
|
|
|
|
|
|
const isReaction = isQuickReaction(model);
|
|
|
|
|
|
|
|
expect(isReaction).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("correctly detects quick reaction with space", () => {
|
2023-03-09 12:15:48 +00:00
|
|
|
const model = new EditorModel([], createPartCreator());
|
2021-10-01 13:35:54 +00:00
|
|
|
model.update("+ 😊", "insertText", new DocumentOffset(4, true));
|
2020-11-17 22:36:58 +00:00
|
|
|
|
|
|
|
const isReaction = isQuickReaction(model);
|
|
|
|
|
|
|
|
expect(isReaction).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("correctly rejects quick reaction with extra text", () => {
|
2023-03-09 12:15:48 +00:00
|
|
|
const model = new EditorModel([], createPartCreator());
|
|
|
|
const model2 = new EditorModel([], createPartCreator());
|
|
|
|
const model3 = new EditorModel([], createPartCreator());
|
|
|
|
const model4 = new EditorModel([], createPartCreator());
|
2022-01-10 12:57:20 +00:00
|
|
|
model.update("+😊hello", "insertText", new DocumentOffset(8, true));
|
|
|
|
model2.update(" +😊", "insertText", new DocumentOffset(4, true));
|
|
|
|
model3.update("+ 😊😊", "insertText", new DocumentOffset(6, true));
|
|
|
|
model4.update("+smiley", "insertText", new DocumentOffset(7, true));
|
2020-11-17 22:36:58 +00:00
|
|
|
|
|
|
|
expect(isQuickReaction(model)).toBeFalsy();
|
|
|
|
expect(isQuickReaction(model2)).toBeFalsy();
|
|
|
|
expect(isQuickReaction(model3)).toBeFalsy();
|
|
|
|
expect(isQuickReaction(model4)).toBeFalsy();
|
|
|
|
});
|
|
|
|
});
|
2023-05-09 10:56:12 +00:00
|
|
|
|
|
|
|
it("should call prepareToEncrypt when the user is typing", async () => {
|
|
|
|
const cli = stubClient();
|
|
|
|
cli.isRoomEncrypted = jest.fn().mockReturnValue(true);
|
|
|
|
const room = mkStubRoom("!roomId:server", "Room", cli);
|
|
|
|
|
2024-10-15 07:50:38 +00:00
|
|
|
expect(cli.getCrypto()!.prepareToEncrypt).not.toHaveBeenCalled();
|
2023-05-09 10:56:12 +00:00
|
|
|
|
|
|
|
const { container } = render(
|
|
|
|
<MatrixClientContext.Provider value={cli}>
|
|
|
|
<SendMessageComposer room={room} toggleStickerPickerOpen={jest.fn()} />
|
|
|
|
</MatrixClientContext.Provider>,
|
|
|
|
);
|
|
|
|
|
|
|
|
const composer = container.querySelector<HTMLDivElement>(".mx_BasicMessageComposer_input")!;
|
|
|
|
|
|
|
|
// Does not trigger on keydown as that'll cause false negatives for global shortcuts
|
|
|
|
await userEvent.type(composer, "[ControlLeft>][KeyK][/ControlLeft]");
|
2024-10-15 07:50:38 +00:00
|
|
|
expect(cli.getCrypto()!.prepareToEncrypt).not.toHaveBeenCalled();
|
2023-05-09 10:56:12 +00:00
|
|
|
|
|
|
|
await userEvent.type(composer, "Hello");
|
2024-10-15 07:50:38 +00:00
|
|
|
expect(cli.getCrypto()!.prepareToEncrypt).toHaveBeenCalled();
|
2023-05-09 10:56:12 +00:00
|
|
|
});
|
2020-01-22 11:56:27 +00:00
|
|
|
});
|