cinny/src/app/organisms/room/RoomViewInput.jsx

409 lines
13 KiB
React
Raw Normal View History

2021-08-04 18:52:59 +09:00
/* eslint-disable react/prop-types */
import React, { useState, useEffect, useRef } from 'react';
import PropTypes from 'prop-types';
2021-08-31 22:13:31 +09:00
import './RoomViewInput.scss';
2021-08-04 18:52:59 +09:00
import TextareaAutosize from 'react-autosize-textarea';
import initMatrix from '../../../client/initMatrix';
import cons from '../../../client/state/cons';
2021-08-09 01:26:34 +09:00
import settings from '../../../client/state/settings';
2021-08-14 13:49:29 +09:00
import { openEmojiBoard } from '../../../client/action/navigation';
import navigation from '../../../client/state/navigation';
import { bytesToSize, getEventCords } from '../../../util/common';
2021-08-11 16:59:01 +09:00
import { getUsername } from '../../../util/matrixUtil';
import colorMXID from '../../../util/colorMXID';
2021-08-04 18:52:59 +09:00
import Text from '../../atoms/text/Text';
import RawIcon from '../../atoms/system-icons/RawIcon';
import IconButton from '../../atoms/button/IconButton';
import ScrollView from '../../atoms/scroll/ScrollView';
2021-08-11 16:59:01 +09:00
import { MessageReply } from '../../molecules/message/Message';
2021-08-04 18:52:59 +09:00
import CirclePlusIC from '../../../../public/res/ic/outlined/circle-plus.svg';
import EmojiIC from '../../../../public/res/ic/outlined/emoji.svg';
import SendIC from '../../../../public/res/ic/outlined/send.svg';
import ShieldIC from '../../../../public/res/ic/outlined/shield.svg';
import VLCIC from '../../../../public/res/ic/outlined/vlc.svg';
import VolumeFullIC from '../../../../public/res/ic/outlined/volume-full.svg';
2021-08-09 01:26:34 +09:00
import MarkdownIC from '../../../../public/res/ic/outlined/markdown.svg';
2021-08-04 18:52:59 +09:00
import FileIC from '../../../../public/res/ic/outlined/file.svg';
2021-08-11 16:59:01 +09:00
import CrossIC from '../../../../public/res/ic/outlined/cross.svg';
2021-08-04 18:52:59 +09:00
const CMD_REGEX = /(^\/|:|@)(\S*)$/;
2021-08-04 18:52:59 +09:00
let isTyping = false;
2021-08-09 01:26:34 +09:00
let isCmdActivated = false;
let cmdCursorPos = null;
2021-08-31 22:13:31 +09:00
function RoomViewInput({
roomId, roomTimeline, viewEvent,
2021-08-04 18:52:59 +09:00
}) {
const [attachment, setAttachment] = useState(null);
2021-08-09 01:26:34 +09:00
const [isMarkdown, setIsMarkdown] = useState(settings.isMarkdown);
2021-08-11 16:59:01 +09:00
const [replyTo, setReplyTo] = useState(null);
2021-08-04 18:52:59 +09:00
const textAreaRef = useRef(null);
const inputBaseRef = useRef(null);
const uploadInputRef = useRef(null);
const uploadProgressRef = useRef(null);
const rightOptionsRef = useRef(null);
2021-08-04 18:52:59 +09:00
const TYPING_TIMEOUT = 5000;
const mx = initMatrix.matrixClient;
const { roomsInput } = initMatrix;
function requestFocusInput() {
if (textAreaRef === null) return;
textAreaRef.current.focus();
}
2021-08-09 01:26:34 +09:00
useEffect(() => {
settings.on(cons.events.settings.MARKDOWN_TOGGLED, setIsMarkdown);
viewEvent.on('focus_msg_input', requestFocusInput);
2021-08-09 01:26:34 +09:00
return () => {
settings.removeListener(cons.events.settings.MARKDOWN_TOGGLED, setIsMarkdown);
viewEvent.removeListener('focus_msg_input', requestFocusInput);
2021-08-09 01:26:34 +09:00
};
}, []);
2021-08-04 18:52:59 +09:00
const sendIsTyping = (isT) => {
mx.sendTyping(roomId, isT, isT ? TYPING_TIMEOUT : undefined);
isTyping = isT;
if (isT === true) {
setTimeout(() => {
if (isTyping) sendIsTyping(false);
}, TYPING_TIMEOUT);
}
};
function uploadingProgress(myRoomId, { loaded, total }) {
if (myRoomId !== roomId) return;
const progressPer = Math.round((loaded * 100) / total);
uploadProgressRef.current.textContent = `Uploading: ${bytesToSize(loaded)}/${bytesToSize(total)} (${progressPer}%)`;
inputBaseRef.current.style.backgroundImage = `linear-gradient(90deg, var(--bg-surface-hover) ${progressPer}%, var(--bg-surface-low) ${progressPer}%)`;
}
function clearAttachment(myRoomId) {
if (roomId !== myRoomId) return;
setAttachment(null);
inputBaseRef.current.style.backgroundImage = 'unset';
uploadInputRef.current.value = null;
}
function rightOptionsA11Y(A11Y) {
const rightOptions = rightOptionsRef.current.children;
for (let index = 0; index < rightOptions.length; index += 1) {
rightOptions[index].tabIndex = A11Y ? 0 : -1;
}
}
2021-08-09 01:26:34 +09:00
function activateCmd(prefix) {
isCmdActivated = true;
rightOptionsA11Y(false);
2021-08-09 01:26:34 +09:00
viewEvent.emit('cmd_activate', prefix);
}
function deactivateCmd() {
isCmdActivated = false;
cmdCursorPos = null;
rightOptionsA11Y(true);
2021-08-09 01:26:34 +09:00
}
function deactivateCmdAndEmit() {
deactivateCmd();
viewEvent.emit('cmd_deactivate');
}
2021-08-09 01:26:34 +09:00
function setCursorPosition(pos) {
setTimeout(() => {
textAreaRef.current.focus();
textAreaRef.current.setSelectionRange(pos, pos);
}, 0);
}
function replaceCmdWith(msg, cursor, replacement) {
if (msg === null) return null;
const targetInput = msg.slice(0, cursor);
const cmdParts = targetInput.match(CMD_REGEX);
const leadingInput = msg.slice(0, cmdParts.index);
if (replacement.length > 0) setCursorPosition(leadingInput.length + replacement.length);
return leadingInput + replacement + msg.slice(cursor);
}
function firedCmd(cmdData) {
const msg = textAreaRef.current.value;
textAreaRef.current.value = replaceCmdWith(
msg, cmdCursorPos, typeof cmdData?.replace !== 'undefined' ? cmdData.replace : '',
);
deactivateCmd();
}
2021-08-25 16:32:18 +09:00
function focusInput() {
if (settings.isTouchScreenDevice) return;
2021-08-25 16:32:18 +09:00
textAreaRef.current.focus();
}
function setUpReply(userId, eventId, body) {
setReplyTo({ userId, eventId, body });
roomsInput.setReplyTo(roomId, { userId, eventId, body });
2021-08-25 16:32:18 +09:00
focusInput();
2021-08-11 16:59:01 +09:00
}
2021-08-04 18:52:59 +09:00
useEffect(() => {
roomsInput.on(cons.events.roomsInput.UPLOAD_PROGRESS_CHANGES, uploadingProgress);
roomsInput.on(cons.events.roomsInput.ATTACHMENT_CANCELED, clearAttachment);
roomsInput.on(cons.events.roomsInput.FILE_UPLOADED, clearAttachment);
2021-08-09 01:26:34 +09:00
viewEvent.on('cmd_fired', firedCmd);
navigation.on(cons.events.navigation.REPLY_TO_CLICKED, setUpReply);
2021-08-04 18:52:59 +09:00
if (textAreaRef?.current !== null) {
isTyping = false;
2021-08-25 16:32:18 +09:00
focusInput();
2021-08-04 18:52:59 +09:00
textAreaRef.current.value = roomsInput.getMessage(roomId);
setAttachment(roomsInput.getAttachment(roomId));
2021-08-11 16:59:01 +09:00
setReplyTo(roomsInput.getReplyTo(roomId));
2021-08-04 18:52:59 +09:00
}
return () => {
roomsInput.removeListener(cons.events.roomsInput.UPLOAD_PROGRESS_CHANGES, uploadingProgress);
roomsInput.removeListener(cons.events.roomsInput.ATTACHMENT_CANCELED, clearAttachment);
roomsInput.removeListener(cons.events.roomsInput.FILE_UPLOADED, clearAttachment);
2021-08-09 01:26:34 +09:00
viewEvent.removeListener('cmd_fired', firedCmd);
navigation.removeListener(cons.events.navigation.REPLY_TO_CLICKED, setUpReply);
2021-08-09 01:26:34 +09:00
if (isCmdActivated) deactivateCmd();
2021-08-04 18:52:59 +09:00
if (textAreaRef?.current === null) return;
const msg = textAreaRef.current.value;
inputBaseRef.current.style.backgroundImage = 'unset';
if (msg.trim() === '') {
roomsInput.setMessage(roomId, '');
return;
}
roomsInput.setMessage(roomId, msg);
};
}, [roomId]);
const sendMessage = async () => {
2021-08-04 18:52:59 +09:00
const msgBody = textAreaRef.current.value;
if (roomsInput.isSending(roomId)) return;
if (msgBody.trim() === '' && attachment === null) return;
sendIsTyping(false);
roomsInput.setMessage(roomId, msgBody);
if (attachment !== null) {
roomsInput.setAttachment(roomId, attachment);
}
textAreaRef.current.disabled = true;
textAreaRef.current.style.cursor = 'not-allowed';
await roomsInput.sendInput(roomId);
textAreaRef.current.disabled = false;
textAreaRef.current.style.cursor = 'unset';
2021-08-25 16:32:18 +09:00
focusInput();
2021-08-04 18:52:59 +09:00
textAreaRef.current.value = roomsInput.getMessage(roomId);
viewEvent.emit('message_sent');
textAreaRef.current.style.height = 'unset';
2021-08-11 16:59:01 +09:00
if (replyTo !== null) setReplyTo(null);
};
2021-08-04 18:52:59 +09:00
function processTyping(msg) {
const isEmptyMsg = msg === '';
if (isEmptyMsg && isTyping) {
sendIsTyping(false);
return;
}
if (!isEmptyMsg && !isTyping) {
sendIsTyping(true);
}
}
2021-08-09 01:26:34 +09:00
function getCursorPosition() {
return textAreaRef.current.selectionStart;
}
function recognizeCmd(rawInput) {
const cursor = getCursorPosition();
const targetInput = rawInput.slice(0, cursor);
const cmdParts = targetInput.match(CMD_REGEX);
if (cmdParts === null) {
if (isCmdActivated) deactivateCmdAndEmit();
2021-08-09 01:26:34 +09:00
return;
}
const cmdPrefix = cmdParts[1];
const cmdSlug = cmdParts[2];
2021-08-10 17:42:00 +09:00
if (cmdPrefix === ':') {
// skip emoji autofill command if link is suspected.
const checkForLink = targetInput.slice(0, cmdParts.index);
if (checkForLink.match(/(http|https|mailto|matrix|ircs|irc)$/)) {
deactivateCmdAndEmit();
2021-08-10 17:42:00 +09:00
return;
}
}
2021-08-09 01:26:34 +09:00
cmdCursorPos = cursor;
if (cmdSlug === '') {
activateCmd(cmdPrefix);
return;
}
if (!isCmdActivated) activateCmd(cmdPrefix);
viewEvent.emit('cmd_process', cmdPrefix, cmdSlug);
}
const handleMsgTyping = (e) => {
2021-08-04 18:52:59 +09:00
const msg = e.target.value;
2021-08-09 01:26:34 +09:00
recognizeCmd(e.target.value);
if (!isCmdActivated) processTyping(msg);
};
2021-08-04 18:52:59 +09:00
const handleKeyDown = (e) => {
2021-08-04 18:52:59 +09:00
if (e.keyCode === 13 && e.shiftKey === false) {
e.preventDefault();
sendMessage();
}
};
2021-08-04 18:52:59 +09:00
const handlePaste = (e) => {
if (e.clipboardData === false) {
return;
}
if (e.clipboardData.items === undefined) {
return;
}
for (let i = 0; i < e.clipboardData.items.length; i += 1) {
const item = e.clipboardData.items[i];
if (item.type.indexOf('image') !== -1) {
const image = item.getAsFile();
if (attachment === null) {
setAttachment(image);
if (image !== null) {
roomsInput.setAttachment(roomId, image);
return;
}
} else {
return;
}
}
}
};
2021-08-04 18:52:59 +09:00
function addEmoji(emoji) {
textAreaRef.current.value += emoji.unicode;
textAreaRef.current.focus();
2021-08-04 18:52:59 +09:00
}
const handleUploadClick = () => {
2021-08-04 18:52:59 +09:00
if (attachment === null) uploadInputRef.current.click();
else {
roomsInput.cancelAttachment(roomId);
}
};
2021-08-04 18:52:59 +09:00
function uploadFileChange(e) {
const file = e.target.files.item(0);
setAttachment(file);
if (file !== null) roomsInput.setAttachment(roomId, file);
}
const canISend = roomTimeline.room.currentState.maySendMessage(mx.getUserId());
2021-08-04 18:52:59 +09:00
function renderInputs() {
if (!canISend) {
return (
<Text className="room-input__alert">You do not have permission to post to this room</Text>
);
}
2021-08-04 18:52:59 +09:00
return (
<>
2021-08-31 22:13:31 +09:00
<div className={`room-input__option-container${attachment === null ? '' : ' room-attachment__option'}`}>
2021-08-04 18:52:59 +09:00
<input onChange={uploadFileChange} style={{ display: 'none' }} ref={uploadInputRef} type="file" />
<IconButton onClick={handleUploadClick} tooltip={attachment === null ? 'Upload' : 'Cancel'} src={CirclePlusIC} />
</div>
2021-08-31 22:13:31 +09:00
<div ref={inputBaseRef} className="room-input__input-container">
{roomTimeline.isEncrypted() && <RawIcon size="extra-small" src={ShieldIC} />}
2021-08-04 18:52:59 +09:00
<ScrollView autoHide>
2021-08-31 22:13:31 +09:00
<Text className="room-input__textarea-wrapper">
2021-08-04 18:52:59 +09:00
<TextareaAutosize
id="message-textarea"
2021-08-04 18:52:59 +09:00
ref={textAreaRef}
onChange={handleMsgTyping}
onPaste={handlePaste}
2021-08-04 18:52:59 +09:00
onKeyDown={handleKeyDown}
placeholder="Send a message..."
/>
</Text>
</ScrollView>
2021-08-09 01:26:34 +09:00
{isMarkdown && <RawIcon size="extra-small" src={MarkdownIC} />}
2021-08-04 18:52:59 +09:00
</div>
<div ref={rightOptionsRef} className="room-input__option-container">
2021-08-14 13:49:29 +09:00
<IconButton
onClick={(e) => {
const cords = getEventCords(e);
cords.x += (document.dir === 'rtl' ? -80 : 80);
cords.y -= 250;
openEmojiBoard(cords, addEmoji);
2021-08-14 13:49:29 +09:00
}}
tooltip="Emoji"
src={EmojiIC}
2021-08-04 18:52:59 +09:00
/>
<IconButton onClick={sendMessage} tooltip="Send" src={SendIC} />
</div>
</>
);
}
function attachFile() {
const fileType = attachment.type.slice(0, attachment.type.indexOf('/'));
return (
2021-08-31 22:13:31 +09:00
<div className="room-attachment">
<div className={`room-attachment__preview${fileType !== 'image' ? ' room-attachment__icon' : ''}`}>
2021-08-04 18:52:59 +09:00
{fileType === 'image' && <img alt={attachment.name} src={URL.createObjectURL(attachment)} />}
{fileType === 'video' && <RawIcon src={VLCIC} />}
{fileType === 'audio' && <RawIcon src={VolumeFullIC} />}
{fileType !== 'image' && fileType !== 'video' && fileType !== 'audio' && <RawIcon src={FileIC} />}
</div>
2021-08-31 22:13:31 +09:00
<div className="room-attachment__info">
2021-08-04 18:52:59 +09:00
<Text variant="b1">{attachment.name}</Text>
<Text variant="b3"><span ref={uploadProgressRef}>{`size: ${bytesToSize(attachment.size)}`}</span></Text>
</div>
</div>
);
}
2021-08-11 16:59:01 +09:00
function attachReply() {
return (
2021-08-31 22:13:31 +09:00
<div className="room-reply">
2021-08-11 16:59:01 +09:00
<IconButton
onClick={() => {
roomsInput.cancelReplyTo(roomId);
setReplyTo(null);
}}
src={CrossIC}
tooltip="Cancel reply"
size="extra-small"
/>
<MessageReply
userId={replyTo.userId}
name={getUsername(replyTo.userId)}
color={colorMXID(replyTo.userId)}
body={replyTo.body}
2021-08-11 16:59:01 +09:00
/>
</div>
);
}
2021-08-04 18:52:59 +09:00
return (
<>
2021-08-11 16:59:01 +09:00
{ replyTo !== null && attachReply()}
2021-08-04 18:52:59 +09:00
{ attachment !== null && attachFile() }
2021-08-31 22:13:31 +09:00
<form className="room-input" onSubmit={(e) => { e.preventDefault(); }}>
2021-08-04 18:52:59 +09:00
{
renderInputs()
2021-08-04 18:52:59 +09:00
}
</form>
</>
);
}
2021-08-31 22:13:31 +09:00
RoomViewInput.propTypes = {
2021-08-04 18:52:59 +09:00
roomId: PropTypes.string.isRequired,
roomTimeline: PropTypes.shape({}).isRequired,
viewEvent: PropTypes.shape({}).isRequired,
};
2021-08-31 22:13:31 +09:00
export default RoomViewInput;