diff --git a/src/components/views/rooms/MessageComposerInput.js b/src/components/views/rooms/MessageComposerInput.js index 4b950c429d..f50acb8bef 100644 --- a/src/components/views/rooms/MessageComposerInput.js +++ b/src/components/views/rooms/MessageComposerInput.js @@ -20,7 +20,7 @@ import PropTypes from 'prop-types'; import type SyntheticKeyboardEvent from 'react/lib/SyntheticKeyboardEvent'; import { Editor } from 'slate-react'; -import { Value, Document, Event } from 'slate'; +import { Value, Document, Event, Inline, Range, Node } from 'slate'; import Html from 'slate-html-serializer'; import { Markdown as Md } from 'slate-md-serializer'; @@ -197,49 +197,6 @@ export default class MessageComposerInput extends React.Component { * - contentState was passed in */ createEditorState(richText: boolean, value: ?Value): Value { -/* - const decorators = richText ? RichText.getScopedRTDecorators(this.props) : - RichText.getScopedMDDecorators(this.props); - const shouldShowPillAvatar = !SettingsStore.getValue("Pill.shouldHidePillAvatar"); - decorators.push({ - strategy: this.findPillEntities.bind(this), - component: (entityProps) => { - const Pill = sdk.getComponent('elements.Pill'); - const type = entityProps.contentState.getEntity(entityProps.entityKey).getType(); - const {url} = entityProps.contentState.getEntity(entityProps.entityKey).getData(); - if (type === ENTITY_TYPES.AT_ROOM_PILL) { - return ; - } else if (Pill.isPillUrl(url)) { - return ; - } - - return ( - - { entityProps.children } - - ); - }, - }); - const compositeDecorator = new CompositeDecorator(decorators); - let editorState = null; - if (contentState) { - editorState = EditorState.createWithContent(contentState, compositeDecorator); - } else { - editorState = EditorState.createEmpty(compositeDecorator); - } - - return EditorState.moveFocusToEnd(editorState); -*/ if (value instanceof Value) { return value; } @@ -566,6 +523,10 @@ export default class MessageComposerInput extends React.Component { 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; @@ -938,15 +899,19 @@ export default class MessageComposerInput extends React.Component { let navigateHistory = false; if (up) { - let scrollCorrection = editorNode.scrollTop; - if (cursorRect.top - editorRect.top + scrollCorrection == 0) { + const scrollCorrection = editorNode.scrollTop; + const distanceFromTop = cursorRect.top - editorRect.top + scrollCorrection; + //console.log(`Cursor distance from editor top is ${distanceFromTop}`); + if (distanceFromTop == 0) { navigateHistory = true; } } else { - let scrollCorrection = + const scrollCorrection = editorNode.scrollHeight - editorNode.clientHeight - editorNode.scrollTop; - if (cursorRect.bottom - editorRect.bottom + scrollCorrection == 0) { + const distanceFromBottom = cursorRect.bottom - editorRect.bottom + scrollCorrection; + //console.log(`Cursor distance from editor bottom is ${distanceFromBottom}`); + if (distanceFromBottom == 0) { navigateHistory = true; } } @@ -1033,38 +998,50 @@ export default class MessageComposerInput extends React.Component { * If passed a non-null displayedCompletion, modifies state.originalEditorState to compute new state.editorState. */ setDisplayedCompletion = async (displayedCompletion: ?Completion): boolean => { -/* const activeEditorState = this.state.originalEditorState || this.state.editorState; if (displayedCompletion == null) { if (this.state.originalEditorState) { let editorState = this.state.originalEditorState; - // This is a workaround from https://github.com/facebook/draft-js/issues/458 - // Due to the way we swap editorStates, Draft does not rerender at times - editorState = EditorState.forceSelection(editorState, - editorState.getSelection()); this.setState({editorState}); } return false; } const {range = null, completion = '', href = null, suffix = ''} = displayedCompletion; - let contentState = activeEditorState.getCurrentContent(); - let entityKey; + let inline; if (href) { - contentState = contentState.createEntity('LINK', 'IMMUTABLE', { - url: href, - isCompletion: true, + inline = Inline.create({ + type: 'pill', + isVoid: true, + data: { url: href }, }); - entityKey = contentState.getLastCreatedEntityKey(); } else if (completion === '@room') { - contentState = contentState.createEntity(ENTITY_TYPES.AT_ROOM_PILL, 'IMMUTABLE', { - isCompletion: true, + inline = Inline.create({ + type: 'pill', + isVoid: true, + data: { type: Pill.TYPE_AT_ROOM_MENTION }, }); - entityKey = contentState.getLastCreatedEntityKey(); } + let editorState = activeEditorState; + + if (range) { + const change = editorState.change().moveOffsetsTo(range.start, range.end); + editorState = change.value; + } + + const change = editorState.change().insertInlineAtRange( + editorState.selection, inline + ); + editorState = change.value; + + this.setState({ editorState, originalEditorState: activeEditorState }, ()=>{ +// this.refs.editor.focus(); + }); + +/* let selection; if (range) { selection = RichText.textOffsetsToSelectionState( @@ -1085,16 +1062,51 @@ export default class MessageComposerInput extends React.Component { let editorState = EditorState.push(activeEditorState, contentState, 'insert-characters'); editorState = EditorState.forceSelection(editorState, contentState.getSelectionAfter()); this.setState({editorState, originalEditorState: activeEditorState}); - - // for some reason, doing this right away does not update the editor :( - // setTimeout(() => this.refs.editor.focus(), 50); -*/ +*/ return true; }; + renderNode = props => { + const { attributes, children, node, isSelected } = props; + + switch (node.type) { + case 'paragraph': { + return

{children}

+ } + case 'pill': { + const { data, text } = node; + const url = data.get('url'); + const type = data.get('type'); + + const shouldShowPillAvatar = !SettingsStore.getValue("Pill.shouldHidePillAvatar"); + const Pill = sdk.getComponent('elements.Pill'); + + if (type === Pill.TYPE_AT_ROOM_MENTION) { + return ; + } + else if (Pill.isPillUrl(url)) { + return ; + } + else { + return + { text } + ; + } + } + } + }; + onFormatButtonClicked = (name: "bold" | "italic" | "strike" | "code" | "underline" | "quote" | "bullet" | "numbullet", e) => { e.preventDefault(); // don't steal focus from the editor! -/* + const command = { code: 'code-block', quote: 'blockquote', @@ -1102,7 +1114,6 @@ export default class MessageComposerInput extends React.Component { numbullet: 'ordered-list-item', }[name] || name; this.handleKeyCommand(command); -*/ }; /* returns inline style and block type of current SelectionState so MessageComposer can render formatting @@ -1140,14 +1151,41 @@ export default class MessageComposerInput extends React.Component { */ } - getAutocompleteQuery(contentState: ContentState) { - return ''; + getAutocompleteQuery(editorState: Value) { + // FIXME: do we really want to regenerate this every time the control is rerendered? + + // We can just return the current block where the selection begins, which + // should be enough to capture any autocompletion input, given autocompletion + // providers only search for the first match which intersects with the current selection. + // This avoids us having to serialize the whole thing to plaintext and convert + // selection offsets in & out of the plaintext domain. + return editorState.document.getDescendant(editorState.selection.anchorKey).text; // Don't send markdown links to the autocompleter // return this.removeMDLinks(contentState, ['@', '#']); } + getSelectionRange(editorState: Value) { + // return a character range suitable for handing to an autocomplete provider. + // the range is relative to the anchor of the current editor selection. + // if the selection spans multiple blocks, then we collapse it for the calculation. + const range = { + start: editorState.selection.anchorOffset, + end: (editorState.selection.anchorKey == editorState.selection.focusKey) ? + editorState.selection.focusOffset : editorState.selection.anchorOffset, + } + if (range.start > range.end) { + const tmp = range.start; + range.start = range.end; + range.end = tmp; + } + return range; + } + /* + // delinkifies any matrix.to markdown links (i.e. pills) of form + // [#foo:matrix.org](https://matrix.to/#/#foo:matrix.org). + // the prefixes is an array of sigils that will be matched on. removeMDLinks(contentState: ContentState, prefixes: string[]) { const plaintext = contentState.getPlainText(); if (!plaintext) return ''; @@ -1189,24 +1227,10 @@ export default class MessageComposerInput extends React.Component { render() { const activeEditorState = this.state.originalEditorState || this.state.editorState; - let hidePlaceholder = false; - // FIXME: in case we need to implement manual placeholdering - const className = classNames('mx_MessageComposer_input', { - mx_MessageComposer_input_empty: hidePlaceholder, mx_MessageComposer_input_error: this.state.someCompletions === false, }); - const content = null; - const selection = { - start: 0, - end: 0, - }; - - // const content = activeEditorState.getCurrentContent(); - // const selection = RichText.selectionStateToTextOffsets(activeEditorState.getSelection(), - // activeEditorState.getCurrentContent().getBlocksAsArray()); - return (
@@ -1216,8 +1240,8 @@ export default class MessageComposerInput extends React.Component { room={this.props.room} onConfirm={this.setDisplayedCompletion} onSelectionChange={this.setDisplayedCompletion} - query={this.getAutocompleteQuery(content)} - selection={selection} + query={this.getAutocompleteQuery(activeEditorState)} + selection={this.getSelectionRange(activeEditorState)} />
@@ -1232,6 +1256,8 @@ export default class MessageComposerInput extends React.Component { value={this.state.editorState} onChange={this.onChange} onKeyDown={this.onKeyDown} + renderNode={this.renderNode} + spellCheck={true} /* blockStyleFn={MessageComposerInput.getBlockStyle} keyBindingFn={MessageComposerInput.getKeyBinding} @@ -1240,11 +1266,6 @@ export default class MessageComposerInput extends React.Component { handlePastedText={this.onTextPasted} handlePastedFiles={this.props.onFilesPasted} stripPastedStyles={!this.state.isRichtextEnabled} - onTab={this.onTab} - onUpArrow={this.onUpArrow} - onDownArrow={this.onDownArrow} - onEscape={this.onEscape} - spellCheck={true} */ />