2016-03-24 19:25:41 +08:00
|
|
|
/*
|
|
|
|
Copyright 2015, 2016 OpenMarket 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.
|
|
|
|
*/
|
2016-06-11 18:22:08 +08:00
|
|
|
import React from 'react';
|
2016-07-08 15:24:28 +08:00
|
|
|
import type SyntheticKeyboardEvent from 'react/lib/SyntheticKeyboardEvent';
|
|
|
|
import marked from 'marked';
|
2016-03-24 19:25:41 +08:00
|
|
|
marked.setOptions({
|
|
|
|
renderer: new marked.Renderer(),
|
|
|
|
gfm: true,
|
|
|
|
tables: true,
|
|
|
|
breaks: true,
|
|
|
|
pedantic: false,
|
|
|
|
sanitize: true,
|
|
|
|
smartLists: true,
|
2016-07-08 15:24:28 +08:00
|
|
|
smartypants: false,
|
2016-03-24 19:25:41 +08:00
|
|
|
});
|
|
|
|
|
2016-06-11 18:22:08 +08:00
|
|
|
import {Editor, EditorState, RichUtils, CompositeDecorator,
|
|
|
|
convertFromRaw, convertToRaw, Modifier, EditorChangeType,
|
2016-09-05 20:08:53 +08:00
|
|
|
getDefaultKeyBinding, KeyBindingUtil, ContentState, SelectionState} from 'draft-js';
|
2016-06-11 18:22:08 +08:00
|
|
|
|
|
|
|
import {stateToMarkdown} from 'draft-js-export-markdown';
|
2016-09-04 23:33:40 +08:00
|
|
|
import classNames from 'classnames';
|
2016-09-05 20:08:53 +08:00
|
|
|
import escape from 'lodash/escape';
|
2016-05-27 12:45:55 +08:00
|
|
|
|
2016-07-08 15:24:28 +08:00
|
|
|
import MatrixClientPeg from '../../../MatrixClientPeg';
|
|
|
|
import type {MatrixClient} from 'matrix-js-sdk/lib/matrix';
|
|
|
|
import SlashCommands from '../../../SlashCommands';
|
|
|
|
import Modal from '../../../Modal';
|
|
|
|
import sdk from '../../../index';
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-07-08 15:24:28 +08:00
|
|
|
import dis from '../../../dispatcher';
|
|
|
|
import KeyCode from '../../../KeyCode';
|
2016-09-05 20:08:53 +08:00
|
|
|
import UserSettingsStore from '../../../UserSettingsStore';
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-06-12 00:54:09 +08:00
|
|
|
import * as RichText from '../../../RichText';
|
2016-05-27 12:45:55 +08:00
|
|
|
|
2016-06-11 18:22:08 +08:00
|
|
|
const TYPING_USER_TIMEOUT = 10000, TYPING_SERVER_TIMEOUT = 30000;
|
|
|
|
|
2016-06-14 21:44:09 +08:00
|
|
|
const KEY_M = 77;
|
|
|
|
|
2016-06-12 02:41:27 +08:00
|
|
|
// FIXME Breaks markdown with multiple paragraphs, since it only strips first and last <p>
|
2016-07-08 15:24:28 +08:00
|
|
|
function mdownToHtml(mdown: string): string {
|
|
|
|
let html = marked(mdown) || "";
|
2016-06-11 18:22:08 +08:00
|
|
|
html = html.trim();
|
|
|
|
// strip start and end <p> tags else you get 'orrible spacing
|
|
|
|
if (html.indexOf("<p>") === 0) {
|
|
|
|
html = html.substring("<p>".length);
|
|
|
|
}
|
|
|
|
if (html.lastIndexOf("</p>") === (html.length - "</p>".length)) {
|
|
|
|
html = html.substring(0, html.length - "</p>".length);
|
|
|
|
}
|
|
|
|
return html;
|
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The textInput part of the MessageComposer
|
|
|
|
*/
|
2016-05-28 14:28:22 +08:00
|
|
|
export default class MessageComposerInput extends React.Component {
|
2016-07-08 15:24:28 +08:00
|
|
|
static getKeyBinding(e: SyntheticKeyboardEvent): string {
|
|
|
|
// C-m => Toggles between rich text and markdown modes
|
|
|
|
if (e.keyCode === KEY_M && KeyBindingUtil.isCtrlKeyCommand(e)) {
|
|
|
|
return 'toggle-mode';
|
|
|
|
}
|
|
|
|
|
|
|
|
return getDefaultKeyBinding(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
client: MatrixClient;
|
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
constructor(props, context) {
|
|
|
|
super(props, context);
|
|
|
|
this.onAction = this.onAction.bind(this);
|
2016-09-08 01:22:14 +08:00
|
|
|
this.focus = this.focus.bind(this);
|
2016-06-12 02:41:27 +08:00
|
|
|
this.handleReturn = this.handleReturn.bind(this);
|
|
|
|
this.handleKeyCommand = this.handleKeyCommand.bind(this);
|
2016-06-20 16:22:55 +08:00
|
|
|
this.setEditorState = this.setEditorState.bind(this);
|
2016-06-21 21:03:39 +08:00
|
|
|
this.onUpArrow = this.onUpArrow.bind(this);
|
|
|
|
this.onDownArrow = this.onDownArrow.bind(this);
|
|
|
|
this.onTab = this.onTab.bind(this);
|
2016-07-04 00:45:13 +08:00
|
|
|
this.onConfirmAutocompletion = this.onConfirmAutocompletion.bind(this);
|
2016-09-08 01:22:14 +08:00
|
|
|
this.onMarkdownToggleClicked = this.onMarkdownToggleClicked.bind(this);
|
2016-05-27 12:45:55 +08:00
|
|
|
|
2016-09-08 01:22:14 +08:00
|
|
|
const isRichtextEnabled = UserSettingsStore.getSyncedSetting('MessageComposerInput.isRichTextEnabled', true);
|
2016-06-15 02:43:34 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
this.state = {
|
2016-09-05 20:08:53 +08:00
|
|
|
isRichtextEnabled,
|
2016-07-04 00:45:13 +08:00
|
|
|
editorState: null,
|
2016-05-27 12:45:55 +08:00
|
|
|
};
|
2016-06-11 18:22:08 +08:00
|
|
|
|
|
|
|
// bit of a hack, but we need to do this here since createEditorState needs isRichtextEnabled
|
|
|
|
this.state.editorState = this.createEditorState();
|
2016-06-12 02:41:27 +08:00
|
|
|
|
|
|
|
this.client = MatrixClientPeg.get();
|
2016-06-11 18:22:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "Does the right thing" to create an EditorState, based on:
|
|
|
|
* - whether we've got rich text mode enabled
|
|
|
|
* - contentState was passed in
|
|
|
|
*/
|
2016-06-12 05:51:18 +08:00
|
|
|
createEditorState(richText: boolean, contentState: ?ContentState): EditorState {
|
|
|
|
let decorators = richText ? RichText.getScopedRTDecorators(this.props) :
|
|
|
|
RichText.getScopedMDDecorators(this.props),
|
|
|
|
compositeDecorator = new CompositeDecorator(decorators);
|
|
|
|
|
2016-06-15 02:43:34 +08:00
|
|
|
let editorState = null;
|
2016-06-12 05:51:18 +08:00
|
|
|
if (contentState) {
|
2016-06-15 02:43:34 +08:00
|
|
|
editorState = EditorState.createWithContent(contentState, compositeDecorator);
|
2016-06-12 05:51:18 +08:00
|
|
|
} else {
|
2016-06-15 02:43:34 +08:00
|
|
|
editorState = EditorState.createEmpty(compositeDecorator);
|
2016-06-12 05:51:18 +08:00
|
|
|
}
|
2016-06-15 02:43:34 +08:00
|
|
|
|
|
|
|
return EditorState.moveFocusToEnd(editorState);
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
componentWillMount() {
|
2016-06-10 02:23:09 +08:00
|
|
|
const component = this;
|
2016-03-24 19:25:41 +08:00
|
|
|
this.sentHistory = {
|
|
|
|
// The list of typed messages. Index 0 is more recent
|
|
|
|
data: [],
|
|
|
|
// The position in data currently displayed
|
|
|
|
position: -1,
|
|
|
|
// The room the history is for.
|
|
|
|
roomId: null,
|
|
|
|
// The original text before they hit UP
|
|
|
|
originalText: null,
|
|
|
|
// The textarea element to set text to.
|
|
|
|
element: null,
|
|
|
|
|
2016-06-12 00:54:09 +08:00
|
|
|
init: function(element, roomId) {
|
2016-03-24 19:25:41 +08:00
|
|
|
this.roomId = roomId;
|
|
|
|
this.element = element;
|
|
|
|
this.position = -1;
|
|
|
|
var storedData = window.sessionStorage.getItem(
|
2016-06-16 17:11:12 +08:00
|
|
|
"mx_messagecomposer_history_" + roomId
|
2016-03-24 19:25:41 +08:00
|
|
|
);
|
|
|
|
if (storedData) {
|
|
|
|
this.data = JSON.parse(storedData);
|
|
|
|
}
|
|
|
|
if (this.roomId) {
|
|
|
|
this.setLastTextEntry();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-06-12 00:54:09 +08:00
|
|
|
push: function(text) {
|
2016-03-24 19:25:41 +08:00
|
|
|
// store a message in the sent history
|
|
|
|
this.data.unshift(text);
|
|
|
|
window.sessionStorage.setItem(
|
2016-06-16 17:11:12 +08:00
|
|
|
"mx_messagecomposer_history_" + this.roomId,
|
2016-03-24 19:25:41 +08:00
|
|
|
JSON.stringify(this.data)
|
|
|
|
);
|
|
|
|
// reset history position
|
|
|
|
this.position = -1;
|
|
|
|
this.originalText = null;
|
|
|
|
},
|
|
|
|
|
|
|
|
// move in the history. Returns true if we managed to move.
|
2016-06-12 00:54:09 +08:00
|
|
|
next: function(offset) {
|
2016-03-24 19:25:41 +08:00
|
|
|
if (this.position === -1) {
|
|
|
|
// user is going into the history, save the current line.
|
|
|
|
this.originalText = this.element.value;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// user may have modified this line in the history; remember it.
|
|
|
|
this.data[this.position] = this.element.value;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (offset > 0 && this.position === (this.data.length - 1)) {
|
|
|
|
// we've run out of history
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// retrieve the next item (bounded).
|
|
|
|
var newPosition = this.position + offset;
|
|
|
|
newPosition = Math.max(-1, newPosition);
|
|
|
|
newPosition = Math.min(newPosition, this.data.length - 1);
|
|
|
|
this.position = newPosition;
|
|
|
|
|
|
|
|
if (this.position !== -1) {
|
|
|
|
// show the message
|
|
|
|
this.element.value = this.data[this.position];
|
|
|
|
}
|
|
|
|
else if (this.originalText !== undefined) {
|
|
|
|
// restore the original text the user was typing.
|
|
|
|
this.element.value = this.originalText;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
2016-06-12 00:54:09 +08:00
|
|
|
saveLastTextEntry: function() {
|
2016-03-24 19:25:41 +08:00
|
|
|
// save the currently entered text in order to restore it later.
|
|
|
|
// NB: This isn't 'originalText' because we want to restore
|
|
|
|
// sent history items too!
|
2016-06-11 18:22:08 +08:00
|
|
|
let contentJSON = JSON.stringify(convertToRaw(component.state.editorState.getCurrentContent()));
|
2016-06-16 17:11:12 +08:00
|
|
|
window.sessionStorage.setItem("mx_messagecomposer_input_" + this.roomId, contentJSON);
|
2016-03-24 19:25:41 +08:00
|
|
|
},
|
|
|
|
|
2016-06-12 00:54:09 +08:00
|
|
|
setLastTextEntry: function() {
|
2016-06-16 17:11:12 +08:00
|
|
|
let contentJSON = window.sessionStorage.getItem("mx_messagecomposer_input_" + this.roomId);
|
2016-06-11 18:22:08 +08:00
|
|
|
if (contentJSON) {
|
|
|
|
let content = convertFromRaw(JSON.parse(contentJSON));
|
2016-06-20 16:22:55 +08:00
|
|
|
component.setEditorState(component.createEditorState(component.state.isRichtextEnabled, content));
|
2016-05-28 14:28:22 +08:00
|
|
|
}
|
2016-07-04 00:45:13 +08:00
|
|
|
},
|
2016-03-24 19:25:41 +08:00
|
|
|
};
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
componentDidMount() {
|
2016-03-24 19:25:41 +08:00
|
|
|
this.dispatcherRef = dis.register(this.onAction);
|
|
|
|
this.sentHistory.init(
|
2016-05-28 14:28:22 +08:00
|
|
|
this.refs.editor,
|
2016-03-24 19:25:41 +08:00
|
|
|
this.props.room.roomId
|
|
|
|
);
|
2016-06-11 18:22:08 +08:00
|
|
|
// this is disabled for now, since https://github.com/matrix-org/matrix-react-sdk/pull/296 will land soon
|
|
|
|
// if (this.props.tabComplete) {
|
|
|
|
// this.props.tabComplete.setEditor(this.refs.editor);
|
|
|
|
// }
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
componentWillUnmount() {
|
2016-03-24 19:25:41 +08:00
|
|
|
dis.unregister(this.dispatcherRef);
|
|
|
|
this.sentHistory.saveLastTextEntry();
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-09-05 20:08:53 +08:00
|
|
|
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;
|
2016-09-08 01:22:14 +08:00
|
|
|
state.wordCount = nextState.editorState.getCurrentContent().getPlainText().split(' ').filter(w => !!w).length;
|
2016-09-05 20:08:53 +08:00
|
|
|
this.props.onInputStateChanged(state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
onAction(payload) {
|
2016-07-04 00:45:13 +08:00
|
|
|
let editor = this.refs.editor;
|
2016-09-05 20:08:53 +08:00
|
|
|
let contentState = this.state.editorState.getCurrentContent();
|
2016-06-11 18:22:08 +08:00
|
|
|
|
2016-03-24 19:25:41 +08:00
|
|
|
switch (payload.action) {
|
|
|
|
case 'focus_composer':
|
2016-05-27 12:45:55 +08:00
|
|
|
editor.focus();
|
2016-03-24 19:25:41 +08:00
|
|
|
break;
|
2016-06-11 18:22:08 +08:00
|
|
|
|
|
|
|
// TODO change this so we insert a complete user alias
|
|
|
|
|
2016-09-05 20:08:53 +08:00
|
|
|
case 'insert_displayname': {
|
|
|
|
contentState = Modifier.replaceText(
|
|
|
|
contentState,
|
|
|
|
this.state.editorState.getSelection(),
|
|
|
|
`${payload.displayname}: `
|
|
|
|
);
|
|
|
|
let editorState = EditorState.push(this.state.editorState, contentState, 'insert-characters');
|
|
|
|
editorState = EditorState.forceSelection(editorState, contentState.getSelectionAfter());
|
|
|
|
this.setEditorState(editorState);
|
|
|
|
editor.focus();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'quote': {
|
|
|
|
let {event: {content: {body, formatted_body}}} = payload.event || {};
|
|
|
|
formatted_body = formatted_body || escape(body);
|
|
|
|
if (formatted_body) {
|
|
|
|
let content = RichText.HTMLtoContentState(`<blockquote>${formatted_body}</blockquote>`);
|
|
|
|
if (!this.state.isRichtextEnabled) {
|
|
|
|
content = ContentState.createFromText(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');
|
|
|
|
this.setEditorState(editorState);
|
2016-06-11 18:22:08 +08:00
|
|
|
editor.focus();
|
2016-03-24 19:25:41 +08:00
|
|
|
}
|
2016-09-05 20:08:53 +08:00
|
|
|
}
|
|
|
|
break;
|
2016-03-24 19:25:41 +08:00
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
onKeyDown(ev) {
|
2016-06-11 18:22:08 +08:00
|
|
|
if (ev.keyCode === KeyCode.UP || ev.keyCode === KeyCode.DOWN) {
|
2016-03-24 19:25:41 +08:00
|
|
|
var oldSelectionStart = this.refs.textarea.selectionStart;
|
|
|
|
// Remember the keyCode because React will recycle the synthetic event
|
|
|
|
var keyCode = ev.keyCode;
|
|
|
|
// set a callback so we can see if the cursor position changes as
|
|
|
|
// a result of this event. If it doesn't, we cycle history.
|
|
|
|
setTimeout(() => {
|
|
|
|
if (this.refs.textarea.selectionStart == oldSelectionStart) {
|
|
|
|
this.sentHistory.next(keyCode === KeyCode.UP ? 1 : -1);
|
|
|
|
}
|
|
|
|
}, 0);
|
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
onTypingActivity() {
|
2016-03-24 19:25:41 +08:00
|
|
|
this.isTyping = true;
|
|
|
|
if (!this.userTypingTimer) {
|
|
|
|
this.sendTyping(true);
|
|
|
|
}
|
|
|
|
this.startUserTypingTimer();
|
|
|
|
this.startServerTypingTimer();
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
onFinishedTyping() {
|
2016-03-24 19:25:41 +08:00
|
|
|
this.isTyping = false;
|
|
|
|
this.sendTyping(false);
|
|
|
|
this.stopUserTypingTimer();
|
|
|
|
this.stopServerTypingTimer();
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
startUserTypingTimer() {
|
2016-03-24 19:25:41 +08:00
|
|
|
this.stopUserTypingTimer();
|
|
|
|
var self = this;
|
|
|
|
this.userTypingTimer = setTimeout(function() {
|
|
|
|
self.isTyping = false;
|
|
|
|
self.sendTyping(self.isTyping);
|
|
|
|
self.userTypingTimer = null;
|
|
|
|
}, TYPING_USER_TIMEOUT);
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
stopUserTypingTimer() {
|
2016-03-24 19:25:41 +08:00
|
|
|
if (this.userTypingTimer) {
|
|
|
|
clearTimeout(this.userTypingTimer);
|
|
|
|
this.userTypingTimer = null;
|
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
startServerTypingTimer() {
|
2016-03-24 19:25:41 +08:00
|
|
|
if (!this.serverTypingTimer) {
|
|
|
|
var self = this;
|
|
|
|
this.serverTypingTimer = setTimeout(function() {
|
|
|
|
if (self.isTyping) {
|
|
|
|
self.sendTyping(self.isTyping);
|
|
|
|
self.startServerTypingTimer();
|
|
|
|
}
|
|
|
|
}, TYPING_SERVER_TIMEOUT / 2);
|
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
stopServerTypingTimer() {
|
2016-03-24 19:25:41 +08:00
|
|
|
if (this.serverTypingTimer) {
|
|
|
|
clearTimeout(this.servrTypingTimer);
|
|
|
|
this.serverTypingTimer = null;
|
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
sendTyping(isTyping) {
|
2016-03-24 19:25:41 +08:00
|
|
|
MatrixClientPeg.get().sendTyping(
|
|
|
|
this.props.room.roomId,
|
|
|
|
this.isTyping, TYPING_SERVER_TIMEOUT
|
|
|
|
).done();
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
refreshTyping() {
|
2016-03-24 19:25:41 +08:00
|
|
|
if (this.typingTimeout) {
|
|
|
|
clearTimeout(this.typingTimeout);
|
|
|
|
this.typingTimeout = null;
|
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
|
|
|
|
2016-09-08 01:22:14 +08:00
|
|
|
focus(ev) {
|
2016-05-27 12:45:55 +08:00
|
|
|
this.refs.editor.focus();
|
|
|
|
}
|
|
|
|
|
2016-06-20 16:22:55 +08:00
|
|
|
setEditorState(editorState: EditorState) {
|
2016-07-08 15:24:28 +08:00
|
|
|
editorState = RichText.attachImmutableEntitiesToEmoji(editorState);
|
2016-05-27 12:45:55 +08:00
|
|
|
this.setState({editorState});
|
2016-06-10 02:23:09 +08:00
|
|
|
|
2016-07-08 15:24:28 +08:00
|
|
|
if (editorState.getCurrentContent().hasText()) {
|
|
|
|
this.onTypingActivity();
|
2016-06-10 02:23:09 +08:00
|
|
|
} else {
|
|
|
|
this.onFinishedTyping();
|
|
|
|
}
|
2016-06-17 07:28:09 +08:00
|
|
|
|
2016-07-08 15:24:28 +08:00
|
|
|
if (this.props.onContentChanged) {
|
2016-09-04 23:33:40 +08:00
|
|
|
const textContent = editorState.getCurrentContent().getPlainText();
|
|
|
|
const selection = RichText.selectionStateToTextOffsets(editorState.getSelection(),
|
|
|
|
editorState.getCurrentContent().getBlocksAsArray());
|
|
|
|
|
2016-09-05 20:08:53 +08:00
|
|
|
this.props.onContentChanged(textContent, selection);
|
2016-06-17 07:28:09 +08:00
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
}
|
|
|
|
|
2016-06-12 00:54:09 +08:00
|
|
|
enableRichtext(enabled: boolean) {
|
2016-06-12 05:51:18 +08:00
|
|
|
if (enabled) {
|
2016-06-12 00:54:09 +08:00
|
|
|
let html = mdownToHtml(this.state.editorState.getCurrentContent().getPlainText());
|
2016-06-20 16:22:55 +08:00
|
|
|
this.setEditorState(this.createEditorState(enabled, RichText.HTMLtoContentState(html)));
|
2016-06-12 00:54:09 +08:00
|
|
|
} else {
|
2016-06-12 05:51:18 +08:00
|
|
|
let markdown = stateToMarkdown(this.state.editorState.getCurrentContent()),
|
|
|
|
contentState = ContentState.createFromText(markdown);
|
2016-06-20 16:22:55 +08:00
|
|
|
this.setEditorState(this.createEditorState(enabled, contentState));
|
2016-06-12 00:54:09 +08:00
|
|
|
}
|
2016-06-12 05:13:57 +08:00
|
|
|
|
|
|
|
this.setState({
|
2016-09-08 01:22:14 +08:00
|
|
|
isRichtextEnabled: enabled,
|
2016-06-12 05:13:57 +08:00
|
|
|
});
|
2016-09-08 01:22:14 +08:00
|
|
|
|
|
|
|
UserSettingsStore.setSyncedSetting('MessageComposerInput.isRichTextEnabled', enabled);
|
2016-06-12 00:54:09 +08:00
|
|
|
}
|
2016-06-11 18:22:08 +08:00
|
|
|
|
2016-06-12 00:54:09 +08:00
|
|
|
handleKeyCommand(command: string): boolean {
|
2016-07-08 15:24:28 +08:00
|
|
|
if (command === 'toggle-mode') {
|
2016-06-12 00:54:09 +08:00
|
|
|
this.enableRichtext(!this.state.isRichtextEnabled);
|
2016-06-11 18:22:08 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-06-12 00:54:09 +08:00
|
|
|
let newState: ?EditorState = null;
|
|
|
|
|
|
|
|
// Draft handles rich text mode commands by default but we need to do it ourselves for Markdown.
|
2016-09-08 01:22:14 +08:00
|
|
|
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'];
|
|
|
|
|
|
|
|
if (blockCommands.includes(command)) {
|
|
|
|
this.setEditorState(RichUtils.toggleBlockType(this.state.editorState, command));
|
|
|
|
} else if (command === 'strike') {
|
|
|
|
// this is the only inline style not handled by Draft by default
|
|
|
|
this.setEditorState(RichUtils.toggleInlineStyle(this.state.editorState, 'STRIKETHROUGH'));
|
|
|
|
}
|
|
|
|
} else {
|
2016-06-12 00:54:09 +08:00
|
|
|
let contentState = this.state.editorState.getCurrentContent(),
|
|
|
|
selection = this.state.editorState.getSelection();
|
|
|
|
|
|
|
|
let modifyFn = {
|
|
|
|
bold: text => `**${text}**`,
|
|
|
|
italic: text => `*${text}*`,
|
|
|
|
underline: text => `_${text}_`, // there's actually no valid underline in Markdown, but *shrug*
|
2016-07-08 15:24:28 +08:00
|
|
|
code: text => `\`${text}\``,
|
2016-09-08 01:22:14 +08:00
|
|
|
blockquote: text => text.split('\n').map(line => `> ${line}\n`).join(''),
|
|
|
|
'unordered-list-item': text => text.split('\n').map(line => `- ${line}\n`).join(''),
|
|
|
|
'ordered-list-item': text => text.split('\n').map((line, i) => `${i+1}. ${line}\n`).join(''),
|
2016-06-12 00:54:09 +08:00
|
|
|
}[command];
|
|
|
|
|
2016-07-08 15:24:28 +08:00
|
|
|
if (modifyFn) {
|
2016-06-12 00:54:09 +08:00
|
|
|
newState = EditorState.push(
|
|
|
|
this.state.editorState,
|
|
|
|
RichText.modifyText(contentState, selection, modifyFn),
|
|
|
|
'insert-characters'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-08 15:24:28 +08:00
|
|
|
if (newState == null)
|
2016-06-12 00:54:09 +08:00
|
|
|
newState = RichUtils.handleKeyCommand(this.state.editorState, command);
|
|
|
|
|
|
|
|
if (newState != null) {
|
2016-06-20 16:22:55 +08:00
|
|
|
this.setEditorState(newState);
|
2016-05-27 12:45:55 +08:00
|
|
|
return true;
|
|
|
|
}
|
2016-09-04 23:33:40 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
handleReturn(ev) {
|
2016-07-04 00:45:13 +08:00
|
|
|
if (ev.shiftKey) {
|
2016-09-05 20:08:53 +08:00
|
|
|
this.setEditorState(RichUtils.insertSoftNewline(this.state.editorState));
|
|
|
|
return true;
|
2016-07-04 00:45:13 +08:00
|
|
|
}
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
const contentState = this.state.editorState.getCurrentContent();
|
2016-07-04 00:45:13 +08:00
|
|
|
if (!contentState.hasText()) {
|
2016-05-28 14:28:22 +08:00
|
|
|
return true;
|
2016-07-04 00:45:13 +08:00
|
|
|
}
|
|
|
|
|
2016-03-24 19:25:41 +08:00
|
|
|
|
2016-06-11 18:22:08 +08:00
|
|
|
let contentText = contentState.getPlainText(), contentHTML;
|
|
|
|
|
2016-06-12 02:41:27 +08:00
|
|
|
var cmd = SlashCommands.processInput(this.props.room.roomId, contentText);
|
|
|
|
if (cmd) {
|
|
|
|
if (!cmd.error) {
|
|
|
|
this.setState({
|
|
|
|
editorState: this.createEditorState()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (cmd.promise) {
|
|
|
|
cmd.promise.done(function() {
|
|
|
|
console.log("Command success.");
|
|
|
|
}, function(err) {
|
|
|
|
console.error("Command failure: %s", err);
|
|
|
|
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
|
|
Modal.createDialog(ErrorDialog, {
|
|
|
|
title: "Server error",
|
|
|
|
description: err.message
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else if (cmd.error) {
|
|
|
|
console.error(cmd.error);
|
|
|
|
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
|
|
Modal.createDialog(ErrorDialog, {
|
|
|
|
title: "Command error",
|
|
|
|
description: cmd.error
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-09-05 20:08:53 +08:00
|
|
|
if (this.state.isRichtextEnabled) {
|
2016-06-12 00:54:09 +08:00
|
|
|
contentHTML = RichText.contentStateToHTML(contentState);
|
2016-06-11 18:22:08 +08:00
|
|
|
} else {
|
|
|
|
contentHTML = mdownToHtml(contentText);
|
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
|
2016-06-12 02:41:27 +08:00
|
|
|
let sendFn = this.client.sendHtmlMessage;
|
|
|
|
|
|
|
|
if (contentText.startsWith('/me')) {
|
|
|
|
contentText = contentText.replace('/me', '');
|
|
|
|
// bit of a hack, but the alternative would be quite complicated
|
|
|
|
contentHTML = contentHTML.replace('/me', '');
|
|
|
|
sendFn = this.client.sendHtmlEmote;
|
|
|
|
}
|
|
|
|
|
2016-06-11 18:22:08 +08:00
|
|
|
this.sentHistory.push(contentHTML);
|
2016-06-12 02:41:27 +08:00
|
|
|
let sendMessagePromise = sendFn.call(this.client, this.props.room.roomId, contentText, contentHTML);
|
2016-06-11 18:22:08 +08:00
|
|
|
|
|
|
|
sendMessagePromise.done(() => {
|
|
|
|
dis.dispatch({
|
|
|
|
action: 'message_sent'
|
|
|
|
});
|
|
|
|
}, () => {
|
|
|
|
dis.dispatch({
|
|
|
|
action: 'message_send_failed'
|
|
|
|
});
|
|
|
|
});
|
2016-05-27 12:45:55 +08:00
|
|
|
|
|
|
|
this.setState({
|
2016-06-11 18:22:08 +08:00
|
|
|
editorState: this.createEditorState()
|
2016-05-27 12:45:55 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-06-21 21:03:39 +08:00
|
|
|
onUpArrow(e) {
|
2016-08-03 20:34:52 +08:00
|
|
|
if (this.props.onUpArrow && this.props.onUpArrow()) {
|
|
|
|
e.preventDefault();
|
2016-06-21 21:03:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onDownArrow(e) {
|
2016-08-03 20:34:52 +08:00
|
|
|
if (this.props.onDownArrow && this.props.onDownArrow()) {
|
|
|
|
e.preventDefault();
|
2016-06-21 21:03:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onTab(e) {
|
2016-08-03 20:21:40 +08:00
|
|
|
if (this.props.tryComplete) {
|
|
|
|
if (this.props.tryComplete()) {
|
2016-06-21 21:03:39 +08:00
|
|
|
e.preventDefault();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-04 00:45:13 +08:00
|
|
|
onConfirmAutocompletion(range, content: string) {
|
|
|
|
let contentState = Modifier.replaceText(
|
|
|
|
this.state.editorState.getCurrentContent(),
|
|
|
|
RichText.textOffsetsToSelectionState(range, this.state.editorState.getCurrentContent().getBlocksAsArray()),
|
|
|
|
content
|
|
|
|
);
|
|
|
|
|
2016-07-08 15:24:28 +08:00
|
|
|
let editorState = EditorState.push(this.state.editorState, contentState, 'insert-characters');
|
|
|
|
|
|
|
|
editorState = EditorState.forceSelection(editorState, contentState.getSelectionAfter());
|
|
|
|
|
|
|
|
this.setEditorState(editorState);
|
2016-07-04 00:45:13 +08:00
|
|
|
|
|
|
|
// for some reason, doing this right away does not update the editor :(
|
|
|
|
setTimeout(() => this.refs.editor.focus(), 50);
|
|
|
|
}
|
|
|
|
|
2016-09-08 01:22:14 +08:00
|
|
|
onFormatButtonClicked(name: "bold" | "italic" | "strike" | "code" | "underline" | "quote" | "bullet" | "numbullet", e) {
|
|
|
|
const command = {
|
|
|
|
code: 'code-block',
|
|
|
|
quote: 'blockquote',
|
|
|
|
bullet: 'unordered-list-item',
|
|
|
|
numbullet: 'ordered-list-item',
|
|
|
|
}[name] || name;
|
|
|
|
this.handleKeyCommand(command);
|
2016-09-04 23:33:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* returns inline style and block type of current SelectionState so MessageComposer can render formatting
|
|
|
|
buttons. */
|
|
|
|
getSelectionInfo(editorState: EditorState) {
|
|
|
|
const styleName = {
|
|
|
|
BOLD: 'bold',
|
|
|
|
ITALIC: 'italic',
|
|
|
|
STRIKETHROUGH: 'strike',
|
2016-09-08 01:22:14 +08:00
|
|
|
UNDERLINE: 'underline',
|
2016-09-04 23:33:40 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
const originalStyle = editorState.getCurrentInlineStyle().toArray();
|
|
|
|
const style = originalStyle
|
|
|
|
.map(style => styleName[style] || null)
|
|
|
|
.filter(styleName => !!styleName);
|
|
|
|
|
|
|
|
const blockName = {
|
2016-09-08 01:22:14 +08:00
|
|
|
'code-block': 'code',
|
2016-09-04 23:33:40 +08:00
|
|
|
blockquote: 'quote',
|
|
|
|
'unordered-list-item': 'bullet',
|
|
|
|
'ordered-list-item': 'numbullet',
|
|
|
|
};
|
|
|
|
const originalBlockType = editorState.getCurrentContent()
|
|
|
|
.getBlockForKey(editorState.getSelection().getStartKey())
|
|
|
|
.getType();
|
|
|
|
const blockType = blockName[originalBlockType] || null;
|
|
|
|
|
|
|
|
return {
|
|
|
|
style,
|
|
|
|
blockType,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-09-08 01:22:14 +08:00
|
|
|
onMarkdownToggleClicked() {
|
|
|
|
this.enableRichtext(!this.state.isRichtextEnabled);
|
|
|
|
}
|
|
|
|
|
2016-05-27 12:45:55 +08:00
|
|
|
render() {
|
2016-09-04 23:33:40 +08:00
|
|
|
const {editorState} = this.state;
|
2016-06-11 18:22:08 +08:00
|
|
|
|
2016-09-04 23:33:40 +08:00
|
|
|
// From https://github.com/facebook/draft-js/blob/master/examples/rich/rich.html#L92
|
|
|
|
// If the user changes block type before entering any text, we can
|
|
|
|
// either style the placeholder or hide it.
|
|
|
|
let hidePlaceholder = false;
|
|
|
|
const contentState = editorState.getCurrentContent();
|
|
|
|
if (!contentState.hasText()) {
|
|
|
|
if (contentState.getBlockMap().first().getType() !== 'unstyled') {
|
|
|
|
hidePlaceholder = true;
|
|
|
|
}
|
2016-06-11 18:22:08 +08:00
|
|
|
}
|
|
|
|
|
2016-09-04 23:33:40 +08:00
|
|
|
const className = classNames('mx_MessageComposer_input', {
|
|
|
|
mx_MessageComposer_input_empty: hidePlaceholder,
|
|
|
|
});
|
|
|
|
|
2016-03-24 19:25:41 +08:00
|
|
|
return (
|
2016-06-11 18:22:08 +08:00
|
|
|
<div className={className}
|
2016-09-08 01:22:14 +08:00
|
|
|
onClick={ this.focus }>
|
2016-09-05 20:08:53 +08:00
|
|
|
<img className="mx_MessageComposer_input_markdownIndicator"
|
2016-09-08 01:22:14 +08:00
|
|
|
onClick={this.onMarkdownToggleClicked}
|
2016-09-05 20:08:53 +08:00
|
|
|
title={`Markdown is ${this.state.isRichtextEnabled ? 'disabled' : 'enabled'}`}
|
|
|
|
src={`img/button-md-${!this.state.isRichtextEnabled}.png`} />
|
2016-05-27 12:45:55 +08:00
|
|
|
<Editor ref="editor"
|
2016-05-28 14:28:22 +08:00
|
|
|
placeholder="Type a message…"
|
2016-05-27 12:45:55 +08:00
|
|
|
editorState={this.state.editorState}
|
2016-06-20 16:22:55 +08:00
|
|
|
onChange={this.setEditorState}
|
2016-06-11 18:22:08 +08:00
|
|
|
keyBindingFn={MessageComposerInput.getKeyBinding}
|
2016-06-12 02:41:27 +08:00
|
|
|
handleKeyCommand={this.handleKeyCommand}
|
|
|
|
handleReturn={this.handleReturn}
|
|
|
|
stripPastedStyles={!this.state.isRichtextEnabled}
|
2016-06-21 21:03:39 +08:00
|
|
|
onTab={this.onTab}
|
|
|
|
onUpArrow={this.onUpArrow}
|
|
|
|
onDownArrow={this.onDownArrow}
|
2016-06-12 02:41:27 +08:00
|
|
|
spellCheck={true} />
|
2016-03-24 19:25:41 +08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
2016-05-27 12:45:55 +08:00
|
|
|
};
|
|
|
|
|
2016-05-28 14:28:22 +08:00
|
|
|
MessageComposerInput.propTypes = {
|
2016-05-27 12:45:55 +08:00
|
|
|
tabComplete: React.PropTypes.any,
|
|
|
|
|
|
|
|
// a callback which is called when the height of the composer is
|
|
|
|
// changed due to a change in content.
|
|
|
|
onResize: React.PropTypes.func,
|
|
|
|
|
|
|
|
// js-sdk Room object
|
2016-06-17 07:28:09 +08:00
|
|
|
room: React.PropTypes.object.isRequired,
|
|
|
|
|
|
|
|
// called with current plaintext content (as a string) whenever it changes
|
2016-06-21 21:03:39 +08:00
|
|
|
onContentChanged: React.PropTypes.func,
|
|
|
|
|
|
|
|
onUpArrow: React.PropTypes.func,
|
|
|
|
|
|
|
|
onDownArrow: React.PropTypes.func,
|
|
|
|
|
2016-08-03 20:21:40 +08:00
|
|
|
// attempts to confirm currently selected completion, returns whether actually confirmed
|
|
|
|
tryComplete: React.PropTypes.func,
|
2016-09-05 20:08:53 +08:00
|
|
|
|
|
|
|
onInputStateChanged: React.PropTypes.func.isRequired,
|
2016-05-27 12:45:55 +08:00
|
|
|
};
|