2020-04-16 11:26:44 +00:00
|
|
|
/* @flow */
|
|
|
|
|
|
|
|
import React, { Component } from 'react';
|
|
|
|
import type { Dispatch } from 'redux';
|
|
|
|
|
|
|
|
import { createE2EEEvent, sendAnalytics } from '../../analytics';
|
2020-06-25 13:44:17 +00:00
|
|
|
import { translate } from '../../base/i18n';
|
2020-04-20 15:04:11 +00:00
|
|
|
import { getParticipants } from '../../base/participants';
|
2020-04-16 11:26:44 +00:00
|
|
|
import { connect } from '../../base/redux';
|
|
|
|
import { setE2EEKey } from '../actions';
|
|
|
|
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
|
2020-04-20 15:04:11 +00:00
|
|
|
/**
|
|
|
|
* Indicates whether all participants in the conference currently support E2EE.
|
|
|
|
*/
|
|
|
|
_everyoneSupportsE2EE: boolean,
|
|
|
|
|
2020-04-16 11:26:44 +00:00
|
|
|
/**
|
|
|
|
* The current E2EE key.
|
|
|
|
*/
|
|
|
|
_key: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The redux {@code dispatch} function.
|
|
|
|
*/
|
|
|
|
dispatch: Dispatch<any>,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invoked to obtain translated strings.
|
|
|
|
*/
|
|
|
|
t: Function
|
|
|
|
};
|
|
|
|
|
|
|
|
type State = {
|
|
|
|
|
2020-06-17 10:53:46 +00:00
|
|
|
/**
|
|
|
|
* True if the key is being edited.
|
|
|
|
*/
|
|
|
|
editing: boolean,
|
|
|
|
|
2020-06-25 13:44:17 +00:00
|
|
|
/**
|
|
|
|
* True if the section description should be expanded, false otherwise.
|
|
|
|
*/
|
|
|
|
expand: boolean,
|
|
|
|
|
2020-04-16 11:26:44 +00:00
|
|
|
/**
|
|
|
|
* The current E2EE key.
|
|
|
|
*/
|
|
|
|
key: string
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2020-06-17 10:53:46 +00:00
|
|
|
* Implements a React {@code Component} for displaying a security dialog section with a field
|
2020-04-16 11:26:44 +00:00
|
|
|
* for setting the E2EE key.
|
|
|
|
*
|
|
|
|
* @extends Component
|
|
|
|
*/
|
2020-06-17 10:53:46 +00:00
|
|
|
class E2EESection extends Component<Props, State> {
|
|
|
|
fieldRef: Object;
|
|
|
|
|
2020-04-16 11:26:44 +00:00
|
|
|
/**
|
|
|
|
* Initializes a new {@code E2EEDialog } instance.
|
|
|
|
*
|
|
|
|
* @param {Object} props - The read-only properties with which the new
|
|
|
|
* instance is to be initialized.
|
|
|
|
*/
|
|
|
|
constructor(props: Props) {
|
|
|
|
super(props);
|
|
|
|
|
2020-06-17 10:53:46 +00:00
|
|
|
this.fieldRef = React.createRef();
|
|
|
|
|
2020-04-16 11:26:44 +00:00
|
|
|
this.state = {
|
2020-06-17 10:53:46 +00:00
|
|
|
editing: false,
|
2020-06-25 13:44:17 +00:00
|
|
|
expand: false,
|
2020-04-16 11:26:44 +00:00
|
|
|
key: this.props._key
|
|
|
|
};
|
|
|
|
|
|
|
|
// Bind event handlers so they are only bound once for every instance.
|
2020-06-25 13:44:17 +00:00
|
|
|
this._onExpand = this._onExpand.bind(this);
|
2020-04-16 11:26:44 +00:00
|
|
|
this._onKeyChange = this._onKeyChange.bind(this);
|
2020-06-17 10:53:46 +00:00
|
|
|
this._onSet = this._onSet.bind(this);
|
|
|
|
this._onToggleSetKey = this._onToggleSetKey.bind(this);
|
2020-04-16 11:26:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Implements React's {@link Component#render()}.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
render() {
|
2020-04-20 15:04:11 +00:00
|
|
|
const { _everyoneSupportsE2EE, t } = this.props;
|
2020-06-25 13:44:17 +00:00
|
|
|
const { editing, expand } = this.state;
|
|
|
|
const description = t('dialog.e2eeDescription');
|
2020-04-16 11:26:44 +00:00
|
|
|
|
|
|
|
return (
|
2020-06-17 10:53:46 +00:00
|
|
|
<div id = 'e2ee-section'>
|
|
|
|
<p className = 'description'>
|
2020-06-25 13:44:17 +00:00
|
|
|
{ expand && description }
|
|
|
|
{ !expand && description.substring(0, 100) }
|
|
|
|
{ !expand && <span
|
|
|
|
className = 'read-more'
|
|
|
|
onClick = { this._onExpand }>
|
|
|
|
... { t('dialog.readMore') }
|
|
|
|
</span> }
|
2020-06-17 10:53:46 +00:00
|
|
|
</p>
|
2020-04-20 15:04:11 +00:00
|
|
|
{
|
|
|
|
!_everyoneSupportsE2EE
|
2020-06-17 10:53:46 +00:00
|
|
|
&& <span className = 'warning'>
|
|
|
|
{ t('dialog.e2eeWarning') }
|
|
|
|
</span>
|
2020-04-20 15:04:11 +00:00
|
|
|
}
|
2020-06-17 10:53:46 +00:00
|
|
|
<div className = 'key-field'>
|
|
|
|
<label>
|
|
|
|
{ t('dialog.e2eeLabel') }:
|
|
|
|
</label>
|
|
|
|
<input
|
|
|
|
disabled = { !editing }
|
|
|
|
name = 'e2eeKey'
|
|
|
|
onChange = { this._onKeyChange }
|
2020-06-24 12:12:23 +00:00
|
|
|
onKeyDown = { this._onKeyDown }
|
2020-06-17 10:53:46 +00:00
|
|
|
placeholder = { t('dialog.e2eeNoKey') }
|
|
|
|
ref = { this.fieldRef }
|
|
|
|
type = 'password'
|
|
|
|
value = { this.state.key } />
|
|
|
|
{ editing && <a onClick = { this._onSet }>
|
|
|
|
{ t('dialog.e2eeSet') }
|
|
|
|
</a> }
|
|
|
|
{ !editing && <a onClick = { this._onToggleSetKey }>
|
|
|
|
{ t('dialog.e2eeToggleSet') }
|
|
|
|
</a> }
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
2020-04-16 11:26:44 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 13:44:17 +00:00
|
|
|
_onExpand: () => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback to be invoked when the description is expanded.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onExpand() {
|
|
|
|
this.setState({
|
|
|
|
expand: true
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-04-16 11:26:44 +00:00
|
|
|
_onKeyChange: (Object) => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates the entered key.
|
|
|
|
*
|
|
|
|
* @param {Object} event - The DOM event triggered from the entered value having changed.
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onKeyChange(event) {
|
|
|
|
this.setState({ key: event.target.value.trim() });
|
|
|
|
}
|
|
|
|
|
2020-06-24 12:12:23 +00:00
|
|
|
_onKeyDown: (Object) => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handler for the keydown event on the form, preventing the closing of the dialog.
|
|
|
|
*
|
|
|
|
* @param {Object} event - The DOM event triggered by keydown events.
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onKeyDown(event) {
|
|
|
|
if (event.key === 'Enter') {
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-17 10:53:46 +00:00
|
|
|
_onSet: () => void;
|
2020-04-16 11:26:44 +00:00
|
|
|
|
|
|
|
/**
|
2020-06-17 10:53:46 +00:00
|
|
|
* Dispatches an action to set/unset the E2EE key.
|
2020-04-16 11:26:44 +00:00
|
|
|
*
|
|
|
|
* @private
|
2020-06-17 10:53:46 +00:00
|
|
|
* @returns {void}
|
2020-04-16 11:26:44 +00:00
|
|
|
*/
|
2020-06-17 10:53:46 +00:00
|
|
|
_onSet() {
|
2020-04-16 11:26:44 +00:00
|
|
|
const { key } = this.state;
|
|
|
|
|
|
|
|
sendAnalytics(createE2EEEvent(`key.${key ? 'set' : 'unset'}`));
|
|
|
|
this.props.dispatch(setE2EEKey(key));
|
|
|
|
|
2020-06-17 10:53:46 +00:00
|
|
|
this.setState({
|
|
|
|
editing: false
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
_onToggleSetKey: () => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the section into edit mode so then the user can set the key.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onToggleSetKey() {
|
|
|
|
this.setState({
|
|
|
|
editing: true
|
|
|
|
}, () => {
|
|
|
|
this.fieldRef.current.focus();
|
|
|
|
});
|
2020-04-16 11:26:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps (parts of) the Redux state to the associated props for this component.
|
|
|
|
*
|
|
|
|
* @param {Object} state - The Redux state.
|
|
|
|
* @private
|
|
|
|
* @returns {Props}
|
|
|
|
*/
|
|
|
|
function mapStateToProps(state) {
|
|
|
|
const { e2eeKey } = state['features/e2ee'];
|
2020-04-20 15:04:11 +00:00
|
|
|
const participants = getParticipants(state).filter(p => !p.local);
|
2020-04-16 11:26:44 +00:00
|
|
|
|
|
|
|
return {
|
2020-04-20 15:04:11 +00:00
|
|
|
_everyoneSupportsE2EE: participants.every(p => Boolean(p.e2eeSupported)),
|
2020-04-16 11:26:44 +00:00
|
|
|
_key: e2eeKey || ''
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-06-17 10:53:46 +00:00
|
|
|
export default translate(connect(mapStateToProps)(E2EESection));
|