mirror of
https://github.com/revoltchat/revite.git
synced 2025-01-15 08:51:27 -05:00
f509acbe80
Co-authored-by: Kuhn Chris <kuhnchris+github@kuhnchris.eu> Co-authored-by: Paul Makles <paulmakles@gmail.com>
277 lines
6.2 KiB
TypeScript
277 lines
6.2 KiB
TypeScript
import { observer } from "mobx-react-lite";
|
|
import { Message } from "revolt.js/dist/maps/Messages";
|
|
import styled, { css, keyframes } from "styled-components/macro";
|
|
import { decodeTime } from "ulid";
|
|
|
|
import { Text } from "preact-i18n";
|
|
|
|
import { useDictionary } from "../../../lib/i18n";
|
|
import { isTouchscreenDevice } from "../../../lib/isTouchscreenDevice";
|
|
|
|
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;
|
|
|
|
${() =>
|
|
isTouchscreenDevice &&
|
|
css`
|
|
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`
|
|
// Position relatively so we can put
|
|
// the overlay in the right place.
|
|
position: relative;
|
|
|
|
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>
|
|
);
|
|
},
|
|
);
|