bigbluebutton-Github/bigbluebutton-html5/imports/ui/components/layout/modal/component.jsx

204 lines
6.3 KiB
JavaScript

import React, { useState } from 'react';
import PropTypes from 'prop-types';
import { defineMessages, injectIntl } from 'react-intl';
import { LAYOUT_TYPE, CAMERADOCK_POSITION } from '/imports/ui/components/layout/enums';
import { withModalMounter } from '/imports/ui/components/common/modal/service';
import SettingsService from '/imports/ui/components/settings/service';
import deviceInfo from '/imports/utils/deviceInfo';
import Toggle from '/imports/ui/components/common/switch/component';
import Button from '/imports/ui/components/common/button/component';
import Styled from './styles';
const LayoutModalComponent = (props) => {
const {
intl,
closeModal,
isModerator,
isPresenter,
showToggleLabel,
application,
updateSettings,
} = props;
const [selectedLayout, setSelectedLayout] = useState(application.selectedLayout);
// eslint-disable-next-line react/prop-types
const [isKeepPushingLayout, setIsKeepPushingLayout] = useState(application.pushLayout);
const BASE_NAME = Meteor.settings.public.app.basename;
const LAYOUTS_PATH = `${BASE_NAME}/resources/images/layouts/`;
const isKeepPushingLayoutEnabled = SettingsService.isKeepPushingLayoutEnabled();
const intlMessages = defineMessages({
title: {
id: 'app.layout.modal.title',
description: 'Modal title',
},
confirm: {
id: 'app.layout.modal.confirm',
description: 'Modal confirm button',
},
cancel: {
id: 'app.layout.modal.cancel',
description: 'Modal cancel button',
},
layoutLabel: {
id: 'app.layout.modal.layoutLabel',
description: 'Layout label',
},
layoutToastLabel: {
id: 'app.layout.modal.layoutToastLabel',
description: 'Layout toast label',
},
keepPushingLayoutLabel: {
id: 'app.layout.modal.keepPushingLayoutLabel',
description: 'Keep push layout Label',
},
customLayout: {
id: 'app.layout.style.custom',
description: 'label for custom layout style',
},
smartLayout: {
id: 'app.layout.style.smart',
description: 'label for smart layout style',
},
presentationFocusLayout: {
id: 'app.layout.style.presentationFocus',
description: 'label for presentationFocus layout style',
},
videoFocusLayout: {
id: 'app.layout.style.videoFocus',
description: 'label for videoFocus layout style',
},
layoutSingular: {
id: 'app.layout.modal.layoutSingular',
description: 'label for singular layout',
},
layoutBtnDesc: {
id: 'app.layout.modal.layoutBtnDesc',
description: 'label for singular layout',
},
});
const handleSwitchLayout = (e) => {
setSelectedLayout(e);
};
const handleKeepPushingLayout = () => {
setIsKeepPushingLayout((newValue) => !newValue);
};
const handleCloseModal = () => {
const obj = {
application:
{ ...application, selectedLayout, pushLayout: isKeepPushingLayout },
};
updateSettings(obj, intlMessages.layoutToastLabel);
closeModal();
};
const renderPushLayoutsOptions = () => {
if (!isModerator && !isPresenter) {
return null;
}
if (isKeepPushingLayoutEnabled) {
return (
<Styled.PushContainer>
<Styled.LabelPushLayout>
{intl.formatMessage(intlMessages.keepPushingLayoutLabel)}
</Styled.LabelPushLayout>
<Toggle
id="TogglePush"
icons={false}
defaultChecked={isKeepPushingLayout}
onChange={handleKeepPushingLayout}
ariaLabel="push"
showToggleLabel={showToggleLabel}
/>
</Styled.PushContainer>
);
}
return null;
};
const renderLayoutButtons = () => (
<Styled.ButtonsContainer>
{Object.values(LAYOUT_TYPE)
.map((layout) => (
<Styled.ButtonLayoutContainer key={layout}>
<Styled.LabelLayoutNames aria-hidden>{intl.formatMessage(intlMessages[`${layout}Layout`])}</Styled.LabelLayoutNames>
<Styled.LayoutBtn
label=""
customIcon={(
<Styled.IconSvg
src={`${LAYOUTS_PATH}${layout}.svg`}
alt={`${layout} ${intl.formatMessage(intlMessages.layoutSingular)}`}
/>
)}
onClick={() => {
handleSwitchLayout(layout);
if (layout === LAYOUT_TYPE.CUSTOM_LAYOUT && application.selectedLayout !== layout) {
document.getElementById('layout')?.setAttribute('data-cam-position', CAMERADOCK_POSITION.CONTENT_TOP);
}
}}
active={(layout === selectedLayout).toString()}
aria-describedby="layout-btn-desc"
data-test={`${layout}Layout`}
/>
</Styled.ButtonLayoutContainer>
))}
</Styled.ButtonsContainer>
);
return (
<Styled.LayoutModal
contentLabel={intl.formatMessage(intlMessages.title)}
shouldShowCloseButton
shouldCloseOnOverlayClick
isPhone={deviceInfo.isPhone}
data-test="layoutChangeModal"
onRequestClose={closeModal}
title={intl.formatMessage(intlMessages.title)}
>
<Styled.Content>
<Styled.BodyContainer>
{renderLayoutButtons()}
{renderPushLayoutsOptions()}
</Styled.BodyContainer>
</Styled.Content>
<Styled.ButtonBottomContainer>
<Styled.BottomButton
label={intl.formatMessage(intlMessages.cancel)}
onClick={closeModal}
color="secondary"
/>
<Button
color="primary"
label={intl.formatMessage(intlMessages.confirm)}
onClick={handleCloseModal}
/>
</Styled.ButtonBottomContainer>
<div style={{ display: 'none' }} id="layout-btn-desc">{intl.formatMessage(intlMessages.layoutBtnDesc)}</div>
</Styled.LayoutModal>
);
};
const propTypes = {
intl: PropTypes.shape({
formatMessage: PropTypes.func.isRequired,
}).isRequired,
closeModal: PropTypes.func.isRequired,
isModerator: PropTypes.bool.isRequired,
isPresenter: PropTypes.bool.isRequired,
showToggleLabel: PropTypes.bool.isRequired,
application: PropTypes.shape({
selectedLayout: PropTypes.string.isRequired,
}).isRequired,
updateSettings: PropTypes.func.isRequired,
};
LayoutModalComponent.propTypes = propTypes;
export default injectIntl(withModalMounter(LayoutModalComponent));