* Switch to importing TypedEventEmitter from main js-sdk export * Switch to importing js-sdk/matrix in cypress * Remove duplicated imports * Fix lint:js-fix to run prettier last otherwise ESLint --fix may cause a mess * Add lint rule * Switch to importing models/Room from main js-sdk export * Fix cypress-axe import * Fix more imports * Fix cypress-axe import * Fix duplicated lint rule
189 lines
6.5 KiB
TypeScript
189 lines
6.5 KiB
TypeScript
/*
|
|
Copyright 2017 - 2019, 2021 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 React, { ReactElement, useRef, useState } from "react";
|
|
import classNames from "classnames";
|
|
import { Room, RoomMember } from "matrix-js-sdk/src/matrix";
|
|
|
|
import { MatrixClientPeg } from "../../../MatrixClientPeg";
|
|
import MatrixClientContext from "../../../contexts/MatrixClientContext";
|
|
import Tooltip, { Alignment } from "../elements/Tooltip";
|
|
import { usePermalink } from "../../../hooks/usePermalink";
|
|
import RoomAvatar from "../avatars/RoomAvatar";
|
|
import MemberAvatar from "../avatars/MemberAvatar";
|
|
import { _t } from "../../../languageHandler";
|
|
import { Icon as LinkIcon } from "../../../../res/img/element-icons/room/composer/link.svg";
|
|
import { Icon as UserIcon } from "../../../../res/img/compound/user.svg";
|
|
|
|
export enum PillType {
|
|
UserMention = "TYPE_USER_MENTION",
|
|
RoomMention = "TYPE_ROOM_MENTION",
|
|
AtRoomMention = "TYPE_AT_ROOM_MENTION", // '@room' mention
|
|
EventInSameRoom = "TYPE_EVENT_IN_SAME_ROOM",
|
|
EventInOtherRoom = "TYPE_EVENT_IN_OTHER_ROOM",
|
|
}
|
|
|
|
export const pillRoomNotifPos = (text: string | null): number => {
|
|
return text?.indexOf("@room") ?? -1;
|
|
};
|
|
|
|
export const pillRoomNotifLen = (): number => {
|
|
return "@room".length;
|
|
};
|
|
|
|
const linkIcon = <LinkIcon className="mx_Pill_LinkIcon mx_BaseAvatar mx_BaseAvatar_image" />;
|
|
|
|
const PillRoomAvatar: React.FC<{
|
|
shouldShowPillAvatar: boolean;
|
|
room: Room | null;
|
|
}> = ({ shouldShowPillAvatar, room }) => {
|
|
if (!shouldShowPillAvatar) {
|
|
return null;
|
|
}
|
|
|
|
if (room) {
|
|
return <RoomAvatar room={room} width={16} height={16} aria-hidden="true" />;
|
|
}
|
|
return linkIcon;
|
|
};
|
|
|
|
const PillMemberAvatar: React.FC<{
|
|
shouldShowPillAvatar: boolean;
|
|
member: RoomMember | null;
|
|
}> = ({ shouldShowPillAvatar, member }) => {
|
|
if (!shouldShowPillAvatar) {
|
|
return null;
|
|
}
|
|
|
|
if (member) {
|
|
return <MemberAvatar member={member} width={16} height={16} aria-hidden="true" hideTitle />;
|
|
}
|
|
return <UserIcon className="mx_Pill_UserIcon mx_BaseAvatar mx_BaseAvatar_image" />;
|
|
};
|
|
|
|
export interface PillProps {
|
|
// The Type of this Pill. If url is given, this is auto-detected.
|
|
type?: PillType;
|
|
// The URL to pillify (no validation is done)
|
|
url?: string;
|
|
/** Whether the pill is in a message. It will act as a link then. */
|
|
inMessage?: boolean;
|
|
// The room in which this pill is being rendered
|
|
room?: Room;
|
|
// Whether to include an avatar in the pill
|
|
shouldShowPillAvatar?: boolean;
|
|
}
|
|
|
|
export const Pill: React.FC<PillProps> = ({ type: propType, url, inMessage, room, shouldShowPillAvatar = true }) => {
|
|
const tooltipId = useRef(`mx_Pill_${Math.random()}`).current;
|
|
const [hover, setHover] = useState(false);
|
|
const { event, member, onClick, resourceId, targetRoom, text, type } = usePermalink({
|
|
room,
|
|
type: propType,
|
|
url,
|
|
});
|
|
|
|
if (!type || !text) {
|
|
return null;
|
|
}
|
|
|
|
const classes = classNames("mx_Pill", {
|
|
mx_AtRoomPill: type === PillType.AtRoomMention,
|
|
mx_RoomPill: type === PillType.RoomMention,
|
|
mx_SpacePill: type === "space",
|
|
mx_UserPill: type === PillType.UserMention,
|
|
mx_UserPill_me: resourceId === MatrixClientPeg.safeGet().getUserId(),
|
|
mx_EventPill: type === PillType.EventInOtherRoom || type === PillType.EventInSameRoom,
|
|
});
|
|
|
|
const onMouseOver = (): void => {
|
|
setHover(true);
|
|
};
|
|
|
|
const onMouseLeave = (): void => {
|
|
setHover(false);
|
|
};
|
|
|
|
const tip = hover && resourceId ? <Tooltip id={tooltipId} label={resourceId} alignment={Alignment.Right} /> : null;
|
|
let avatar: ReactElement | null = null;
|
|
let pillText: string | null = text;
|
|
|
|
switch (type) {
|
|
case PillType.EventInOtherRoom:
|
|
{
|
|
avatar = <PillRoomAvatar shouldShowPillAvatar={shouldShowPillAvatar} room={targetRoom} />;
|
|
pillText = _t("Message in %(room)s", {
|
|
room: text,
|
|
});
|
|
}
|
|
break;
|
|
case PillType.EventInSameRoom:
|
|
{
|
|
if (event) {
|
|
avatar = <PillMemberAvatar shouldShowPillAvatar={shouldShowPillAvatar} member={member} />;
|
|
pillText = _t("Message from %(user)s", {
|
|
user: text,
|
|
});
|
|
} else {
|
|
avatar = linkIcon;
|
|
pillText = _t("Message");
|
|
}
|
|
}
|
|
break;
|
|
case PillType.AtRoomMention:
|
|
case PillType.RoomMention:
|
|
case "space":
|
|
avatar = <PillRoomAvatar shouldShowPillAvatar={shouldShowPillAvatar} room={targetRoom} />;
|
|
break;
|
|
case PillType.UserMention:
|
|
avatar = <PillMemberAvatar shouldShowPillAvatar={shouldShowPillAvatar} member={member} />;
|
|
break;
|
|
default:
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<bdi>
|
|
<MatrixClientContext.Provider value={MatrixClientPeg.safeGet()}>
|
|
{inMessage && url ? (
|
|
<a
|
|
className={classes}
|
|
href={url}
|
|
onClick={onClick}
|
|
onMouseOver={onMouseOver}
|
|
onMouseLeave={onMouseLeave}
|
|
aria-describedby={tooltipId}
|
|
>
|
|
{avatar}
|
|
<span className="mx_Pill_text">{pillText}</span>
|
|
{tip}
|
|
</a>
|
|
) : (
|
|
<span
|
|
className={classes}
|
|
onMouseOver={onMouseOver}
|
|
onMouseLeave={onMouseLeave}
|
|
aria-describedby={tooltipId}
|
|
>
|
|
{avatar}
|
|
<span className="mx_Pill_text">{pillText}</span>
|
|
{tip}
|
|
</span>
|
|
)}
|
|
</MatrixClientContext.Provider>
|
|
</bdi>
|
|
);
|
|
};
|