import { Popover } from 'antd'; import React, { FC, useMemo, useState } from 'react'; import { useRecoilValue } from 'recoil'; import { Transforms, createEditor, BaseEditor, Text, Descendant, Editor } from 'slate'; import { Slate, DefaultPlaceholder, Editable, withReact, ReactEditor, useSelected, useFocused, } from 'slate-react'; import dynamic from 'next/dynamic'; import classNames from 'classnames'; import WebsocketService from '../../../services/websocket-service'; import { websocketServiceAtom } from '../../stores/ClientConfigStore'; import { MessageType } from '../../../interfaces/socket-events'; import styles from './ChatTextField.module.scss'; // Lazy loaded components const EmojiPicker = dynamic(() => import('./EmojiPicker').then(mod => mod.EmojiPicker), { ssr: false, }); const SendOutlined = dynamic(() => import('@ant-design/icons/SendOutlined'), { ssr: false, }); const SmileOutlined = dynamic(() => import('@ant-design/icons/SmileOutlined'), { ssr: false, }); type CustomElement = { type: 'paragraph' | 'span'; children: CustomText[] } | ImageNode; type CustomText = { text: string }; type EmptyText = { text: string; }; type ImageNode = { type: 'image'; alt: string; src: string; name: string; children: EmptyText[]; }; declare module 'slate' { interface CustomTypes { Editor: BaseEditor & ReactEditor; Element: CustomElement; Text: CustomText; } } const Image = p => { const { attributes, element, children } = p; const selected = useSelected(); const focused = useFocused(); return ( {element.alt} {children} ); }; const withImages = editor => { const { isVoid } = editor; // eslint-disable-next-line no-param-reassign editor.isVoid = element => (element.type === 'image' ? true : isVoid(element)); // eslint-disable-next-line no-param-reassign editor.isInline = element => element.type === 'image'; return editor; }; const serialize = node => { if (Text.isText(node)) { const string = node.text; return string; } let children; if (node.children.length === 0) { children = [{ text: '' }]; } else { children = node.children?.map(n => serialize(n)).join(''); } switch (node.type) { case 'paragraph': return `

${children}

`; case 'image': return `${node.alt}`; default: return children; } }; const getCharacterCount = node => { if (Text.isText(node)) { return node.text.length; } if (node.type === 'image') { return 5; } let count = 0; node.children.forEach(child => { count += getCharacterCount(child); }); return count; }; export type ChatTextFieldProps = { defaultText?: string; enabled: boolean; focusInput: boolean; }; const characterLimit = 300; export const ChatTextField: FC = ({ defaultText, enabled, focusInput }) => { const [showEmojis, setShowEmojis] = useState(false); const [characterCount, setCharacterCount] = useState(defaultText?.length); const websocketService = useRecoilValue(websocketServiceAtom); const editor = useMemo(() => withReact(withImages(createEditor())), []); const defaultEditorValue: Descendant[] = [ { type: 'paragraph', children: [{ text: defaultText || '' }], }, ]; const sendMessage = () => { if (!websocketService) { console.log('websocketService is not defined'); return; } let message = serialize(editor); // Strip the opening and closing

tags. message = message.replace(/^

|<\/p>$/g, ''); websocketService.send({ type: MessageType.CHAT, body: message }); // Clear the editor. Transforms.delete(editor, { at: { anchor: Editor.start(editor, []), focus: Editor.end(editor, []), }, }); setCharacterCount(0); }; const createImageNode = (alt, src, name): ImageNode => ({ type: 'image', alt, src, name, children: [{ text: '' }], }); const insertImage = (url, name) => { if (!url) return; const image = createImageNode(name, url, name); Transforms.insertNodes(editor, image); Editor.normalize(editor, { force: true }); }; // Native emoji const onEmojiSelect = (emoji: string) => { ReactEditor.focus(editor); Transforms.insertText(editor, emoji); }; const onCustomEmojiSelect = (name: string, emoji: string) => { ReactEditor.focus(editor); insertImage(emoji, name); }; const onKeyDown = (e: React.KeyboardEvent) => { const charCount = getCharacterCount(editor) + 1; // Send the message when hitting enter. if (e.key === 'Enter') { e.preventDefault(); sendMessage(); return; } // Always allow backspace. if (e.key === 'Backspace') { setCharacterCount(charCount - 1); return; } // Limit the number of characters. if (charCount + 1 > characterLimit) { e.preventDefault(); } setCharacterCount(charCount + 1); }; const onPaste = (e: React.ClipboardEvent) => { const text = e.clipboardData.getData('text/plain'); const { length } = text; if (characterCount + length > characterLimit) { e.preventDefault(); } }; const renderElement = p => { switch (p.element.type) { case 'image': return ; default: return

; } }; return (

= characterLimit && styles.maxCharacters, )} > ( {children} )} placeholder={enabled ? 'Send a message to chat' : 'Chat is currently unavailable.'} style={{ width: '100%' }} role="textbox" aria-label="Chat text input" autoFocus={focusInput} /> } trigger="click" placement="topRight" onOpenChange={open => setShowEmojis(open)} open={showEmojis} /> {enabled && (
)}
); };