Skip to content
Snippets Groups Projects
Message.tsx 3.7 KiB
Newer Older
insert's avatar
insert committed
import Embed from "./embed/Embed";
insert's avatar
insert committed
import UserIcon from "../user/UserIcon";
import { Username } from "../user/UserShort";
import Markdown from "../../markdown/Markdown";
import { Children } from "../../../types/Preact";
insert's avatar
insert committed
import Attachment from "./attachments/Attachment";
import { attachContextMenu } from "preact-context-menu";
import { useUser } from "../../../context/revoltjs/hooks";
insert's avatar
insert committed
import { QueuedMessage } from "../../../redux/reducers/queue";
import { MessageObject } from "../../../context/revoltjs/util";
import MessageBase, { MessageContent, MessageDetail, MessageInfo } from "./MessageBase";
insert's avatar
insert committed
import Overline from "../../ui/Overline";
insert's avatar
insert committed
import { useContext } from "preact/hooks";
import { AppContext } from "../../../context/revoltjs/RevoltClient";
import { memo } from "preact/compat";
insert's avatar
insert committed
import { MessageReply } from "./attachments/MessageReply";
import { useIntermediate } from "../../../context/intermediate/Intermediate";

interface Props {
    attachContext?: boolean
insert's avatar
insert committed
    queued?: QueuedMessage
    message: MessageObject
    contrast?: boolean
    content?: Children
    head?: boolean
}

function Message({ attachContext, message, contrast, content: replacement, head: preferHead, queued }: Props) {
    // TODO: Can improve re-renders here by providing a list
    // TODO: of dependencies. We only need to update on u/avatar.
insert's avatar
insert committed
    const user = useUser(message.author);
    const client = useContext(AppContext);
    const { openScreen } = useIntermediate();
insert's avatar
insert committed
    const content = message.content as string;
insert's avatar
insert committed
    const head = preferHead || (message.replies && message.replies.length > 0);
    const userContext = attachContext ? attachContextMenu('Menu', { user: message.author, contextualChannel: message.channel }) : undefined as any; // ! FIXME: tell fatal to make this type generic
    const openProfile = () => openScreen({ id: 'profile', user_id: message.author });
        <div id={message._id}>
insert's avatar
insert committed
            { message.replies?.map((message_id, index) => <MessageReply index={index} id={message_id} channel={message.channel} />) }
insert's avatar
insert committed
                head={head && !(message.replies && message.replies.length > 0)}
insert's avatar
insert committed
                contrast={contrast}
                sending={typeof queued !== 'undefined'}
                mention={message.mentions?.includes(client.user!._id)}
                failed={typeof queued?.error !== 'undefined'}
                onContextMenu={attachContext ? attachContextMenu('Menu', { message, contextualChannel: message.channel, queued }) : undefined}>
                <MessageInfo>
                    { head ?
                        <UserIcon target={user} size={36} onContextMenu={userContext} onClick={openProfile} /> :
insert's avatar
insert committed
                        <MessageDetail message={message} position="left" /> }
                </MessageInfo>
                <MessageContent>
                    { head && <span className="detail">
                        <span className="author">
                            <Username user={user} onContextMenu={userContext} onClick={openProfile} />
                        </span>
insert's avatar
insert committed
                        <MessageDetail message={message} position="top" />
                    </span> }
                    { replacement ?? <Markdown content={content} /> }
                    { queued?.error && <Overline type="error" error={queued.error} /> }
                    { message.attachments?.map((attachment, index) =>
                        <Attachment key={index} attachment={attachment} hasContent={ index > 0 || content.length > 0 } />) }
                    { message.embeds?.map((embed, index) =>
                        <Embed key={index} embed={embed} />) }
                </MessageContent>
            </MessageBase>

export default memo(Message);