* Very first pass at shared component views Turn the trivial TextualEvent into a shared component with a separate view model for element web. Args to view model will probably change to be more specific and VM typer needs abstracting out into an interface, but should give the general idea. * Remove old TextualEvent * Pass showHiddenEvents Because we used it anyway, we just cheated by getting it from the context * Factor out common view model stuff * Move ViewModel interface into the shared components * Add tiny wrapper hook * Move showHiddenEvents into props fully * Fill in stories / test * chore: setup storybook cherry pick edc5e8705674b8708d986910b02b5d2545777fb3 from florianduros/storybook * Add TextualEvent component to storybook * Add mock view model & snapshot * Remove old style stories entry * Change import * Change import * Prettier * Add paxckage patch to @types/mdx for React 19 compat * Pass getSnapshot as getServerSnapshot too * Maybe make sonar regognise tests as tests * Typo * Use storybook reacvt-vite There's no reason to use the react-webpack plugin just because our app is stuck on webpack, it just means we have vite as a dependency too. * Change here too * Workaround for incomatible types in rollup https://github.com/rollup/rollup/issues/5199 * Remove webpack styling addon Not necessary now we're using vite * Hopefully do screenshot testing... * need newer node * quote issues * Make it an npm script * colons * use right port * Install playwright browsers * Try without the if * Oh right, we need the headless shell * Pass flag to store received screenshots and upload diffs too * Update snapshot from received * Include platform in snapshot / received dir because font rendering differs between platforms * Suffix snapshots with platform instead like we do for playwright * Remove unnecessary env vars and better name * Add some comments * Prettier * Fix yarn.lock * Memoise vm creation Co-authored-by: Florian Duros <florianduros@element.io> * Add implements Co-authored-by: Florian Duros <florianduros@element.io> * Fix listener interface * Add implements Co-authored-by: Florian Duros <florianduros@element.io> * Fix types * Fix more types * Revert useMemo as this isn't a hook * Unused import * Add missing playwright step * Add return type annotation * Change to add / remove subscription callback * Change to 'add' rather than 'subs.subscribe' * Add cache specifier for only shell playwright browsers * Add copyright headers --------- Co-authored-by: Florian Duros <florian.duros@ormaz.fr> Co-authored-by: Florian Duros <florianduros@element.io>
460 lines
18 KiB
TypeScript
460 lines
18 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only OR LicenseRef-Element-Commercial
|
|
Please see LICENSE files in the repository root for full details.
|
|
*/
|
|
|
|
import React, { type JSX } from "react";
|
|
import {
|
|
type MatrixEvent,
|
|
EventType,
|
|
MsgType,
|
|
RelationType,
|
|
type MatrixClient,
|
|
GroupCallIntent,
|
|
M_POLL_END,
|
|
M_POLL_START,
|
|
} from "matrix-js-sdk/src/matrix";
|
|
import { type Optional } from "matrix-events-sdk";
|
|
|
|
import SettingsStore from "../settings/SettingsStore";
|
|
import type LegacyCallEventGrouper from "../components/structures/LegacyCallEventGrouper";
|
|
import { type EventTileProps } from "../components/views/rooms/EventTile";
|
|
import { TimelineRenderingType } from "../contexts/RoomContext";
|
|
import MessageEvent from "../components/views/messages/MessageEvent";
|
|
import LegacyCallEvent from "../components/views/messages/LegacyCallEvent";
|
|
import { CallEvent } from "../components/views/messages/CallEvent";
|
|
import EncryptionEvent from "../components/views/messages/EncryptionEvent";
|
|
import { RoomPredecessorTile } from "../components/views/messages/RoomPredecessorTile";
|
|
import RoomAvatarEvent from "../components/views/messages/RoomAvatarEvent";
|
|
import { WIDGET_LAYOUT_EVENT_TYPE } from "../stores/widgets/WidgetLayoutStore";
|
|
import { ALL_RULE_TYPES } from "../mjolnir/BanList";
|
|
import { MatrixClientPeg } from "../MatrixClientPeg";
|
|
import MKeyVerificationRequest from "../components/views/messages/MKeyVerificationRequest";
|
|
import { WidgetType } from "../widgets/WidgetType";
|
|
import MJitsiWidgetEvent from "../components/views/messages/MJitsiWidgetEvent";
|
|
import { hasText } from "../TextForEvent";
|
|
import { getMessageModerationState, MessageModerationState } from "../utils/EventUtils";
|
|
import HiddenBody from "../components/views/messages/HiddenBody";
|
|
import ViewSourceEvent from "../components/views/messages/ViewSourceEvent";
|
|
import { shouldDisplayAsBeaconTile } from "../utils/beacon/timeline";
|
|
import { ElementCall } from "../models/Call";
|
|
import { type IBodyProps } from "../components/views/messages/IBodyProps";
|
|
import ModuleApi from "../modules/Api";
|
|
import { TextualEventViewModel } from "../viewmodels/event-tiles/TextualEventViewModel";
|
|
import { TextualEvent } from "../shared-components/event-tiles/TextualEvent";
|
|
|
|
// Subset of EventTile's IProps plus some mixins
|
|
export interface EventTileTypeProps
|
|
extends Pick<
|
|
EventTileProps,
|
|
| "mxEvent"
|
|
| "highlights"
|
|
| "highlightLink"
|
|
| "showUrlPreview"
|
|
| "forExport"
|
|
| "getRelationsForEvent"
|
|
| "editState"
|
|
| "replacingEventId"
|
|
| "permalinkCreator"
|
|
| "callEventGrouper"
|
|
| "isSeeingThroughMessageHiddenForModeration"
|
|
| "inhibitInteraction"
|
|
> {
|
|
ref?: React.RefObject<any>; // `any` because it's effectively impossible to convince TS of a reasonable type
|
|
timestamp?: JSX.Element;
|
|
maxImageHeight?: number; // pixels
|
|
overrideBodyTypes?: Record<string, React.ComponentType<IBodyProps>>;
|
|
overrideEventTypes?: Record<string, React.ComponentType<IBodyProps>>;
|
|
showHiddenEvents: boolean;
|
|
}
|
|
|
|
type FactoryProps = Omit<EventTileTypeProps, "ref">;
|
|
type Factory<X = FactoryProps> = (ref: React.RefObject<any> | undefined, props: X) => JSX.Element;
|
|
|
|
export const MessageEventFactory: Factory = (ref, props) => <MessageEvent ref={ref} {...props} />;
|
|
const LegacyCallEventFactory: Factory<FactoryProps & { callEventGrouper: LegacyCallEventGrouper }> = (ref, props) => (
|
|
<LegacyCallEvent ref={ref} {...props} />
|
|
);
|
|
const CallEventFactory: Factory = (ref, props) => <CallEvent ref={ref} {...props} />;
|
|
export const TextualEventFactory: Factory = (ref, props) => {
|
|
const vm = new TextualEventViewModel(props);
|
|
return <TextualEvent vm={vm} />;
|
|
};
|
|
const VerificationReqFactory: Factory = (_ref, props) => <MKeyVerificationRequest {...props} />;
|
|
const HiddenEventFactory: Factory = (ref, props) => <HiddenBody ref={ref} {...props} />;
|
|
|
|
// These factories are exported for reference comparison against pickFactory()
|
|
export const JitsiEventFactory: Factory = (ref, props) => <MJitsiWidgetEvent ref={ref} {...props} />;
|
|
export const JSONEventFactory: Factory = (ref, props) => <ViewSourceEvent ref={ref} {...props} />;
|
|
export const RoomCreateEventFactory: Factory = (_ref, props) => <RoomPredecessorTile {...props} />;
|
|
|
|
const EVENT_TILE_TYPES = new Map<string, Factory>([
|
|
[EventType.RoomMessage, MessageEventFactory], // note that verification requests are handled in pickFactory()
|
|
[EventType.Sticker, MessageEventFactory],
|
|
[M_POLL_START.name, MessageEventFactory],
|
|
[M_POLL_START.altName, MessageEventFactory],
|
|
[M_POLL_END.name, MessageEventFactory],
|
|
[M_POLL_END.altName, MessageEventFactory],
|
|
[EventType.CallInvite, LegacyCallEventFactory as Factory], // note that this requires a special factory type
|
|
]);
|
|
|
|
const STATE_EVENT_TILE_TYPES = new Map<string, Factory>([
|
|
[EventType.RoomEncryption, (ref, props) => <EncryptionEvent ref={ref} {...props} />],
|
|
[EventType.RoomCanonicalAlias, TextualEventFactory],
|
|
[EventType.RoomCreate, RoomCreateEventFactory],
|
|
[EventType.RoomMember, TextualEventFactory],
|
|
[EventType.RoomName, TextualEventFactory],
|
|
[EventType.RoomAvatar, (ref, props) => <RoomAvatarEvent ref={ref} {...props} />],
|
|
[EventType.RoomThirdPartyInvite, TextualEventFactory],
|
|
[EventType.RoomHistoryVisibility, TextualEventFactory],
|
|
[EventType.RoomTopic, TextualEventFactory],
|
|
[EventType.RoomPowerLevels, TextualEventFactory],
|
|
[EventType.RoomPinnedEvents, TextualEventFactory],
|
|
[EventType.RoomServerAcl, TextualEventFactory],
|
|
// TODO: Enable support for m.widget event type (https://github.com/vector-im/element-web/issues/13111)
|
|
["im.vector.modular.widgets", TextualEventFactory], // note that Jitsi widgets are special in pickFactory()
|
|
[WIDGET_LAYOUT_EVENT_TYPE, TextualEventFactory],
|
|
[EventType.RoomTombstone, TextualEventFactory],
|
|
[EventType.RoomJoinRules, TextualEventFactory],
|
|
[EventType.RoomGuestAccess, TextualEventFactory],
|
|
]);
|
|
|
|
for (const evType of ElementCall.CALL_EVENT_TYPE.names) {
|
|
STATE_EVENT_TILE_TYPES.set(evType, CallEventFactory);
|
|
}
|
|
|
|
// Add all the Mjolnir stuff to the renderer too
|
|
for (const evType of ALL_RULE_TYPES) {
|
|
STATE_EVENT_TILE_TYPES.set(evType, TextualEventFactory);
|
|
}
|
|
|
|
// These events should be recorded in the STATE_EVENT_TILE_TYPES
|
|
const SINGULAR_STATE_EVENTS = new Set([
|
|
EventType.RoomEncryption,
|
|
EventType.RoomCanonicalAlias,
|
|
EventType.RoomCreate,
|
|
EventType.RoomName,
|
|
EventType.RoomAvatar,
|
|
EventType.RoomHistoryVisibility,
|
|
EventType.RoomTopic,
|
|
EventType.RoomPowerLevels,
|
|
EventType.RoomPinnedEvents,
|
|
EventType.RoomServerAcl,
|
|
WIDGET_LAYOUT_EVENT_TYPE,
|
|
EventType.RoomTombstone,
|
|
EventType.RoomJoinRules,
|
|
EventType.RoomGuestAccess,
|
|
]);
|
|
|
|
/**
|
|
* Find an event tile factory for the given conditions.
|
|
* @param mxEvent The event.
|
|
* @param cli The matrix client to reference when needed.
|
|
* @param showHiddenEvents Whether hidden events should be shown.
|
|
* @param asHiddenEv When true, treat the event as always hidden.
|
|
* @returns The factory, or falsy if not possible.
|
|
*/
|
|
export function pickFactory(
|
|
mxEvent: MatrixEvent,
|
|
cli: MatrixClient,
|
|
showHiddenEvents: boolean,
|
|
asHiddenEv?: boolean,
|
|
): Optional<Factory> {
|
|
const evType = mxEvent.getType(); // cache this to reduce call stack execution hits
|
|
|
|
// Note: we avoid calling SettingsStore unless absolutely necessary - this code is on the critical path.
|
|
|
|
if (asHiddenEv && showHiddenEvents) {
|
|
return JSONEventFactory;
|
|
}
|
|
|
|
const noEventFactoryFactory: () => Optional<Factory> = () => (showHiddenEvents ? JSONEventFactory : undefined); // just don't render things that we shouldn't render
|
|
|
|
// We run all the event type checks first as they might override the factory entirely.
|
|
|
|
const moderationState = getMessageModerationState(mxEvent, cli);
|
|
if (moderationState === MessageModerationState.HIDDEN_TO_CURRENT_USER) {
|
|
return HiddenEventFactory;
|
|
}
|
|
|
|
if (evType === EventType.RoomMessage) {
|
|
// don't show verification requests we're not involved in,
|
|
// not even when showing hidden events
|
|
const content = mxEvent.getContent();
|
|
if (content?.msgtype === MsgType.KeyVerificationRequest) {
|
|
const me = cli.getUserId();
|
|
if (mxEvent.getSender() !== me && content["to"] !== me) {
|
|
return noEventFactoryFactory(); // not for/from us
|
|
} else {
|
|
// override the factory
|
|
return VerificationReqFactory;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (evType === EventType.RoomCreate) {
|
|
const dynamicPredecessorsEnabled = SettingsStore.getValue("feature_dynamic_room_predecessors");
|
|
const predecessor = cli.getRoom(mxEvent.getRoomId())?.findPredecessor(dynamicPredecessorsEnabled);
|
|
if (!predecessor) {
|
|
return noEventFactoryFactory();
|
|
}
|
|
}
|
|
|
|
// TODO: Enable support for m.widget event type (https://github.com/vector-im/element-web/issues/13111)
|
|
if (evType === "im.vector.modular.widgets") {
|
|
let type = mxEvent.getContent()["type"];
|
|
if (!type) {
|
|
// deleted/invalid widget - try the past widget type
|
|
type = mxEvent.getPrevContent()["type"];
|
|
}
|
|
|
|
if (WidgetType.JITSI.matches(type)) {
|
|
// override the factory
|
|
return JitsiEventFactory;
|
|
}
|
|
}
|
|
|
|
// Try and pick a state event factory, if we can.
|
|
if (mxEvent.isState()) {
|
|
if (shouldDisplayAsBeaconTile(mxEvent)) {
|
|
return MessageEventFactory;
|
|
}
|
|
|
|
if (SINGULAR_STATE_EVENTS.has(evType) && mxEvent.getStateKey() !== "") {
|
|
return noEventFactoryFactory(); // improper event type to render
|
|
}
|
|
|
|
if (STATE_EVENT_TILE_TYPES.get(evType) === TextualEventFactory && !hasText(mxEvent, cli, showHiddenEvents)) {
|
|
return noEventFactoryFactory();
|
|
}
|
|
|
|
return STATE_EVENT_TILE_TYPES.get(evType) ?? noEventFactoryFactory();
|
|
}
|
|
|
|
// Blanket override for all events. The MessageEvent component handles redacted states for us.
|
|
if (mxEvent.isRedacted()) {
|
|
return MessageEventFactory;
|
|
}
|
|
|
|
if (mxEvent.isRelation(RelationType.Replace)) {
|
|
return noEventFactoryFactory();
|
|
}
|
|
|
|
return EVENT_TILE_TYPES.get(evType) ?? noEventFactoryFactory();
|
|
}
|
|
|
|
/**
|
|
* Render an event as a tile
|
|
* @param renderType The render type. Used to inform properties given to the eventual component.
|
|
* @param props The properties to provide to the eventual component.
|
|
* @param showHiddenEvents Whether hidden events should be shown.
|
|
* @param cli Optional client instance to use, otherwise the default MatrixClientPeg will be used.
|
|
* @returns The tile as JSX, or falsy if unable to render.
|
|
*/
|
|
export function renderTile(
|
|
renderType: TimelineRenderingType,
|
|
props: EventTileTypeProps,
|
|
cli?: MatrixClient,
|
|
): Optional<JSX.Element> {
|
|
cli = cli ?? MatrixClientPeg.safeGet(); // because param defaults don't do the correct thing
|
|
|
|
const factory = pickFactory(props.mxEvent, cli, props.showHiddenEvents);
|
|
if (!factory) {
|
|
// If we don't have a factory for this event, attempt
|
|
// to find a custom component that can render it.
|
|
// Will return null if no custom component can render it.
|
|
return ModuleApi.customComponents.renderMessage({
|
|
mxEvent: props.mxEvent,
|
|
});
|
|
}
|
|
|
|
// Note that we split off the ones we actually care about here just to be sure that we're
|
|
// not going to accidentally send things we shouldn't from lazy callers. Eg: EventTile's
|
|
// lazy calls of `renderTile(..., this.props)` will have a lot more than we actually care
|
|
// about.
|
|
const {
|
|
ref,
|
|
mxEvent,
|
|
forExport,
|
|
replacingEventId,
|
|
editState,
|
|
highlights,
|
|
highlightLink,
|
|
showUrlPreview,
|
|
permalinkCreator,
|
|
callEventGrouper,
|
|
getRelationsForEvent,
|
|
isSeeingThroughMessageHiddenForModeration,
|
|
timestamp,
|
|
inhibitInteraction,
|
|
showHiddenEvents,
|
|
} = props;
|
|
|
|
switch (renderType) {
|
|
case TimelineRenderingType.File:
|
|
case TimelineRenderingType.Notification:
|
|
case TimelineRenderingType.Thread:
|
|
return ModuleApi.customComponents.renderMessage(
|
|
{
|
|
mxEvent: props.mxEvent,
|
|
},
|
|
(origProps) =>
|
|
factory(props.ref, {
|
|
// We only want a subset of props, so we don't end up causing issues for downstream components.
|
|
mxEvent,
|
|
highlights,
|
|
highlightLink,
|
|
showUrlPreview: origProps?.showUrlPreview ?? showUrlPreview,
|
|
editState,
|
|
replacingEventId,
|
|
getRelationsForEvent,
|
|
isSeeingThroughMessageHiddenForModeration,
|
|
permalinkCreator,
|
|
inhibitInteraction,
|
|
showHiddenEvents,
|
|
}),
|
|
);
|
|
default:
|
|
return ModuleApi.customComponents.renderMessage(
|
|
{
|
|
mxEvent: props.mxEvent,
|
|
},
|
|
(origProps) =>
|
|
factory(ref, {
|
|
// NEARLY ALL THE OPTIONS!
|
|
mxEvent,
|
|
forExport,
|
|
replacingEventId,
|
|
editState,
|
|
highlights,
|
|
highlightLink,
|
|
showUrlPreview: origProps?.showUrlPreview ?? showUrlPreview,
|
|
permalinkCreator,
|
|
callEventGrouper,
|
|
getRelationsForEvent,
|
|
isSeeingThroughMessageHiddenForModeration,
|
|
timestamp,
|
|
inhibitInteraction,
|
|
showHiddenEvents,
|
|
}),
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* A version of renderTile() specifically for replies.
|
|
* @param props The properties to specify on the eventual object.
|
|
* @param showHiddenEvents Whether hidden events should be shown.
|
|
* @param cli Optional client instance to use, otherwise the default MatrixClientPeg will be used.
|
|
* @returns The tile as JSX, or falsy if unable to render.
|
|
*/
|
|
export function renderReplyTile(
|
|
props: EventTileTypeProps,
|
|
showHiddenEvents: boolean,
|
|
cli?: MatrixClient,
|
|
): Optional<JSX.Element> {
|
|
cli = cli ?? MatrixClientPeg.safeGet(); // because param defaults don't do the correct thing
|
|
|
|
const factory = pickFactory(props.mxEvent, cli, showHiddenEvents);
|
|
if (!factory) {
|
|
// If we don't have a factory for this event, attempt
|
|
// to find a custom component that can render it.
|
|
// Will return null if no custom component can render it.
|
|
return ModuleApi.customComponents.renderMessage({
|
|
mxEvent: props.mxEvent,
|
|
});
|
|
}
|
|
|
|
// See renderTile() for why we split off so much
|
|
const {
|
|
ref,
|
|
mxEvent,
|
|
highlights,
|
|
highlightLink,
|
|
showUrlPreview,
|
|
overrideBodyTypes,
|
|
overrideEventTypes,
|
|
replacingEventId,
|
|
maxImageHeight,
|
|
getRelationsForEvent,
|
|
isSeeingThroughMessageHiddenForModeration,
|
|
permalinkCreator,
|
|
} = props;
|
|
|
|
return ModuleApi.customComponents.renderMessage(
|
|
{
|
|
mxEvent: props.mxEvent,
|
|
},
|
|
(origProps) =>
|
|
factory(ref, {
|
|
mxEvent,
|
|
highlights,
|
|
highlightLink,
|
|
showUrlPreview: origProps?.showUrlPreview ?? showUrlPreview,
|
|
overrideBodyTypes,
|
|
overrideEventTypes,
|
|
replacingEventId,
|
|
maxImageHeight,
|
|
getRelationsForEvent,
|
|
isSeeingThroughMessageHiddenForModeration,
|
|
permalinkCreator,
|
|
showHiddenEvents,
|
|
}),
|
|
);
|
|
}
|
|
|
|
// XXX: this'll eventually be dynamic based on the fields once we have extensible event types
|
|
const messageTypes = [EventType.RoomMessage, EventType.Sticker];
|
|
export function isMessageEvent(ev: MatrixEvent): boolean {
|
|
return (
|
|
messageTypes.includes(ev.getType() as EventType) ||
|
|
M_POLL_START.matches(ev.getType()) ||
|
|
M_POLL_END.matches(ev.getType())
|
|
);
|
|
}
|
|
|
|
export function haveRendererForEvent(
|
|
mxEvent: MatrixEvent,
|
|
matrixClient: MatrixClient,
|
|
showHiddenEvents: boolean,
|
|
): boolean {
|
|
// Only show "Message deleted" tile for plain message events, encrypted events,
|
|
// and state events as they'll likely still contain enough keys to be relevant.
|
|
if (mxEvent.isRedacted() && !mxEvent.isEncrypted() && !isMessageEvent(mxEvent) && !mxEvent.isState()) {
|
|
return false;
|
|
}
|
|
|
|
// Check to see if we have any hints for this message, which indicates
|
|
// there is a custom renderer for the event.
|
|
if (ModuleApi.customComponents.getHintsForMessage(mxEvent)) {
|
|
return true;
|
|
}
|
|
|
|
// No tile for replacement events since they update the original tile
|
|
if (mxEvent.isRelation(RelationType.Replace)) return false;
|
|
|
|
const handler = pickFactory(mxEvent, matrixClient, showHiddenEvents);
|
|
if (!handler) return false;
|
|
if (handler === TextualEventFactory) {
|
|
return hasText(mxEvent, matrixClient, showHiddenEvents);
|
|
} else if (handler === STATE_EVENT_TILE_TYPES.get(EventType.RoomCreate)) {
|
|
const dynamicPredecessorsEnabled = SettingsStore.getValue("feature_dynamic_room_predecessors");
|
|
const predecessor = matrixClient.getRoom(mxEvent.getRoomId())?.findPredecessor(dynamicPredecessorsEnabled);
|
|
return Boolean(predecessor);
|
|
} else if (
|
|
ElementCall.CALL_EVENT_TYPE.names.some((eventType) => handler === STATE_EVENT_TILE_TYPES.get(eventType))
|
|
) {
|
|
const intent = mxEvent.getContent()["m.intent"];
|
|
const newlyStarted = Object.keys(mxEvent.getPrevContent()).length === 0;
|
|
// Only interested in events that mark the start of a non-room call
|
|
return newlyStarted && typeof intent === "string" && intent !== GroupCallIntent.Room;
|
|
} else if (handler === JSONEventFactory) {
|
|
return false;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|