revite/src/components/common/messaging/MessageBox.tsx

389 lines
13 KiB
TypeScript
Raw Normal View History

2021-06-20 15:30:42 -04:00
import { ulid } from "ulid";
2021-06-26 17:16:43 -04:00
import { Text } from "preact-i18n";
import { Channel } from "revolt.js";
import styled from "styled-components";
import { dispatch } from "../../../redux";
2021-06-20 15:30:42 -04:00
import { defer } from "../../../lib/defer";
2021-06-20 15:36:52 -04:00
import IconButton from "../../ui/IconButton";
import { PermissionTooltip } from "../Tooltip";
import { Send } from '@styled-icons/boxicons-solid';
2021-06-23 13:26:41 -04:00
import { debounce } from "../../../lib/debounce";
import Axios, { CancelTokenSource } from "axios";
import { useTranslation } from "../../../lib/i18n";
2021-06-23 13:26:41 -04:00
import { Reply } from "../../../redux/reducers/queue";
import { connectState } from "../../../redux/connector";
2021-06-26 17:16:43 -04:00
import { SoundContext } from "../../../context/Settings";
import { takeError } from "../../../context/revoltjs/util";
import TextAreaAutoSize from "../../../lib/TextAreaAutoSize";
2021-06-26 17:16:43 -04:00
import AutoComplete, { useAutoComplete } from "../AutoComplete";
import { ChannelPermission } from "revolt.js/dist/api/permissions";
import { AppContext } from "../../../context/revoltjs/RevoltClient";
2021-06-26 17:16:43 -04:00
import { useChannelPermission } from "../../../context/revoltjs/hooks";
import { isTouchscreenDevice } from "../../../lib/isTouchscreenDevice";
import { internalEmit, internalSubscribe } from "../../../lib/eventEmitter";
import { useCallback, useContext, useEffect, useState } from "preact/hooks";
import { useIntermediate } from "../../../context/intermediate/Intermediate";
import { FileUploader, grabFiles, uploadFile } from "../../../context/revoltjs/FileUploads";
import { SingletonMessageRenderer, SMOOTH_SCROLL_ON_RECEIVE } from "../../../lib/renderer/Singleton";
import { ShieldX } from "@styled-icons/boxicons-regular";
2021-06-20 15:30:42 -04:00
2021-06-23 13:26:41 -04:00
import ReplyBar from "./bars/ReplyBar";
import FilePreview from './bars/FilePreview';
type Props = {
2021-06-20 15:30:42 -04:00
channel: Channel;
draft?: string;
};
export type UploadState =
| { type: "none" }
| { type: "attached"; files: File[] }
| { type: "uploading"; files: File[]; percent: number; cancel: CancelTokenSource }
| { type: "sending"; files: File[] }
| { type: "failed"; files: File[]; error: string };
const Base = styled.div`
display: flex;
padding: 0 12px;
background: var(--message-box);
textarea {
font-size: .875rem;
background: transparent;
}
`;
2021-06-26 17:16:43 -04:00
const Blocked = styled.div`
display: flex;
align-items: center;
padding: 14px 0;
2021-06-26 17:16:43 -04:00
user-select: none;
font-size: .875rem;
color: var(--tertiary-foreground);
svg {
flex-shrink: 0;
2021-07-01 12:54:11 -04:00
margin-inline-end: 10px;
}
2021-06-26 17:16:43 -04:00
`;
const Action = styled.div`
display: grid;
place-items: center;
`;
// ! FIXME: add to app config and load from app config
export const CAN_UPLOAD_AT_ONCE = 5;
function MessageBox({ channel, draft }: Props) {
const [ uploadState, setUploadState ] = useState<UploadState>({ type: 'none' });
2021-06-23 13:26:41 -04:00
const [ typing, setTyping ] = useState<boolean | number>(false);
const [ replies, setReplies ] = useState<Reply[]>([]);
const playSound = useContext(SoundContext);
const { openScreen } = useIntermediate();
2021-06-20 15:30:42 -04:00
const client = useContext(AppContext);
const translate = useTranslation();
2021-06-20 15:30:42 -04:00
2021-06-26 17:16:43 -04:00
const permissions = useChannelPermission(channel._id);
if (!(permissions & ChannelPermission.SendMessage)) {
return (
<Base>
<Blocked>
<PermissionTooltip permission="SendMessages" placement="top">
2021-06-28 05:12:19 -04:00
<ShieldX size={22}/>
</PermissionTooltip>
<Text id="app.main.channel.misc.no_sending" />
</Blocked>
2021-06-26 17:16:43 -04:00
</Base>
)
}
2021-06-20 15:30:42 -04:00
function setMessage(content?: string) {
if (content) {
dispatch({
2021-06-20 15:30:42 -04:00
type: "SET_DRAFT",
channel: channel._id,
content
});
} else {
dispatch({
2021-06-20 15:30:42 -04:00
type: "CLEAR_DRAFT",
channel: channel._id
});
}
}
useEffect(() => {
function append(content: string, action: 'quote' | 'mention') {
const text =
action === "quote"
? `${content
.split("\n")
.map(x => `> ${x}`)
.join("\n")}\n\n`
: `${content} `;
if (!draft || draft.length === 0) {
setMessage(text);
} else {
setMessage(`${draft}\n${text}`);
}
}
return internalSubscribe("MessageBox", "append", append);
}, [ draft ]);
2021-06-20 15:30:42 -04:00
async function send() {
if (uploadState.type === 'uploading' || uploadState.type === 'sending') return;
2021-06-20 15:30:42 -04:00
const content = draft?.trim() ?? '';
if (uploadState.type === 'attached') return sendFile(content);
2021-06-20 15:30:42 -04:00
if (content.length === 0) return;
stopTyping();
2021-06-20 15:30:42 -04:00
setMessage();
2021-06-23 13:26:41 -04:00
setReplies([]);
playSound('outbound');
const nonce = ulid();
dispatch({
2021-06-20 15:30:42 -04:00
type: "QUEUE_ADD",
nonce,
channel: channel._id,
message: {
_id: nonce,
channel: channel._id,
author: client.user!._id,
2021-06-23 13:26:41 -04:00
content,
replies
2021-06-20 15:30:42 -04:00
}
});
defer(() => SingletonMessageRenderer.jumpToBottom(channel._id, SMOOTH_SCROLL_ON_RECEIVE));
try {
await client.channels.sendMessage(channel._id, {
content,
2021-06-23 13:26:41 -04:00
nonce,
replies
2021-06-20 15:30:42 -04:00
});
} catch (error) {
dispatch({
2021-06-20 15:30:42 -04:00
type: "QUEUE_FAIL",
error: takeError(error),
nonce
});
}
}
async function sendFile(content: string) {
if (uploadState.type !== 'attached') return;
let attachments: string[] = [];
const cancel = Axios.CancelToken.source();
const files = uploadState.files;
stopTyping();
setUploadState({ type: "uploading", files, percent: 0, cancel });
try {
for (let i=0;i<files.length&&i<CAN_UPLOAD_AT_ONCE;i++) {
const file = files[i];
attachments.push(
await uploadFile(client.configuration!.features.autumn.url, 'attachments', file, {
onUploadProgress: e =>
setUploadState({
type: "uploading",
files,
percent: Math.round(((i * 100) + (100 * e.loaded) / e.total) / Math.min(files.length, CAN_UPLOAD_AT_ONCE)),
cancel
}),
cancelToken: cancel.token
})
);
}
} catch (err) {
if (err?.message === "cancel") {
setUploadState({
type: "attached",
files
});
} else {
setUploadState({
type: "failed",
files,
error: takeError(err)
});
}
return;
}
setUploadState({
type: "sending",
files
});
const nonce = ulid();
try {
await client.channels.sendMessage(channel._id, {
content,
nonce,
2021-06-23 13:26:41 -04:00
replies,
attachments
});
} catch (err) {
setUploadState({
type: "failed",
files,
error: takeError(err)
});
return;
}
setMessage();
2021-06-23 13:26:41 -04:00
setReplies([]);
playSound('outbound');
if (files.length > CAN_UPLOAD_AT_ONCE) {
setUploadState({
type: "attached",
files: files.slice(CAN_UPLOAD_AT_ONCE)
});
} else {
setUploadState({ type: "none" });
}
}
function startTyping() {
if (typeof typing === 'number' && + new Date() < typing) return;
const ws = client.websocket;
if (ws.connected) {
setTyping(+ new Date() + 4000);
ws.send({
type: "BeginTyping",
channel: channel._id
});
}
}
function stopTyping(force?: boolean) {
if (force || typing) {
const ws = client.websocket;
if (ws.connected) {
setTyping(false);
ws.send({
type: "EndTyping",
channel: channel._id
});
}
}
}
const debouncedStopTyping = useCallback(debounce(stopTyping, 1000), [ channel._id ]);
2021-06-22 08:35:43 -04:00
const { onChange, onKeyUp, onKeyDown, onFocus, onBlur, ...autoCompleteProps } =
useAutoComplete(setMessage, {
users: { type: 'channel', id: channel._id },
channels: channel.channel_type === 'TextChannel' ? { server: channel.server } : undefined
});
2021-06-20 15:30:42 -04:00
return (
<>
2021-06-22 08:28:03 -04:00
<AutoComplete {...autoCompleteProps} />
<FilePreview state={uploadState} addFile={() => uploadState.type === 'attached' &&
grabFiles(20_000_000, files => setUploadState({ type: 'attached', files: [ ...uploadState.files, ...files ] }),
() => openScreen({ id: "error", error: "FileTooLarge" }), true)}
removeFile={index => {
if (uploadState.type !== 'attached') return;
if (uploadState.files.length === 1) {
setUploadState({ type: 'none' });
} else {
setUploadState({ type: 'attached', files: uploadState.files.filter((_, i) => index !== i) });
}
}} />
2021-06-23 13:26:41 -04:00
<ReplyBar channel={channel._id} replies={replies} setReplies={setReplies} />
<Base>
2021-06-26 17:16:43 -04:00
{ (permissions & ChannelPermission.UploadFiles) ? <Action>
<FileUploader
size={24}
behaviour='multi'
style='attachment'
fileType='attachments'
maxFileSize={20_000_000}
attached={uploadState.type !== 'none'}
uploading={uploadState.type === 'uploading' || uploadState.type === 'sending'}
remove={async () => setUploadState({ type: "none" })}
onChange={files => setUploadState({ type: "attached", files })}
cancel={() => uploadState.type === 'uploading' && uploadState.cancel.cancel("cancel")}
2021-06-22 07:08:39 -04:00
append={files => {
if (files.length === 0) return;
2021-06-22 07:08:39 -04:00
if (uploadState.type === 'none') {
setUploadState({ type: 'attached', files });
} else if (uploadState.type === 'attached') {
setUploadState({ type: 'attached', files: [ ...uploadState.files, ...files ] });
}
}}
/>
2021-06-26 17:16:43 -04:00
</Action> : undefined }
<TextAreaAutoSize
autoFocus
hideBorder
maxRows={5}
padding={14}
id="message"
value={draft ?? ''}
2021-06-22 08:28:03 -04:00
onKeyUp={onKeyUp}
onKeyDown={e => {
2021-06-22 08:28:03 -04:00
if (onKeyDown(e)) return;
if (
e.key === "ArrowUp" &&
(!draft || draft.length === 0)
) {
e.preventDefault();
internalEmit("MessageRenderer", "edit_last");
return;
}
if (!e.shiftKey && e.key === "Enter" && !isTouchscreenDevice) {
e.preventDefault();
return send();
}
debouncedStopTyping(true);
}}
placeholder={
channel.channel_type === "DirectMessage" ? translate("app.main.channel.message_who", {
person: client.users.get(client.channels.getRecipient(channel._id))?.username })
: channel.channel_type === "SavedMessages" ? translate("app.main.channel.message_saved")
: translate("app.main.channel.message_where", { channel_name: channel.name })
2021-06-20 15:36:52 -04:00
}
disabled={uploadState.type === 'uploading' || uploadState.type === 'sending'}
onChange={e => {
setMessage(e.currentTarget.value);
startTyping();
2021-06-22 08:28:03 -04:00
onChange(e);
}}
onFocus={onFocus}
onBlur={onBlur} />
2021-06-26 17:16:43 -04:00
{ isTouchscreenDevice && <Action>
<IconButton onClick={send}>
<Send size={20} />
</IconButton>
2021-06-26 17:16:43 -04:00
</Action> }
</Base>
</>
2021-06-20 15:30:42 -04:00
)
}
export default connectState<Omit<Props, "dispatch" | "draft">>(MessageBox, (state, { channel }) => {
2021-06-20 15:30:42 -04:00
return {
draft: state.drafts[channel._id]
}
}, true)