2019-08-06 18:03:44 +03:00
|
|
|
/*
|
|
|
|
Copyright 2019 New Vector Ltd
|
|
|
|
Copyright 2019 The Matrix.org Foundation C.I.C.
|
|
|
|
|
|
|
|
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 PropTypes from 'prop-types';
|
|
|
|
import dis from '../../../dispatcher';
|
|
|
|
import EditorModel from '../../../editor/model';
|
|
|
|
import {getCaretOffsetAndText} from '../../../editor/dom';
|
|
|
|
import {htmlSerializeIfNeeded, textSerialize} from '../../../editor/serialize';
|
|
|
|
import {PartCreator} from '../../../editor/parts';
|
|
|
|
import {MatrixClient} from 'matrix-js-sdk';
|
|
|
|
import BasicMessageComposer from "./BasicMessageComposer";
|
2019-08-07 16:14:50 +03:00
|
|
|
import ReplyPreview from "./ReplyPreview";
|
2019-08-20 12:41:13 +03:00
|
|
|
import RoomViewStore from '../../../stores/RoomViewStore';
|
|
|
|
import ReplyThread from "../elements/ReplyThread";
|
2019-08-20 13:36:19 +03:00
|
|
|
import {parseEvent} from '../../../editor/deserialize';
|
2019-08-20 12:41:13 +03:00
|
|
|
|
|
|
|
function addReplyToMessageContent(content, repliedToEvent, permalinkCreator) {
|
|
|
|
const replyContent = ReplyThread.makeReplyMixIn(repliedToEvent);
|
|
|
|
Object.assign(content, replyContent);
|
|
|
|
|
|
|
|
// Part of Replies fallback support - prepend the text we're sending
|
|
|
|
// with the text we're replying to
|
|
|
|
const nestedReply = ReplyThread.getNestedReplyText(repliedToEvent, permalinkCreator);
|
|
|
|
if (nestedReply) {
|
|
|
|
if (content.formatted_body) {
|
|
|
|
content.formatted_body = nestedReply.html + content.formatted_body;
|
|
|
|
}
|
|
|
|
content.body = nestedReply.body + content.body;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear reply_to_event as we put the message into the queue
|
|
|
|
// if the send fails, retry will handle resending.
|
|
|
|
dis.dispatch({
|
|
|
|
action: 'reply_to_event',
|
|
|
|
event: null,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function createMessageContent(model, permalinkCreator) {
|
|
|
|
const repliedToEvent = RoomViewStore.getQuotingEvent();
|
2019-08-06 18:03:44 +03:00
|
|
|
|
|
|
|
const body = textSerialize(model);
|
|
|
|
const content = {
|
|
|
|
msgtype: "m.text",
|
2019-08-20 12:41:13 +03:00
|
|
|
body: body,
|
2019-08-06 18:03:44 +03:00
|
|
|
};
|
2019-08-20 12:41:13 +03:00
|
|
|
const formattedBody = htmlSerializeIfNeeded(model, {forceHTML: !!repliedToEvent});
|
2019-08-06 18:03:44 +03:00
|
|
|
if (formattedBody) {
|
|
|
|
content.format = "org.matrix.custom.html";
|
|
|
|
content.formatted_body = formattedBody;
|
|
|
|
}
|
2019-08-20 12:41:13 +03:00
|
|
|
|
|
|
|
if (repliedToEvent) {
|
|
|
|
addReplyToMessageContent(content, repliedToEvent, permalinkCreator);
|
|
|
|
}
|
|
|
|
|
2019-08-06 18:03:44 +03:00
|
|
|
return content;
|
|
|
|
}
|
|
|
|
|
|
|
|
export default class SendMessageComposer extends React.Component {
|
|
|
|
static propTypes = {
|
2019-08-07 18:47:24 +03:00
|
|
|
room: PropTypes.object.isRequired,
|
|
|
|
placeholder: PropTypes.string,
|
|
|
|
permalinkCreator: PropTypes.object.isRequired,
|
2019-08-06 18:03:44 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
static contextTypes = {
|
|
|
|
matrixClient: PropTypes.instanceOf(MatrixClient).isRequired,
|
|
|
|
};
|
|
|
|
|
|
|
|
constructor(props, context) {
|
|
|
|
super(props, context);
|
|
|
|
this.model = null;
|
|
|
|
this._editorRef = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
_setEditorRef = ref => {
|
|
|
|
this._editorRef = ref;
|
|
|
|
};
|
|
|
|
|
|
|
|
_onKeyDown = (event) => {
|
|
|
|
if (event.metaKey || event.altKey || event.shiftKey) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (event.key === "Enter") {
|
|
|
|
this._sendMessage();
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-06 18:22:26 +03:00
|
|
|
_sendMessage() {
|
2019-08-06 18:03:44 +03:00
|
|
|
const {roomId} = this.props.room;
|
2019-08-20 12:41:13 +03:00
|
|
|
this.context.matrixClient.sendMessage(roomId, createMessageContent(this.model, this.props.permalinkCreator));
|
2019-08-06 18:03:44 +03:00
|
|
|
this.model.reset([]);
|
|
|
|
dis.dispatch({action: 'focus_composer'});
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillUnmount() {
|
|
|
|
const sel = document.getSelection();
|
|
|
|
const {caret} = getCaretOffsetAndText(this._editorRef, sel);
|
|
|
|
const parts = this.model.serializeParts();
|
|
|
|
this.props.editState.setEditorState(caret, parts);
|
2019-08-07 16:14:16 +03:00
|
|
|
dis.unregister(this.dispatcherRef);
|
2019-08-06 18:03:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
componentWillMount() {
|
|
|
|
const partCreator = new PartCreator(this.props.room, this.context.matrixClient);
|
|
|
|
this.model = new EditorModel([], partCreator);
|
2019-08-07 16:14:16 +03:00
|
|
|
this.dispatcherRef = dis.register(this.onAction);
|
2019-08-06 18:03:44 +03:00
|
|
|
}
|
|
|
|
|
2019-08-07 16:14:16 +03:00
|
|
|
onAction = (payload) => {
|
|
|
|
switch (payload.action) {
|
2019-08-07 16:14:50 +03:00
|
|
|
case 'reply_to_event':
|
2019-08-07 16:14:16 +03:00
|
|
|
case 'focus_composer':
|
|
|
|
this._editorRef.focus();
|
|
|
|
break;
|
2019-08-20 14:51:06 +03:00
|
|
|
case 'insert_mention':
|
|
|
|
this._insertMention(payload.user_id);
|
2019-08-07 18:44:49 +03:00
|
|
|
break;
|
2019-08-20 14:51:06 +03:00
|
|
|
case 'quote':
|
|
|
|
this._insertQuotedMessage(payload.event);
|
2019-08-20 13:36:19 +03:00
|
|
|
break;
|
2019-08-07 16:14:16 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-08-20 14:51:06 +03:00
|
|
|
_insertMention(userId) {
|
|
|
|
const member = this.props.room.getMember(userId);
|
|
|
|
const displayName = member ?
|
|
|
|
member.rawDisplayName : userId;
|
|
|
|
const userPillPart = this.model.partCreator.userPill(displayName, userId);
|
|
|
|
this.model.insertPartsAt([userPillPart], this._editorRef.getCaret());
|
|
|
|
// refocus on composer, as we just clicked "Mention"
|
|
|
|
this._editorRef.focus();
|
|
|
|
}
|
|
|
|
|
|
|
|
_insertQuotedMessage(event) {
|
|
|
|
const {partCreator} = this.model;
|
|
|
|
const quoteParts = parseEvent(event, partCreator, { isQuotedMessage: true });
|
|
|
|
// add two newlines
|
|
|
|
quoteParts.push(partCreator.newline());
|
|
|
|
quoteParts.push(partCreator.newline());
|
|
|
|
this.model.insertPartsAt(quoteParts, {offset: 0});
|
|
|
|
// refocus on composer, as we just clicked "Quote"
|
|
|
|
this._editorRef.focus();
|
|
|
|
}
|
|
|
|
|
2019-08-06 18:03:44 +03:00
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<div className="mx_SendMessageComposer" onClick={this.focusComposer} onKeyDown={this._onKeyDown}>
|
2019-08-07 16:14:50 +03:00
|
|
|
<div className="mx_SendMessageComposer_overlayWrapper">
|
|
|
|
<ReplyPreview permalinkCreator={this.props.permalinkCreator} />
|
|
|
|
</div>
|
2019-08-06 18:03:44 +03:00
|
|
|
<BasicMessageComposer
|
|
|
|
ref={this._setEditorRef}
|
|
|
|
model={this.model}
|
|
|
|
room={this.props.room}
|
2019-08-06 18:53:23 +03:00
|
|
|
label={this.props.placeholder}
|
2019-08-06 18:52:47 +03:00
|
|
|
placeholder={this.props.placeholder}
|
2019-08-06 18:03:44 +03:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|