|
|
|
import { RefObject, useState } from 'react';
|
|
|
|
import { Mention, MentionsInput } from 'react-mentions';
|
|
|
|
import { getConversationController } from '../../../session/conversations';
|
|
|
|
import {
|
|
|
|
useSelectedConversationKey,
|
|
|
|
useSelectedIsBlocked,
|
|
|
|
useSelectedIsKickedFromGroup,
|
|
|
|
useSelectedIsLeft,
|
|
|
|
} from '../../../state/selectors/selectedConversation';
|
|
|
|
import { HTMLDirection, useHTMLDirection } from '../../../util/i18n';
|
|
|
|
import { updateDraftForConversation } from '../SessionConversationDrafts';
|
|
|
|
import { renderEmojiQuickResultRow, searchEmojiForQuery } from './EmojiQuickResult';
|
|
|
|
import { renderUserMentionRow, styleForCompositionBoxSuggestions } from './UserMentions';
|
|
|
|
|
|
|
|
const sendMessageStyle = (dir?: HTMLDirection) => {
|
|
|
|
return {
|
|
|
|
control: {
|
|
|
|
wordBreak: 'break-all',
|
|
|
|
},
|
|
|
|
input: {
|
|
|
|
overflow: 'auto',
|
|
|
|
maxHeight: '50vh',
|
|
|
|
wordBreak: 'break-word',
|
|
|
|
padding: '0px',
|
|
|
|
margin: '0px',
|
|
|
|
},
|
|
|
|
highlighter: {
|
|
|
|
boxSizing: 'border-box',
|
|
|
|
overflow: 'hidden',
|
|
|
|
maxHeight: '50vh',
|
|
|
|
},
|
|
|
|
flexGrow: 1,
|
|
|
|
minHeight: '24px',
|
|
|
|
width: '100%',
|
|
|
|
...styleForCompositionBoxSuggestions(dir),
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
draft: string;
|
|
|
|
setDraft: (draft: string) => void;
|
|
|
|
container: HTMLDivElement | null;
|
|
|
|
textAreaRef: RefObject<HTMLTextAreaElement>;
|
|
|
|
fetchUsersForGroup: (query: string, callback: (data: any) => void) => void;
|
|
|
|
typingEnabled: boolean;
|
|
|
|
onKeyDown: (event: any) => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const CompositionTextArea = (props: Props) => {
|
|
|
|
const { draft, setDraft, container, textAreaRef, fetchUsersForGroup, typingEnabled, onKeyDown } =
|
|
|
|
props;
|
|
|
|
|
|
|
|
const [lastBumpTypingMessageLength, setLastBumpTypingMessageLength] = useState(0);
|
|
|
|
|
|
|
|
const selectedConversationKey = useSelectedConversationKey();
|
|
|
|
const htmlDirection = useHTMLDirection();
|
|
|
|
const isKickedFromGroup = useSelectedIsKickedFromGroup();
|
|
|
|
const left = useSelectedIsLeft();
|
|
|
|
const isBlocked = useSelectedIsBlocked();
|
|
|
|
|
|
|
|
if (!selectedConversationKey) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const makeMessagePlaceHolderText = () => {
|
|
|
|
if (isKickedFromGroup) {
|
|
|
|
return window.i18n('youGotKickedFromGroup');
|
|
|
|
}
|
|
|
|
if (left) {
|
|
|
|
return window.i18n('youLeftTheGroup');
|
|
|
|
}
|
|
|
|
if (isBlocked) {
|
|
|
|
return window.i18n('unblockToSend');
|
|
|
|
}
|
|
|
|
return window.i18n('sendMessage');
|
|
|
|
};
|
|
|
|
|
|
|
|
const messagePlaceHolder = makeMessagePlaceHolderText();
|
|
|
|
const neverMatchingRegex = /($a)/;
|
|
|
|
|
|
|
|
const style = sendMessageStyle(htmlDirection);
|
|
|
|
|
|
|
|
const handleOnChange = (event: any) => {
|
|
|
|
if (!selectedConversationKey) {
|
|
|
|
throw new Error('selectedConversationKey is needed');
|
|
|
|
}
|
|
|
|
|
|
|
|
const newDraft = event.target.value ?? '';
|
|
|
|
setDraft(newDraft);
|
|
|
|
updateDraftForConversation({ conversationKey: selectedConversationKey, draft: newDraft });
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleKeyUp = async () => {
|
|
|
|
if (!selectedConversationKey) {
|
|
|
|
throw new Error('selectedConversationKey is needed');
|
|
|
|
}
|
|
|
|
/** Called whenever the user changes the message composition field. But only fires if there's content in the message field after the change.
|
|
|
|
Also, check for a message length change before firing it up, to avoid catching ESC, tab, or whatever which is not typing
|
|
|
|
*/
|
|
|
|
if (draft && draft.length && draft.length !== lastBumpTypingMessageLength) {
|
|
|
|
const conversationModel = getConversationController().get(selectedConversationKey);
|
|
|
|
if (!conversationModel) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
conversationModel.throttledBumpTyping();
|
|
|
|
setLastBumpTypingMessageLength(draft.length);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<MentionsInput
|
|
|
|
value={draft}
|
|
|
|
onChange={handleOnChange}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-misused-promises
|
|
|
|
onKeyDown={onKeyDown}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-misused-promises
|
|
|
|
onKeyUp={handleKeyUp}
|
|
|
|
placeholder={messagePlaceHolder}
|
|
|
|
spellCheck={true}
|
|
|
|
dir={htmlDirection}
|
|
|
|
inputRef={textAreaRef}
|
|
|
|
disabled={!typingEnabled}
|
|
|
|
rows={1}
|
|
|
|
data-testid="message-input-text-area"
|
|
|
|
style={style}
|
|
|
|
suggestionsPortalHost={container as any}
|
|
|
|
forceSuggestionsAboveCursor={true} // force mentions to be rendered on top of the cursor, this is working with a fork of react-mentions for now
|
|
|
|
>
|
|
|
|
<Mention
|
|
|
|
appendSpaceOnAdd={true}
|
|
|
|
// this will be cleaned on cleanMentions()
|
|
|
|
markup="@ᅭ__id__ᅲ__display__ᅭ" // ᅭ = \uFFD2 is one of the forbidden char for a display name (check displayNameRegex)
|
|
|
|
trigger="@"
|
|
|
|
// this is only for the composition box visible content. The real stuff on the backend box is the @markup
|
|
|
|
displayTransform={(_id, display) =>
|
|
|
|
htmlDirection === 'rtl' ? `${display}@` : `@${display}`
|
|
|
|
}
|
|
|
|
data={fetchUsersForGroup}
|
|
|
|
renderSuggestion={renderUserMentionRow}
|
|
|
|
/>
|
|
|
|
<Mention
|
|
|
|
trigger=":"
|
|
|
|
markup="__id__"
|
|
|
|
appendSpaceOnAdd={true}
|
|
|
|
regex={neverMatchingRegex}
|
|
|
|
data={searchEmojiForQuery}
|
|
|
|
renderSuggestion={renderEmojiQuickResultRow}
|
|
|
|
/>
|
|
|
|
</MentionsInput>
|
|
|
|
);
|
|
|
|
};
|