import { observer } from "mobx-react-lite"; import { Message } from "revolt.js/dist/maps/Messages"; import styled, { css, keyframes } from "styled-components"; import { decodeTime } from "ulid"; import { Text } from "preact-i18n"; import { useDictionary } from "../../../lib/i18n"; import { dayjs } from "../../../context/Locale"; import Tooltip from "../Tooltip"; export interface BaseMessageProps { head?: boolean; failed?: boolean; mention?: boolean; blocked?: boolean; sending?: boolean; contrast?: boolean; highlight?: boolean; } const highlight = keyframes` 0% { background: var(--mention); } 66% { background: var(--mention); } 100% { background: transparent; } `; export default styled.div<BaseMessageProps>` display: flex; overflow: none; padding: 0.125rem; flex-direction: row; padding-inline-end: 16px; @media (pointer: coarse) { user-select: none; } ${(props) => props.contrast && css` padding: 0.3rem; background: var(--hover); border-radius: var(--border-radius); `} ${(props) => props.head && css` margin-top: 12px; `} ${(props) => props.mention && css` background: var(--mention); `} ${(props) => props.blocked && css` filter: blur(4px); transition: 0.2s ease filter; &:hover { filter: none; } `} ${(props) => props.sending && css` opacity: 0.8; color: var(--tertiary-foreground); `} ${(props) => props.failed && css` color: var(--error); `} ${(props) => props.highlight && css` animation-name: ${highlight}; animation-timing-function: ease; animation-duration: 3s; `} .detail { gap: 8px; display: flex; align-items: center; flex-shrink: 0; } .author { overflow: hidden; cursor: pointer; font-weight: 600 !important; display: -webkit-box; -webkit-line-clamp: 1; -webkit-box-orient: vertical; text-overflow: ellipsis; white-space: normal; &:hover { text-decoration: underline; } } .copy { display: block; overflow: hidden; } &:hover { background: var(--hover); time { opacity: 1; } .system-message-icon { display: none; } } `; export const MessageInfo = styled.div` width: 62px; display: flex; flex-shrink: 0; padding-top: 2px; flex-direction: row; justify-content: center; .copyBracket { opacity: 0; position: absolute; } .copyTime { opacity: 0; position: absolute; } svg { user-select: none; cursor: pointer; &:active { transform: translateY(1px); } } time { opacity: 0; } time, .edited { margin-top: 1px; cursor: default; display: inline; font-size: 10px; color: var(--tertiary-foreground); } time, .edited > div { &::selection { background-color: transparent; color: var(--tertiary-foreground); } } .header { cursor: pointer; } .systemIcon { height: 1.33em; width: 1.33em; margin-right: 0.5em; color: var(--tertiary-foreground); } `; export const MessageContent = styled.div` min-width: 0; flex-grow: 1; display: flex; // overflow: hidden; flex-direction: column; justify-content: center; font-size: var(--text-size); `; export const DetailBase = styled.div` flex-shrink: 0; gap: 4px; font-size: 10px; display: inline-flex; color: var(--tertiary-foreground); .edited { cursor: default; &::selection { background-color: transparent; color: var(--tertiary-foreground); } } `; export const MessageDetail = observer( ({ message, position }: { message: Message; position: "left" | "top" }) => { const dict = useDictionary(); if (position === "left") { if (message.edited) { return ( <> <time className="copyTime"> <i className="copyBracket">[</i> {dayjs(decodeTime(message._id)).format( dict.dayjs?.timeFormat, )} <i className="copyBracket">]</i> </time> <span className="edited"> <Tooltip content={dayjs(message.edited).format("LLLL")}> <Text id="app.main.channel.edited" /> </Tooltip> </span> </> ); } return ( <> <time> <i className="copyBracket">[</i> {dayjs(decodeTime(message._id)).format( dict.dayjs?.timeFormat, )} <i className="copyBracket">]</i> </time> </> ); } return ( <DetailBase> <time>{dayjs(decodeTime(message._id)).calendar()}</time> {message.edited && ( <Tooltip content={dayjs(message.edited).format("LLLL")}> <span className="edited"> <Text id="app.main.channel.edited" /> </span> </Tooltip> )} </DetailBase> ); }, );