2023-03-03 08:42:59 +00:00
|
|
|
import { Theme } from '@mui/material';
|
|
|
|
import { withStyles } from '@mui/styles';
|
2018-06-20 20:19:53 +00:00
|
|
|
import React from 'react';
|
2022-08-08 09:36:06 +00:00
|
|
|
import { WithTranslation } from 'react-i18next';
|
2018-06-20 20:19:53 +00:00
|
|
|
|
2022-10-19 08:42:54 +00:00
|
|
|
// @ts-expect-error
|
2018-06-20 20:19:53 +00:00
|
|
|
import UIEvents from '../../../../../service/UI/UIEvents';
|
2022-09-07 08:20:05 +00:00
|
|
|
import { createProfilePanelButtonEvent } from '../../../analytics/AnalyticsEvents';
|
|
|
|
import { sendAnalytics } from '../../../analytics/functions';
|
2023-03-03 08:42:59 +00:00
|
|
|
// eslint-disable-next-line lines-around-comment
|
2022-08-01 07:04:36 +00:00
|
|
|
// @ts-ignore
|
2023-03-03 08:42:59 +00:00
|
|
|
import Avatar from '../../../base/avatar/components/Avatar';
|
|
|
|
import AbstractDialogTab, {
|
|
|
|
IProps as AbstractDialogTabProps } from '../../../base/dialog/components/web/AbstractDialogTab';
|
2022-08-08 09:36:06 +00:00
|
|
|
import { translate } from '../../../base/i18n/functions';
|
2022-11-01 12:36:32 +00:00
|
|
|
import Button from '../../../base/ui/components/web/Button';
|
2023-03-03 08:42:59 +00:00
|
|
|
import Checkbox from '../../../base/ui/components/web/Checkbox';
|
2022-08-01 07:04:36 +00:00
|
|
|
import Input from '../../../base/ui/components/web/Input';
|
2023-03-03 08:42:59 +00:00
|
|
|
import Select from '../../../base/ui/components/web/Select';
|
2021-04-23 15:00:51 +00:00
|
|
|
import { openLogoutDialog } from '../../actions';
|
2018-06-20 20:19:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The type of the React {@code Component} props of {@link ProfileTab}.
|
|
|
|
*/
|
2023-03-03 08:42:59 +00:00
|
|
|
export interface IProps extends AbstractDialogTabProps, WithTranslation {
|
2018-06-20 20:19:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not server-side authentication is available.
|
|
|
|
*/
|
2022-09-08 09:52:36 +00:00
|
|
|
authEnabled: boolean;
|
2018-06-20 20:19:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The name of the currently (server-side) authenticated user.
|
|
|
|
*/
|
2022-09-08 09:52:36 +00:00
|
|
|
authLogin: string;
|
2018-06-20 20:19:53 +00:00
|
|
|
|
2023-03-03 08:42:59 +00:00
|
|
|
/**
|
|
|
|
* CSS classes object.
|
|
|
|
*/
|
|
|
|
classes: any;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The currently selected language to display in the language select
|
|
|
|
* dropdown.
|
|
|
|
*/
|
|
|
|
currentLanguage: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether to show hide self view setting.
|
|
|
|
*/
|
|
|
|
disableHideSelfView: boolean;
|
|
|
|
|
2018-06-20 20:19:53 +00:00
|
|
|
/**
|
|
|
|
* The display name to display for the local participant.
|
|
|
|
*/
|
2022-09-08 09:52:36 +00:00
|
|
|
displayName: string;
|
2018-06-20 20:19:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The email to display for the local participant.
|
|
|
|
*/
|
2022-09-08 09:52:36 +00:00
|
|
|
email: string;
|
2018-06-20 20:19:53 +00:00
|
|
|
|
2021-09-06 10:01:13 +00:00
|
|
|
/**
|
2022-08-01 07:04:36 +00:00
|
|
|
* Whether to hide the email input in the profile settings.
|
2021-09-06 10:01:13 +00:00
|
|
|
*/
|
2022-09-08 09:52:36 +00:00
|
|
|
hideEmailInSettings?: boolean;
|
2021-09-06 10:01:13 +00:00
|
|
|
|
2023-03-03 08:42:59 +00:00
|
|
|
/**
|
|
|
|
* Whether or not to hide self-view screen.
|
|
|
|
*/
|
|
|
|
hideSelfView: boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The id of the local participant.
|
|
|
|
*/
|
|
|
|
id: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* All available languages to display in the language select dropdown.
|
|
|
|
*/
|
|
|
|
languages: Array<string>;
|
|
|
|
|
2021-12-13 15:17:38 +00:00
|
|
|
/**
|
2022-08-01 07:04:36 +00:00
|
|
|
* If the display name is read only.
|
2021-12-13 15:17:38 +00:00
|
|
|
*/
|
2022-09-08 09:52:36 +00:00
|
|
|
readOnlyName: boolean;
|
2021-12-13 15:17:38 +00:00
|
|
|
|
2018-06-20 20:19:53 +00:00
|
|
|
/**
|
2023-03-03 08:42:59 +00:00
|
|
|
* Whether or not to display the language select dropdown.
|
2018-06-20 20:19:53 +00:00
|
|
|
*/
|
2023-03-03 08:42:59 +00:00
|
|
|
showLanguageSettings: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const styles = (theme: Theme) => {
|
|
|
|
return {
|
|
|
|
container: {
|
|
|
|
display: 'flex',
|
|
|
|
flexDirection: 'column' as const,
|
|
|
|
width: '100%',
|
|
|
|
padding: '0 2px'
|
|
|
|
},
|
|
|
|
|
|
|
|
avatarContainer: {
|
|
|
|
display: 'flex',
|
|
|
|
width: '100%',
|
|
|
|
justifyContent: 'center',
|
|
|
|
marginBottom: theme.spacing(4)
|
|
|
|
},
|
|
|
|
|
|
|
|
bottomMargin: {
|
|
|
|
marginBottom: theme.spacing(4)
|
|
|
|
}
|
|
|
|
};
|
2022-09-08 09:52:36 +00:00
|
|
|
};
|
2018-06-20 20:19:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* React {@code Component} for modifying the local user's profile.
|
|
|
|
*
|
2021-11-04 21:10:43 +00:00
|
|
|
* @augments Component
|
2018-06-20 20:19:53 +00:00
|
|
|
*/
|
2023-03-03 08:42:59 +00:00
|
|
|
class ProfileTab extends AbstractDialogTab<IProps, any> {
|
2019-06-26 14:08:23 +00:00
|
|
|
static defaultProps = {
|
|
|
|
displayName: '',
|
|
|
|
email: ''
|
|
|
|
};
|
|
|
|
|
2018-06-20 20:19:53 +00:00
|
|
|
/**
|
|
|
|
* Initializes a new {@code ConnectedSettingsDialog} instance.
|
|
|
|
*
|
2023-03-03 08:42:59 +00:00
|
|
|
* @param {IProps} props - The React {@code Component} props to initialize
|
2018-06-20 20:19:53 +00:00
|
|
|
* the new {@code ConnectedSettingsDialog} instance with.
|
|
|
|
*/
|
2023-03-03 08:42:59 +00:00
|
|
|
constructor(props: IProps) {
|
2018-06-20 20:19:53 +00:00
|
|
|
super(props);
|
|
|
|
|
|
|
|
// Bind event handlers so they are only bound once for every instance.
|
|
|
|
this._onAuthToggle = this._onAuthToggle.bind(this);
|
2021-06-10 12:48:44 +00:00
|
|
|
this._onDisplayNameChange = this._onDisplayNameChange.bind(this);
|
|
|
|
this._onEmailChange = this._onEmailChange.bind(this);
|
2023-03-03 08:42:59 +00:00
|
|
|
this._onHideSelfViewChanged = this._onHideSelfViewChanged.bind(this);
|
|
|
|
this._onLanguageItemSelect = this._onLanguageItemSelect.bind(this);
|
2021-06-10 12:48:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes display name of the user.
|
|
|
|
*
|
2022-08-01 07:04:36 +00:00
|
|
|
* @param {string} value - The key event to handle.
|
2021-06-10 12:48:44 +00:00
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
2022-08-01 07:04:36 +00:00
|
|
|
_onDisplayNameChange(value: string) {
|
2021-06-10 12:48:44 +00:00
|
|
|
super._onChange({ displayName: value });
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes email of the user.
|
|
|
|
*
|
2022-08-01 07:04:36 +00:00
|
|
|
* @param {string} value - The key event to handle.
|
2021-06-10 12:48:44 +00:00
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
2022-08-01 07:04:36 +00:00
|
|
|
_onEmailChange(value: string) {
|
2021-06-10 12:48:44 +00:00
|
|
|
super._onChange({ email: value });
|
2018-06-20 20:19:53 +00:00
|
|
|
}
|
|
|
|
|
2023-03-03 08:42:59 +00:00
|
|
|
/**
|
|
|
|
* Callback invoked to select if hide self view should be enabled.
|
|
|
|
*
|
|
|
|
* @param {Object} e - The key event to handle.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onHideSelfViewChanged({ target: { checked } }: React.ChangeEvent<HTMLInputElement>) {
|
|
|
|
super._onChange({ hideSelfView: checked });
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback invoked to select a language from select dropdown.
|
|
|
|
*
|
|
|
|
* @param {Object} e - The key event to handle.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onLanguageItemSelect(e: React.ChangeEvent<HTMLSelectElement>) {
|
|
|
|
const language = e.target.value;
|
|
|
|
|
|
|
|
super._onChange({ currentLanguage: language });
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the menu item for changing displayed language.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
_renderLanguageSelect() {
|
|
|
|
const {
|
|
|
|
classes,
|
|
|
|
currentLanguage,
|
|
|
|
languages,
|
|
|
|
t
|
|
|
|
} = this.props;
|
|
|
|
|
|
|
|
const languageItems
|
|
|
|
= languages.map((language: string) => {
|
|
|
|
return {
|
|
|
|
value: language,
|
|
|
|
label: t(`languages:${language}`)
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Select
|
|
|
|
className = { classes.bottomMargin }
|
|
|
|
label = { t('settings.language') }
|
|
|
|
onChange = { this._onLanguageItemSelect }
|
|
|
|
options = { languageItems }
|
|
|
|
value = { currentLanguage } />
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-06-20 20:19:53 +00:00
|
|
|
/**
|
|
|
|
* Implements React's {@link Component#render()}.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
render() {
|
|
|
|
const {
|
|
|
|
authEnabled,
|
2023-03-03 08:42:59 +00:00
|
|
|
classes,
|
|
|
|
disableHideSelfView,
|
2018-06-20 20:19:53 +00:00
|
|
|
displayName,
|
|
|
|
email,
|
2021-12-13 15:17:38 +00:00
|
|
|
hideEmailInSettings,
|
2023-03-03 08:42:59 +00:00
|
|
|
hideSelfView,
|
|
|
|
id,
|
2021-09-06 10:01:13 +00:00
|
|
|
readOnlyName,
|
2023-03-03 08:42:59 +00:00
|
|
|
showLanguageSettings,
|
|
|
|
t
|
2018-06-20 20:19:53 +00:00
|
|
|
} = this.props;
|
|
|
|
|
|
|
|
return (
|
2023-03-03 08:42:59 +00:00
|
|
|
<div className = { classes.container } >
|
|
|
|
<div className = { classes.avatarContainer }>
|
|
|
|
<Avatar
|
|
|
|
participantId = { id }
|
|
|
|
size = { 60 } />
|
2018-06-20 20:19:53 +00:00
|
|
|
</div>
|
2023-03-03 08:42:59 +00:00
|
|
|
<Input
|
|
|
|
className = { classes.bottomMargin }
|
|
|
|
disabled = { readOnlyName }
|
|
|
|
id = 'setDisplayName'
|
|
|
|
label = { t('profile.setDisplayNameLabel') }
|
|
|
|
name = 'name'
|
|
|
|
onChange = { this._onDisplayNameChange }
|
|
|
|
placeholder = { t('settings.name') }
|
|
|
|
type = 'text'
|
|
|
|
value = { displayName } />
|
|
|
|
{!hideEmailInSettings && <div className = 'profile-edit-field'>
|
|
|
|
<Input
|
|
|
|
className = { classes.bottomMargin }
|
|
|
|
id = 'setEmail'
|
|
|
|
label = { t('profile.setEmailLabel') }
|
|
|
|
name = 'email'
|
|
|
|
onChange = { this._onEmailChange }
|
|
|
|
placeholder = { t('profile.setEmailInput') }
|
|
|
|
type = 'text'
|
|
|
|
value = { email } />
|
|
|
|
</div>}
|
|
|
|
{!disableHideSelfView && (
|
|
|
|
<Checkbox
|
|
|
|
checked = { hideSelfView }
|
|
|
|
className = { classes.bottomMargin }
|
|
|
|
label = { t('videothumbnail.hideSelfView') }
|
|
|
|
name = 'hide-self-view'
|
|
|
|
onChange = { this._onHideSelfViewChanged } />
|
|
|
|
)}
|
|
|
|
{showLanguageSettings && this._renderLanguageSelect()}
|
2018-06-20 20:19:53 +00:00
|
|
|
{ authEnabled && this._renderAuth() }
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shows the dialog for logging in or out of a server and closes this
|
|
|
|
* dialog.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onAuthToggle() {
|
|
|
|
if (this.props.authLogin) {
|
|
|
|
sendAnalytics(createProfilePanelButtonEvent('logout.button'));
|
|
|
|
|
2021-04-23 15:00:51 +00:00
|
|
|
APP.store.dispatch(openLogoutDialog(
|
|
|
|
() => APP.UI.emitEvent(UIEvents.LOGOUT)
|
|
|
|
));
|
2018-06-20 20:19:53 +00:00
|
|
|
} else {
|
|
|
|
sendAnalytics(createProfilePanelButtonEvent('login.button'));
|
|
|
|
|
|
|
|
APP.UI.emitEvent(UIEvents.AUTH_CLICKED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a React Element for interacting with server-side authentication.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
_renderAuth() {
|
|
|
|
const {
|
|
|
|
authLogin,
|
|
|
|
t
|
|
|
|
} = this.props;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div>
|
2021-06-10 12:48:44 +00:00
|
|
|
<h2 className = 'mock-atlaskit-label'>
|
2018-06-20 20:19:53 +00:00
|
|
|
{ t('toolbar.authenticate') }
|
2021-06-10 12:48:44 +00:00
|
|
|
</h2>
|
2018-06-20 20:19:53 +00:00
|
|
|
{ authLogin
|
|
|
|
&& <div className = 'auth-name'>
|
|
|
|
{ t('settings.loggedIn', { name: authLogin }) }
|
|
|
|
</div> }
|
|
|
|
<Button
|
2022-08-04 10:39:22 +00:00
|
|
|
accessibilityLabel = { authLogin ? t('toolbar.logout') : t('toolbar.login') }
|
2018-06-20 20:19:53 +00:00
|
|
|
id = 'login_button'
|
2022-08-04 10:39:22 +00:00
|
|
|
label = { authLogin ? t('toolbar.logout') : t('toolbar.login') }
|
|
|
|
onClick = { this._onAuthToggle } />
|
2018-06-20 20:19:53 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-03 08:42:59 +00:00
|
|
|
export default withStyles(styles)(translate(ProfileTab));
|