/* Copyright 2015, 2016 OpenMarket Ltd Copyright 2017 New Vector Ltd Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import React from 'react'; import ReactDOM from 'react-dom'; import PropTypes from 'prop-types'; import type SyntheticKeyboardEvent from 'react/lib/SyntheticKeyboardEvent'; import { Editor } from 'slate-react'; import { Value, Document, Event, Inline, Text, Range, Node } from 'slate'; import Html from 'slate-html-serializer'; import { Markdown as Md } from 'slate-md-serializer'; import Plain from 'slate-plain-serializer'; import PlainWithPillsSerializer from "../../../autocomplete/PlainWithPillsSerializer"; // import {Editor, EditorState, RichUtils, CompositeDecorator, Modifier, // getDefaultKeyBinding, KeyBindingUtil, ContentState, ContentBlock, SelectionState, // Entity} from 'draft-js'; import classNames from 'classnames'; import escape from 'lodash/escape'; import Promise from 'bluebird'; import MatrixClientPeg from '../../../MatrixClientPeg'; import type {MatrixClient} from 'matrix-js-sdk/lib/matrix'; import SlashCommands from '../../../SlashCommands'; import { KeyCode, isOnlyCtrlOrCmdKeyEvent } from '../../../Keyboard'; import Modal from '../../../Modal'; import sdk from '../../../index'; import { _t, _td } from '../../../languageHandler'; import Analytics from '../../../Analytics'; import dis from '../../../dispatcher'; import * as RichText from '../../../RichText'; import * as HtmlUtils from '../../../HtmlUtils'; import Autocomplete from './Autocomplete'; import {Completion} from "../../../autocomplete/Autocompleter"; import Markdown from '../../../Markdown'; import ComposerHistoryManager from '../../../ComposerHistoryManager'; import MessageComposerStore from '../../../stores/MessageComposerStore'; import {MATRIXTO_URL_PATTERN, MATRIXTO_MD_LINK_PATTERN} from '../../../linkify-matrix'; const REGEX_MATRIXTO = new RegExp(MATRIXTO_URL_PATTERN); const REGEX_MATRIXTO_MARKDOWN_GLOBAL = new RegExp(MATRIXTO_MD_LINK_PATTERN, 'g'); import {asciiRegexp, unicodeRegexp, shortnameToUnicode, emojioneList, asciiList, mapUnicodeToShort, toShort} from 'emojione'; import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; import {makeUserPermalink} from "../../../matrix-to"; import ReplyPreview from "./ReplyPreview"; import RoomViewStore from '../../../stores/RoomViewStore'; import ReplyThread from "../elements/ReplyThread"; import {ContentHelpers} from 'matrix-js-sdk'; const EMOJI_SHORTNAMES = Object.keys(emojioneList); const EMOJI_UNICODE_TO_SHORTNAME = mapUnicodeToShort(); const REGEX_EMOJI_WHITESPACE = new RegExp('(?:^|\\s)(' + asciiRegexp + ')\\s$'); const EMOJI_REGEX = new RegExp(unicodeRegexp, 'g'); const TYPING_USER_TIMEOUT = 10000, TYPING_SERVER_TIMEOUT = 30000; const ENTITY_TYPES = { AT_ROOM_PILL: 'ATROOMPILL', }; function onSendMessageFailed(err, room) { // XXX: temporary logging to try to diagnose // https://github.com/vector-im/riot-web/issues/3148 console.log('MessageComposer got send failure: ' + err.name + '('+err+')'); dis.dispatch({ action: 'message_send_failed', }); } /* * The textInput part of the MessageComposer */ export default class MessageComposerInput extends React.Component { static propTypes = { // a callback which is called when the height of the composer is // changed due to a change in content. onResize: PropTypes.func, // js-sdk Room object room: PropTypes.object.isRequired, // called with current plaintext content (as a string) whenever it changes onContentChanged: PropTypes.func, onFilesPasted: PropTypes.func, onInputStateChanged: PropTypes.func, }; /* static getKeyBinding(ev: SyntheticKeyboardEvent): string { // Restrict a subset of key bindings to ONLY having ctrl/meta* pressed and // importantly NOT having alt, shift, meta/ctrl* pressed. draft-js does not // handle this in `getDefaultKeyBinding` so we do it ourselves here. // // * if macOS, read second option const ctrlCmdCommand = { // C-m => Toggles between rich text and markdown modes [KeyCode.KEY_M]: 'toggle-mode', [KeyCode.KEY_B]: 'bold', [KeyCode.KEY_I]: 'italic', [KeyCode.KEY_U]: 'underline', [KeyCode.KEY_J]: 'code', [KeyCode.KEY_O]: 'split-block', }[ev.keyCode]; if (ctrlCmdCommand) { if (!isOnlyCtrlOrCmdKeyEvent(ev)) { return null; } return ctrlCmdCommand; } // Handle keys such as return, left and right arrows etc. return getDefaultKeyBinding(ev); } static getBlockStyle(block: ContentBlock): ?string { if (block.getType() === 'strikethrough') { return 'mx_Markdown_STRIKETHROUGH'; } return null; } */ client: MatrixClient; autocomplete: Autocomplete; historyManager: ComposerHistoryManager; constructor(props, context) { super(props, context); const isRichtextEnabled = SettingsStore.getValue('MessageComposerInput.isRichTextEnabled'); Analytics.setRichtextMode(isRichtextEnabled); this.state = { // whether we're in rich text or markdown mode isRichtextEnabled, // the currently displayed editor state (note: this is always what is modified on input) editorState: this.createEditorState( isRichtextEnabled, MessageComposerStore.getEditorState(this.props.room.roomId), ), // the original editor state, before we started tabbing through completions originalEditorState: null, // the virtual state "above" the history stack, the message currently being composed that // we want to persist whilst browsing history currentlyComposedEditorState: null, // whether there were any completions someCompletions: null, }; this.client = MatrixClientPeg.get(); this.plainWithMdPills = new PlainWithPillsSerializer({ pillFormat: 'md' }); this.plainWithIdPills = new PlainWithPillsSerializer({ pillFormat: 'id' }); this.plainWithPlainPills = new PlainWithPillsSerializer({ pillFormat: 'plain' }); this.suppressAutoComplete = false; this.direction = ''; } /* * "Does the right thing" to create an Editor value, based on: * - whether we've got rich text mode enabled * - contentState was passed in */ createEditorState(richText: boolean, value: ?Value): Value { if (value instanceof Value) { return value; } else { // ...or create a new one. return Plain.deserialize('') } } componentDidMount() { this.dispatcherRef = dis.register(this.onAction); this.historyManager = new ComposerHistoryManager(this.props.room.roomId); } componentWillUnmount() { dis.unregister(this.dispatcherRef); } componentWillUpdate(nextProps, nextState) { // this is dirty, but moving all this state to MessageComposer is dirtier if (this.props.onInputStateChanged && nextState !== this.state) { const state = this.getSelectionInfo(nextState.editorState); state.isRichtextEnabled = nextState.isRichtextEnabled; this.props.onInputStateChanged(state); } } onAction = (payload) => { const editor = this.refs.editor; let editorState = this.state.editorState; switch (payload.action) { case 'reply_to_event': case 'focus_composer': editor.focus(); break; case 'insert_mention': { // Pretend that we've autocompleted this user because keeping two code // paths for inserting a user pill is not fun const selection = this.getSelectionRange(this.state.editorState); const member = this.props.room.getMember(payload.user_id); const completion = member ? member.rawDisplayName.replace(' (IRC)', '') : payload.user_id; this.setDisplayedCompletion({ completion, completionId: payload.user_id, selection, href: makeUserPermalink(payload.user_id), suffix: (selection.beginning && selection.start === 0) ? ': ' : ' ', }); } break; /* case 'quote': { // old quoting, whilst rich quoting is in labs /// XXX: Not doing rich-text quoting from formatted-body because draft-js /// has regressed such that when links are quoted, errors are thrown. See /// https://github.com/vector-im/riot-web/issues/4756. const body = escape(payload.text); if (body) { let content = RichText.htmlToContentState(`
${body}`); if (!this.state.isRichtextEnabled) { content = ContentState.createFromText(RichText.stateToMarkdown(content)); } const blockMap = content.getBlockMap(); let startSelection = SelectionState.createEmpty(contentState.getFirstBlock().getKey()); contentState = Modifier.splitBlock(contentState, startSelection); startSelection = SelectionState.createEmpty(contentState.getFirstBlock().getKey()); contentState = Modifier.replaceWithFragment(contentState, startSelection, blockMap); startSelection = SelectionState.createEmpty(contentState.getFirstBlock().getKey()); if (this.state.isRichtextEnabled) { contentState = Modifier.setBlockType(contentState, startSelection, 'blockquote'); } let editorState = EditorState.push(this.state.editorState, contentState, 'insert-characters'); editorState = EditorState.moveSelectionToEnd(editorState); this.onEditorContentChanged(editorState); editor.focus(); } } break; */ } }; onTypingActivity() { this.isTyping = true; if (!this.userTypingTimer) { this.sendTyping(true); } this.startUserTypingTimer(); this.startServerTypingTimer(); } onFinishedTyping() { this.isTyping = false; this.sendTyping(false); this.stopUserTypingTimer(); this.stopServerTypingTimer(); } startUserTypingTimer() { this.stopUserTypingTimer(); const self = this; this.userTypingTimer = setTimeout(function() { self.isTyping = false; self.sendTyping(self.isTyping); self.userTypingTimer = null; }, TYPING_USER_TIMEOUT); } stopUserTypingTimer() { if (this.userTypingTimer) { clearTimeout(this.userTypingTimer); this.userTypingTimer = null; } } startServerTypingTimer() { if (!this.serverTypingTimer) { const self = this; this.serverTypingTimer = setTimeout(function() { if (self.isTyping) { self.sendTyping(self.isTyping); self.startServerTypingTimer(); } }, TYPING_SERVER_TIMEOUT / 2); } } stopServerTypingTimer() { if (this.serverTypingTimer) { clearTimeout(this.serverTypingTimer); this.serverTypingTimer = null; } } sendTyping(isTyping) { if (SettingsStore.getValue('dontSendTypingNotifications')) return; MatrixClientPeg.get().sendTyping( this.props.room.roomId, this.isTyping, TYPING_SERVER_TIMEOUT, ).done(); } refreshTyping() { if (this.typingTimeout) { clearTimeout(this.typingTimeout); this.typingTimeout = null; } } onChange = (change: Change, originalEditorState: value) => { let editorState = change.value; if (this.direction !== '') { const focusedNode = editorState.focusInline || editorState.focusText; if (focusedNode.isVoid) { if (editorState.isCollapsed) { change = change[`collapseToEndOf${ this.direction }Text`](); } else { const block = this.direction === 'Previous' ? editorState.previousText : editorState.nextText; if (block) { change = change.moveFocusToEndOf(block) } } editorState = change.value; } } if (!editorState.document.isEmpty) { this.onTypingActivity(); } else { this.onFinishedTyping(); } /* // XXX: what was this ever doing? if (!state.hasOwnProperty('originalEditorState')) { state.originalEditorState = null; } */ // emojioneify any emoji // XXX: is getTextsAsArray a private API? editorState.document.getTextsAsArray().forEach(node => { if (node.text !== '' && HtmlUtils.containsEmoji(node.text)) { let match; while ((match = EMOJI_REGEX.exec(node.text)) !== null) { const range = Range.create({ anchorKey: node.key, anchorOffset: match.index, focusKey: node.key, focusOffset: match.index + match[0].length, }); const inline = Inline.create({ type: 'emoji', data: { emojiUnicode: match[0] }, isVoid: true, }); change = change.insertInlineAtRange(range, inline); editorState = change.value; } } }); /* const currentBlock = editorState.getSelection().getStartKey(); const currentSelection = editorState.getSelection(); const currentStartOffset = editorState.getSelection().getStartOffset(); const block = editorState.getCurrentContent().getBlockForKey(currentBlock); const text = block.getText(); const entityBeforeCurrentOffset = block.getEntityAt(currentStartOffset - 1); const entityAtCurrentOffset = block.getEntityAt(currentStartOffset); // If the cursor is on the boundary between an entity and a non-entity and the // text before the cursor has whitespace at the end, set the entity state of the // character before the cursor (the whitespace) to null. This allows the user to // stop editing the link. if (entityBeforeCurrentOffset && !entityAtCurrentOffset && /\s$/.test(text.slice(0, currentStartOffset))) { editorState = RichUtils.toggleLink( editorState, currentSelection.merge({ anchorOffset: currentStartOffset - 1, focusOffset: currentStartOffset, }), null, ); // Reset selection editorState = EditorState.forceSelection(editorState, currentSelection); } */ const text = editorState.startText.text; const currentStartOffset = editorState.startOffset; // Automatic replacement of plaintext emoji to Unicode emoji if (SettingsStore.getValue('MessageComposerInput.autoReplaceEmoji')) { // The first matched group includes just the matched plaintext emoji const emojiMatch = REGEX_EMOJI_WHITESPACE.exec(text.slice(0, currentStartOffset)); if (emojiMatch) { // plaintext -> hex unicode const emojiUc = asciiList[emojiMatch[1]]; // hex unicode -> shortname -> actual unicode const unicodeEmoji = shortnameToUnicode(EMOJI_UNICODE_TO_SHORTNAME[emojiUc]); const range = Range.create({ anchorKey: editorState.selection.startKey, anchorOffset: currentStartOffset - emojiMatch[1].length - 1, focusKey: editorState.selection.startKey, focusOffset: currentStartOffset, }); change = change.insertTextAtRange(range, unicodeEmoji); editorState = change.value; } } // Record the editor state for this room so that it can be retrieved after // switching to another room and back dis.dispatch({ action: 'editor_state', room_id: this.props.room.roomId, editor_state: editorState, }); /* Since a modification was made, set originalEditorState to null, since newState is now our original */ this.setState({ editorState, originalEditorState: originalEditorState || null }); }; enableRichtext(enabled: boolean) { if (enabled === this.state.isRichtextEnabled) return; // FIXME: this conversion should be handled in the store, surely // i.e. "convert my current composer value into Rich or MD, as ComposerHistoryManager already does" let value = null; if (enabled) { // const md = new Markdown(this.state.editorState.getCurrentContent().getPlainText()); // contentState = RichText.htmlToContentState(md.toHTML()); value = Md.deserialize(Plain.serialize(this.state.editorState)); } else { // let markdown = RichText.stateToMarkdown(this.state.editorState.getCurrentContent()); // value = ContentState.createFromText(markdown); value = Plain.deserialize(Md.serialize(this.state.editorState)); } Analytics.setRichtextMode(enabled); this.setState({ editorState: this.createEditorState(enabled, value), isRichtextEnabled: enabled, }); SettingsStore.setValue("MessageComposerInput.isRichTextEnabled", null, SettingLevel.ACCOUNT, enabled); } onKeyDown = (ev: Event, change: Change, editor: Editor) => { this.suppressAutoComplete = false; // skip void nodes - see // https://github.com/ianstormtaylor/slate/issues/762#issuecomment-304855095 if (ev.keyCode === KeyCode.LEFT) { this.direction = 'Previous'; } else if (ev.keyCode === KeyCode.RIGHT) { this.direction = 'Next'; } else { this.direction = ''; } switch (ev.keyCode) { case KeyCode.ENTER: return this.handleReturn(ev); case KeyCode.UP: return this.onVerticalArrow(ev, true); case KeyCode.DOWN: return this.onVerticalArrow(ev, false); case KeyCode.TAB: return this.onTab(ev); case KeyCode.ESCAPE: return this.onEscape(ev); default: // don't intercept it return; } } handleKeyCommand = (command: string): boolean => { if (command === 'toggle-mode') { this.enableRichtext(!this.state.isRichtextEnabled); return true; } /* let newState: ?EditorState = null; // Draft handles rich text mode commands by default but we need to do it ourselves for Markdown. if (this.state.isRichtextEnabled) { // These are block types, not handled by RichUtils by default. const blockCommands = ['code-block', 'blockquote', 'unordered-list-item', 'ordered-list-item']; const currentBlockType = RichUtils.getCurrentBlockType(this.state.editorState); const shouldToggleBlockFormat = ( command === 'backspace' || command === 'split-block' ) && currentBlockType !== 'unstyled'; if (blockCommands.includes(command)) { newState = RichUtils.toggleBlockType(this.state.editorState, command); } else if (command === 'strike') { // this is the only inline style not handled by Draft by default newState = RichUtils.toggleInlineStyle(this.state.editorState, 'STRIKETHROUGH'); } else if (shouldToggleBlockFormat) { const currentStartOffset = this.state.editorState.getSelection().getStartOffset(); const currentEndOffset = this.state.editorState.getSelection().getEndOffset(); if (currentStartOffset === 0 && currentEndOffset === 0) { // Toggle current block type (setting it to 'unstyled') newState = RichUtils.toggleBlockType(this.state.editorState, currentBlockType); } } } else { const contentState = this.state.editorState.getCurrentContent(); const multipleLinesSelected = RichText.hasMultiLineSelection(this.state.editorState); const selectionState = this.state.editorState.getSelection(); const start = selectionState.getStartOffset(); const end = selectionState.getEndOffset(); // If multiple lines are selected or nothing is selected, insert a code block // instead of applying inline code formatting. This is an attempt to mimic what // happens in non-MD mode. const treatInlineCodeAsBlock = multipleLinesSelected || start === end; const textMdCodeBlock = (text) => `\`\`\`\n${text}\n\`\`\`\n`; const modifyFn = { 'bold': (text) => `**${text}**`, 'italic': (text) => `*${text}*`, 'underline': (text) => `${text}`, 'strike': (text) => `
{children}
} case 'pill': { const { data } = node; const url = data.get('url'); const completion = data.get('completion'); const shouldShowPillAvatar = !SettingsStore.getValue("Pill.shouldHidePillAvatar"); const Pill = sdk.getComponent('elements.Pill'); if (completion === '@room') { return