2020-04-01 07:47:51 +00:00
|
|
|
/* eslint-disable react/no-multi-comp */
|
|
|
|
// @flow
|
|
|
|
|
|
|
|
import React, { useRef } from 'react';
|
|
|
|
|
2021-04-09 12:30:25 +00:00
|
|
|
import { translate } from '../../../../base/i18n';
|
|
|
|
import { copyText } from '../../../../base/util';
|
2020-04-01 07:47:51 +00:00
|
|
|
|
|
|
|
import PasswordForm from './PasswordForm';
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not the current user can modify the current password.
|
|
|
|
*/
|
|
|
|
canEditPassword: boolean,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The JitsiConference for which to display a lock state and change the
|
|
|
|
* password.
|
|
|
|
*/
|
|
|
|
conference: Object,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The value for how the conference is locked (or undefined if not locked)
|
|
|
|
* as defined by room-lock constants.
|
|
|
|
*/
|
|
|
|
locked: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The current known password for the JitsiConference.
|
|
|
|
*/
|
|
|
|
password: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not to show the password in editing mode.
|
|
|
|
*/
|
|
|
|
passwordEditEnabled: boolean,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The number of digits to be used in the password.
|
|
|
|
*/
|
|
|
|
passwordNumberOfDigits: ?number,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Action that sets the conference password.
|
|
|
|
*/
|
|
|
|
setPassword: Function,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Method that sets whether the password editing is enabled or not.
|
|
|
|
*/
|
|
|
|
setPasswordEditEnabled: Function,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invoked to obtain translated strings.
|
|
|
|
*/
|
|
|
|
t: Function
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Component that handles the password manipulation from the invite dialog.
|
|
|
|
*
|
|
|
|
* @returns {React$Element<any>}
|
|
|
|
*/
|
|
|
|
function PasswordSection({
|
|
|
|
canEditPassword,
|
|
|
|
conference,
|
|
|
|
locked,
|
|
|
|
password,
|
|
|
|
passwordEditEnabled,
|
|
|
|
passwordNumberOfDigits,
|
|
|
|
setPassword,
|
|
|
|
setPasswordEditEnabled,
|
|
|
|
t }: Props) {
|
|
|
|
|
|
|
|
const formRef: Object = useRef(null);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback invoked to set a password on the current JitsiConference.
|
|
|
|
*
|
|
|
|
* @param {string} enteredPassword - The new password to be used to lock the
|
|
|
|
* current JitsiConference.
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onPasswordSubmit(enteredPassword) {
|
|
|
|
setPassword(conference, conference.lock, enteredPassword);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Toggles whether or not the password should currently be shown as being
|
|
|
|
* edited locally.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onTogglePasswordEditState() {
|
|
|
|
setPasswordEditEnabled(!passwordEditEnabled);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Method to remotely submit the password from outside of the password form.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onPasswordSave() {
|
|
|
|
if (formRef.current) {
|
2020-12-18 12:47:00 +00:00
|
|
|
const { value } = formRef.current.querySelector('form > input');
|
|
|
|
|
|
|
|
if (value) {
|
|
|
|
onPasswordSubmit(value);
|
|
|
|
}
|
2020-04-01 07:47:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback invoked to unlock the current JitsiConference.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onPasswordRemove() {
|
|
|
|
onPasswordSubmit('');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copies the password to the clipboard.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onPasswordCopy() {
|
|
|
|
copyText(password);
|
|
|
|
}
|
|
|
|
|
2021-06-10 12:48:44 +00:00
|
|
|
/**
|
|
|
|
* Toggles whether or not the password should currently be shown as being
|
|
|
|
* edited locally.
|
|
|
|
*
|
|
|
|
* @param {Object} e - The key event to handle.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onTogglePasswordEditStateKeyPressHandler(e) {
|
|
|
|
if (e.key === ' ' || e.key === 'Enter') {
|
|
|
|
e.preventDefault();
|
|
|
|
onTogglePasswordEditState();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Method to remotely submit the password from outside of the password form.
|
|
|
|
*
|
|
|
|
* @param {Object} e - The key event to handle.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onPasswordSaveKeyPressHandler(e) {
|
|
|
|
if (e.key === ' ' || e.key === 'Enter') {
|
|
|
|
e.preventDefault();
|
|
|
|
onPasswordSave();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback invoked to unlock the current JitsiConference.
|
|
|
|
*
|
|
|
|
* @param {Object} e - The key event to handle.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onPasswordRemoveKeyPressHandler(e) {
|
|
|
|
if (e.key === ' ' || e.key === 'Enter') {
|
|
|
|
e.preventDefault();
|
|
|
|
onPasswordRemove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copies the password to the clipboard.
|
|
|
|
*
|
|
|
|
* @param {Object} e - The key event to handle.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
function onPasswordCopyKeyPressHandler(e) {
|
|
|
|
if (e.key === ' ' || e.key === 'Enter') {
|
|
|
|
e.preventDefault();
|
|
|
|
onPasswordCopy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-01 07:47:51 +00:00
|
|
|
/**
|
|
|
|
* Method that renders the password action(s) based on the current
|
|
|
|
* locked-status of the conference.
|
|
|
|
*
|
|
|
|
* @returns {React$Element<any>}
|
|
|
|
*/
|
|
|
|
function renderPasswordActions() {
|
|
|
|
if (!canEditPassword) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (passwordEditEnabled) {
|
|
|
|
return (
|
|
|
|
<>
|
2021-06-10 12:48:44 +00:00
|
|
|
<a
|
|
|
|
aria-label = { t('dialog.Cancel') }
|
|
|
|
onClick = { onTogglePasswordEditState }
|
|
|
|
onKeyPress = { onTogglePasswordEditStateKeyPressHandler }
|
|
|
|
role = 'button'
|
|
|
|
tabIndex = { 0 }>{ t('dialog.Cancel') }</a>
|
|
|
|
<a
|
|
|
|
aria-label = { t('dialog.add') }
|
|
|
|
onClick = { onPasswordSave }
|
|
|
|
onKeyPress = { onPasswordSaveKeyPressHandler }
|
|
|
|
role = 'button'
|
|
|
|
tabIndex = { 0 }>{ t('dialog.add') }</a>
|
2020-04-01 07:47:51 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (locked) {
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<a
|
2021-06-10 12:48:44 +00:00
|
|
|
aria-label = { t('dialog.Remove') }
|
2020-04-01 07:47:51 +00:00
|
|
|
className = 'remove-password'
|
2021-06-10 12:48:44 +00:00
|
|
|
onClick = { onPasswordRemove }
|
|
|
|
onKeyPress = { onPasswordRemoveKeyPressHandler }
|
|
|
|
role = 'button'
|
|
|
|
tabIndex = { 0 }>{ t('dialog.Remove') }</a>
|
2020-11-05 17:43:48 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
// There are cases like lobby and grant moderator when password is not available
|
|
|
|
password ? <>
|
|
|
|
<a
|
2021-06-10 12:48:44 +00:00
|
|
|
aria-label = { t('dialog.copy') }
|
2020-11-05 17:43:48 +00:00
|
|
|
className = 'copy-password'
|
2021-06-10 12:48:44 +00:00
|
|
|
onClick = { onPasswordCopy }
|
|
|
|
onKeyPress = { onPasswordCopyKeyPressHandler }
|
|
|
|
role = 'button'
|
|
|
|
tabIndex = { 0 }>{ t('dialog.copy') }</a>
|
2020-11-05 17:43:48 +00:00
|
|
|
</> : null
|
|
|
|
}
|
2020-04-01 07:47:51 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<a
|
2021-06-10 12:48:44 +00:00
|
|
|
aria-label = { t('info.addPassword') }
|
2020-04-01 07:47:51 +00:00
|
|
|
className = 'add-password'
|
2021-06-10 12:48:44 +00:00
|
|
|
onClick = { onTogglePasswordEditState }
|
|
|
|
onKeyPress = { onTogglePasswordEditStateKeyPressHandler }
|
|
|
|
role = 'button'
|
|
|
|
tabIndex = { 0 }>{ t('info.addPassword') }</a>
|
2020-04-01 07:47:51 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
2020-05-20 08:25:31 +00:00
|
|
|
<div className = 'security-dialog password-section'>
|
2020-06-25 13:44:17 +00:00
|
|
|
<p className = 'description'>
|
2020-06-25 13:59:55 +00:00
|
|
|
{ t(canEditPassword ? 'security.about' : 'security.aboutReadOnly') }
|
2020-06-25 13:44:17 +00:00
|
|
|
</p>
|
2020-05-20 08:25:31 +00:00
|
|
|
<div className = 'security-dialog password'>
|
|
|
|
<div
|
|
|
|
className = 'info-dialog info-dialog-column info-dialog-password'
|
|
|
|
ref = { formRef }>
|
|
|
|
<PasswordForm
|
|
|
|
editEnabled = { passwordEditEnabled }
|
|
|
|
locked = { locked }
|
|
|
|
onSubmit = { onPasswordSubmit }
|
|
|
|
password = { password }
|
|
|
|
passwordNumberOfDigits = { passwordNumberOfDigits } />
|
|
|
|
</div>
|
|
|
|
<div className = 'security-dialog password-actions'>
|
|
|
|
{ renderPasswordActions() }
|
|
|
|
</div>
|
2020-04-01 07:47:51 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default translate(PasswordSection);
|