mirror of
https://github.com/element-hq/element-web.git
synced 2024-12-16 16:31:38 +03:00
420 lines
17 KiB
JavaScript
420 lines
17 KiB
JavaScript
/*
|
|
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 PropTypes from 'prop-types';
|
|
import { _t } from '../../../languageHandler';
|
|
import CallHandler from '../../../CallHandler';
|
|
import MatrixClientPeg from '../../../MatrixClientPeg';
|
|
import Modal from '../../../Modal';
|
|
import sdk from '../../../index';
|
|
import dis from '../../../dispatcher';
|
|
import RoomViewStore from '../../../stores/RoomViewStore';
|
|
import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore";
|
|
import Stickerpicker from './Stickerpicker';
|
|
|
|
export default class MessageComposer extends React.Component {
|
|
constructor(props, context) {
|
|
super(props, context);
|
|
this.onCallClick = this.onCallClick.bind(this);
|
|
this.onHangupClick = this.onHangupClick.bind(this);
|
|
this.onUploadClick = this.onUploadClick.bind(this);
|
|
this.onUploadFileSelected = this.onUploadFileSelected.bind(this);
|
|
this.uploadFiles = this.uploadFiles.bind(this);
|
|
this.onVoiceCallClick = this.onVoiceCallClick.bind(this);
|
|
this.onInputContentChanged = this.onInputContentChanged.bind(this);
|
|
this._onAutocompleteConfirm = this._onAutocompleteConfirm.bind(this);
|
|
this.onToggleFormattingClicked = this.onToggleFormattingClicked.bind(this);
|
|
this.onToggleMarkdownClicked = this.onToggleMarkdownClicked.bind(this);
|
|
this.onInputStateChanged = this.onInputStateChanged.bind(this);
|
|
this.onEvent = this.onEvent.bind(this);
|
|
this._onRoomViewStoreUpdate = this._onRoomViewStoreUpdate.bind(this);
|
|
|
|
this.state = {
|
|
autocompleteQuery: '',
|
|
selection: null,
|
|
inputState: {
|
|
style: [],
|
|
blockType: null,
|
|
isRichtextEnabled: SettingsStore.getValue('MessageComposerInput.isRichTextEnabled'),
|
|
wordCount: 0,
|
|
},
|
|
showFormatting: SettingsStore.getValue('MessageComposer.showFormatting'),
|
|
isQuoting: Boolean(RoomViewStore.getQuotingEvent()),
|
|
};
|
|
}
|
|
|
|
componentDidMount() {
|
|
// N.B. using 'event' rather than 'RoomEvents' otherwise the crypto handler
|
|
// for 'event' fires *after* 'RoomEvent', and our room won't have yet been
|
|
// marked as encrypted.
|
|
// XXX: fragile as all hell - fixme somehow, perhaps with a dedicated Room.encryption event or something.
|
|
MatrixClientPeg.get().on("event", this.onEvent);
|
|
this._roomStoreToken = RoomViewStore.addListener(this._onRoomViewStoreUpdate);
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
if (MatrixClientPeg.get()) {
|
|
MatrixClientPeg.get().removeListener("event", this.onEvent);
|
|
}
|
|
if (this._roomStoreToken) {
|
|
this._roomStoreToken.remove();
|
|
}
|
|
}
|
|
|
|
onEvent(event) {
|
|
if (event.getType() !== 'm.room.encryption') return;
|
|
if (event.getRoomId() !== this.props.room.roomId) return;
|
|
this.forceUpdate();
|
|
}
|
|
|
|
_onRoomViewStoreUpdate() {
|
|
const isQuoting = Boolean(RoomViewStore.getQuotingEvent());
|
|
if (this.state.isQuoting === isQuoting) return;
|
|
this.setState({ isQuoting });
|
|
}
|
|
|
|
onUploadClick(ev) {
|
|
if (MatrixClientPeg.get().isGuest()) {
|
|
dis.dispatch({action: 'view_set_mxid'});
|
|
return;
|
|
}
|
|
|
|
this.refs.uploadInput.click();
|
|
}
|
|
|
|
onUploadFileSelected(files) {
|
|
this.uploadFiles(files.target.files);
|
|
}
|
|
|
|
uploadFiles(files) {
|
|
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
|
const TintableSvg = sdk.getComponent("elements.TintableSvg");
|
|
|
|
const fileList = [];
|
|
for (let i=0; i<files.length; i++) {
|
|
fileList.push(<li key={i}>
|
|
<TintableSvg key={i} src="img/files.svg" width="16" height="16" /> { files[i].name || _t('Attachment') }
|
|
</li>);
|
|
}
|
|
|
|
Modal.createTrackedDialog('Upload Files confirmation', '', QuestionDialog, {
|
|
title: _t('Upload Files'),
|
|
description: (
|
|
<div>
|
|
<p>{ _t('Are you sure you want to upload the following files?') }</p>
|
|
<ul style={{listStyle: 'none', textAlign: 'left'}}>
|
|
{ fileList }
|
|
</ul>
|
|
</div>
|
|
),
|
|
onFinished: (shouldUpload) => {
|
|
if (shouldUpload) {
|
|
// MessageComposer shouldn't have to rely on its parent passing in a callback to upload a file
|
|
if (files) {
|
|
for (let i=0; i<files.length; i++) {
|
|
this.props.uploadFile(files[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
this.refs.uploadInput.value = null;
|
|
},
|
|
});
|
|
}
|
|
|
|
onHangupClick() {
|
|
const call = CallHandler.getCallForRoom(this.props.room.roomId);
|
|
//var call = CallHandler.getAnyActiveCall();
|
|
if (!call) {
|
|
return;
|
|
}
|
|
dis.dispatch({
|
|
action: 'hangup',
|
|
// hangup the call for this room, which may not be the room in props
|
|
// (e.g. conferences which will hangup the 1:1 room instead)
|
|
room_id: call.roomId,
|
|
});
|
|
}
|
|
|
|
// _startCallApp(isAudioConf) {
|
|
// dis.dispatch({
|
|
// action: 'appsDrawer',
|
|
// show: true,
|
|
// });
|
|
|
|
// const appsStateEvents = this.props.room.currentState.getStateEvents('im.vector.modular.widgets', '');
|
|
// let appsStateEvent = {};
|
|
// if (appsStateEvents) {
|
|
// appsStateEvent = appsStateEvents.getContent();
|
|
// }
|
|
// if (!appsStateEvent.videoConf) {
|
|
// appsStateEvent.videoConf = {
|
|
// type: 'jitsi',
|
|
// // FIXME -- This should not be localhost
|
|
// url: 'http://localhost:8000/jitsi.html',
|
|
// data: {
|
|
// confId: this.props.room.roomId.replace(/[^A-Za-z0-9]/g, '_') + Date.now(),
|
|
// isAudioConf: isAudioConf,
|
|
// },
|
|
// };
|
|
// MatrixClientPeg.get().sendStateEvent(
|
|
// this.props.room.roomId,
|
|
// 'im.vector.modular.widgets',
|
|
// appsStateEvent,
|
|
// '',
|
|
// ).then(() => console.log('Sent state'), (e) => console.error(e));
|
|
// }
|
|
// }
|
|
|
|
onCallClick(ev) {
|
|
// NOTE -- Will be replaced by Jitsi code (currently commented)
|
|
dis.dispatch({
|
|
action: 'place_call',
|
|
type: ev.shiftKey ? "screensharing" : "video",
|
|
room_id: this.props.room.roomId,
|
|
});
|
|
// this._startCallApp(false);
|
|
}
|
|
|
|
onVoiceCallClick(ev) {
|
|
// NOTE -- Will be replaced by Jitsi code (currently commented)
|
|
dis.dispatch({
|
|
action: 'place_call',
|
|
type: "voice",
|
|
room_id: this.props.room.roomId,
|
|
});
|
|
// this._startCallApp(true);
|
|
}
|
|
|
|
onInputContentChanged(content: string, selection: {start: number, end: number}) {
|
|
this.setState({
|
|
autocompleteQuery: content,
|
|
selection,
|
|
});
|
|
}
|
|
|
|
onInputStateChanged(inputState) {
|
|
this.setState({inputState});
|
|
}
|
|
|
|
_onAutocompleteConfirm(range, completion) {
|
|
if (this.messageComposerInput) {
|
|
this.messageComposerInput.setDisplayedCompletion(range, completion);
|
|
}
|
|
}
|
|
|
|
onFormatButtonClicked(name: "bold" | "italic" | "strike" | "code" | "underline" | "quote" | "bullet" | "numbullet", event) {
|
|
event.preventDefault();
|
|
this.messageComposerInput.onFormatButtonClicked(name, event);
|
|
}
|
|
|
|
onToggleFormattingClicked() {
|
|
SettingsStore.setValue("MessageComposer.showFormatting", null, SettingLevel.DEVICE, !this.state.showFormatting);
|
|
this.setState({showFormatting: !this.state.showFormatting});
|
|
}
|
|
|
|
onToggleMarkdownClicked(e) {
|
|
e.preventDefault(); // don't steal focus from the editor!
|
|
this.messageComposerInput.enableRichtext(!this.state.inputState.isRichtextEnabled);
|
|
}
|
|
|
|
render() {
|
|
const me = this.props.room.getMember(MatrixClientPeg.get().credentials.userId);
|
|
const uploadInputStyle = {display: 'none'};
|
|
const MemberPresenceAvatar = sdk.getComponent('avatars.MemberPresenceAvatar');
|
|
const TintableSvg = sdk.getComponent("elements.TintableSvg");
|
|
const MessageComposerInput = sdk.getComponent("rooms.MessageComposerInput");
|
|
|
|
const controls = [];
|
|
|
|
controls.push(
|
|
<div key="controls_avatar" className="mx_MessageComposer_avatar">
|
|
<MemberPresenceAvatar member={me} width={24} height={24} />
|
|
</div>,
|
|
);
|
|
|
|
let e2eImg, e2eTitle, e2eClass;
|
|
const roomIsEncrypted = MatrixClientPeg.get().isRoomEncrypted(this.props.room.roomId);
|
|
if (roomIsEncrypted) {
|
|
// FIXME: show a /!\ if there are untrusted devices in the room...
|
|
e2eImg = 'img/e2e-verified.svg';
|
|
e2eTitle = _t('Encrypted room');
|
|
e2eClass = 'mx_MessageComposer_e2eIcon';
|
|
} else {
|
|
e2eImg = 'img/e2e-unencrypted.svg';
|
|
e2eTitle = _t('Unencrypted room');
|
|
e2eClass = 'mx_MessageComposer_e2eIcon mx_filterFlipColor';
|
|
}
|
|
|
|
controls.push(
|
|
<img key="e2eIcon" className={e2eClass} src={e2eImg} width="12" height="12"
|
|
alt={e2eTitle} title={e2eTitle}
|
|
/>,
|
|
);
|
|
|
|
let callButton;
|
|
let videoCallButton;
|
|
let hangupButton;
|
|
|
|
// Call buttons
|
|
if (this.props.callState && this.props.callState !== 'ended') {
|
|
hangupButton =
|
|
<div key="controls_hangup" className="mx_MessageComposer_hangup" onClick={this.onHangupClick}>
|
|
<img src="img/hangup.svg" alt={_t('Hangup')} title={_t('Hangup')} width="25" height="26" />
|
|
</div>;
|
|
} else {
|
|
callButton =
|
|
<div key="controls_call" className="mx_MessageComposer_voicecall" onClick={this.onVoiceCallClick} title={_t('Voice call')}>
|
|
<TintableSvg src="img/icon-call.svg" width="35" height="35" />
|
|
</div>;
|
|
videoCallButton =
|
|
<div key="controls_videocall" className="mx_MessageComposer_videocall" onClick={this.onCallClick} title={_t('Video call')}>
|
|
<TintableSvg src="img/icons-video.svg" width="35" height="35" />
|
|
</div>;
|
|
}
|
|
|
|
const canSendMessages = this.props.room.currentState.maySendMessage(
|
|
MatrixClientPeg.get().credentials.userId);
|
|
|
|
if (canSendMessages) {
|
|
// This also currently includes the call buttons. Really we should
|
|
// check separately for whether we can call, but this is slightly
|
|
// complex because of conference calls.
|
|
const uploadButton = (
|
|
<div key="controls_upload" className="mx_MessageComposer_upload"
|
|
onClick={this.onUploadClick} title={_t('Upload file')}>
|
|
<TintableSvg src="img/icons-upload.svg" width="35" height="35" />
|
|
<input ref="uploadInput" type="file"
|
|
style={uploadInputStyle}
|
|
multiple
|
|
onChange={this.onUploadFileSelected} />
|
|
</div>
|
|
);
|
|
|
|
const formattingButton = (
|
|
<img className="mx_MessageComposer_formatting"
|
|
title={_t("Show Text Formatting Toolbar")}
|
|
src="img/button-text-formatting.svg"
|
|
onClick={this.onToggleFormattingClicked}
|
|
style={{visibility: this.state.showFormatting ? 'hidden' : 'visible'}}
|
|
key="controls_formatting" />
|
|
);
|
|
|
|
let placeholderText;
|
|
if (this.state.isQuoting) {
|
|
if (roomIsEncrypted) {
|
|
placeholderText = _t('Send an encrypted reply…');
|
|
} else {
|
|
placeholderText = _t('Send a reply (unencrypted)…');
|
|
}
|
|
} else {
|
|
if (roomIsEncrypted) {
|
|
placeholderText = _t('Send an encrypted message…');
|
|
} else {
|
|
placeholderText = _t('Send a message (unencrypted)…');
|
|
}
|
|
}
|
|
|
|
let stickerpickerButton;
|
|
if (SettingsStore.isFeatureEnabled('feature_sticker_messages')) {
|
|
stickerpickerButton = <Stickerpicker key='stickerpicker_controls_button' room={this.props.room} />;
|
|
}
|
|
|
|
controls.push(
|
|
<MessageComposerInput
|
|
ref={(c) => this.messageComposerInput = c}
|
|
key="controls_input"
|
|
onResize={this.props.onResize}
|
|
room={this.props.room}
|
|
placeholder={placeholderText}
|
|
onFilesPasted={this.uploadFiles}
|
|
onContentChanged={this.onInputContentChanged}
|
|
onInputStateChanged={this.onInputStateChanged} />,
|
|
formattingButton,
|
|
stickerpickerButton,
|
|
uploadButton,
|
|
hangupButton,
|
|
callButton,
|
|
videoCallButton,
|
|
);
|
|
} else {
|
|
controls.push(
|
|
<div key="controls_error" className="mx_MessageComposer_noperm_error">
|
|
{ _t('You do not have permission to post to this room') }
|
|
</div>,
|
|
);
|
|
}
|
|
|
|
const {style, blockType} = this.state.inputState;
|
|
const formatButtons = ["bold", "italic", "strike", "underline", "code", "quote", "bullet", "numbullet"].map(
|
|
(name) => {
|
|
const active = style.includes(name) || blockType === name;
|
|
const suffix = active ? '-o-n' : '';
|
|
const onFormatButtonClicked = this.onFormatButtonClicked.bind(this, name);
|
|
const className = 'mx_MessageComposer_format_button mx_filterFlipColor';
|
|
return <img className={className}
|
|
title={_t(name)}
|
|
onMouseDown={onFormatButtonClicked}
|
|
key={name}
|
|
src={`img/button-text-${name}${suffix}.svg`}
|
|
height="17" />;
|
|
},
|
|
);
|
|
|
|
return (
|
|
<div className="mx_MessageComposer">
|
|
<div className="mx_MessageComposer_wrapper">
|
|
<div className="mx_MessageComposer_row">
|
|
{ controls }
|
|
</div>
|
|
</div>
|
|
<div className="mx_MessageComposer_formatbar_wrapper">
|
|
<div className="mx_MessageComposer_formatbar" style={this.state.showFormatting ? {} : {display: 'none'}}>
|
|
{ formatButtons }
|
|
<div style={{flex: 1}}></div>
|
|
<img title={this.state.inputState.isRichtextEnabled ? _t("Turn Markdown on") : _t("Turn Markdown off")}
|
|
onMouseDown={this.onToggleMarkdownClicked}
|
|
className="mx_MessageComposer_formatbar_markdown mx_filterFlipColor"
|
|
src={`img/button-md-${!this.state.inputState.isRichtextEnabled}.png`} />
|
|
<img title={_t("Hide Text Formatting Toolbar")}
|
|
onClick={this.onToggleFormattingClicked}
|
|
className="mx_MessageComposer_formatbar_cancel mx_filterFlipColor"
|
|
src="img/icon-text-cancel.svg" />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
MessageComposer.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,
|
|
|
|
// string representing the current voip call state
|
|
callState: PropTypes.string,
|
|
|
|
// callback when a file to upload is chosen
|
|
uploadFile: PropTypes.func.isRequired,
|
|
|
|
// string representing the current room app drawer state
|
|
showApps: PropTypes.bool,
|
|
};
|