@selfcommunity/react-ui
Version:
React UI Components to integrate a Community created with SelfCommunity Platform.
213 lines (208 loc) • 16 kB
JavaScript
import { __rest } from "tslib";
import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
import { Alert, AlertTitle, Box, Button, CircularProgress, Stack, Typography } from '@mui/material';
import { styled } from '@mui/material/styles';
import { useThemeProps } from '@mui/system';
import { Link, SCPreferences, SCPreferencesContext, SCRoutes, useSCFetchLiveStream, useSCPreferences, useSCRouting, useSCUser } from '@selfcommunity/react-core';
import { SCFeatureName, SCLiveStreamConnectionDetailsErrorType } from '@selfcommunity/types';
import classNames from 'classnames';
import { FormattedMessage, useIntl } from 'react-intl';
import { PREFIX } from './constants';
import { useCallback, useContext, useMemo, useState } from 'react';
import LiveStreamVideoConference from './LiveStreamVideoConference';
import '@livekit/components-styles';
import { LiveStreamService } from '@selfcommunity/api-services';
import { camelCase, Logger } from '@selfcommunity/utils';
import { SCOPE_SC_UI } from '../../constants/Errors';
import { PreJoin } from './LiveStreamVideoConference/PreJoin';
import { LiveStreamContext } from './LiveStreamVideoConference/LiveStreamProvider';
import { useSnackbar } from 'notistack';
import DialogContent from '@mui/material/DialogContent';
import BaseDialog from '../../shared/BaseDialog';
import CopyTextField from '../../shared/CopyTextArea';
const classes = {
root: `${PREFIX}-root`,
content: `${PREFIX}-content`,
title: `${PREFIX}-title`,
logo: `${PREFIX}-logo`,
description: `${PREFIX}-description`,
endConferenceWrap: `${PREFIX}-end-conference-wrap`,
btnBackHome: `${PREFIX}-btn-back-home`,
startPrejoinContent: `${PREFIX}-start-prejoin-content`,
preJoin: `${PREFIX}-prejoin`,
preJoinLoading: `${PREFIX}-prejoin-loading`,
prejoinLoader: `${PREFIX}-prejoin-loader`,
preJoinAlert: `${PREFIX}-prejoin-alert`,
shareLink: `${PREFIX}-share-link`,
endPrejoinContent: `${PREFIX}-end-prejoin-content`,
endPrejoinContentBox: `${PREFIX}-end-prejoin-content-box`,
conference: `${PREFIX}-conference`,
error: `${PREFIX}-error`
};
const Root = styled(Box, {
name: PREFIX,
slot: 'Root'
})(({ theme }) => ({}));
const DialogRoot = styled(BaseDialog, {
name: PREFIX,
slot: 'Root',
overridesResolver: (props, styles) => styles.dialogRoot
})(({ theme }) => ({}));
/**
*> API documentation for the Community-JS LiveStreamRoom component. Learn about the available props and the CSS API.
*
#### Import
```jsx
import {LiveStreamRoom} from '@selfcommunity/react-ui';
```
#### Component Name
The name `LiveStreamRoom` can be used when providing style overrides in the theme.
#### CSS
|Rule Name|Global class|Description|
|---|---|---|
|root|.SCLiveStreamRoom-root|Styles applied to the root element.|
|title|.SCLiveStreamRoom-title|Styles applied to the title element.|
|description|.SCLiveStreamRoom-description|Styles applied to the description element.|
|content|.SCLiveStreamRoom-content|Styles applied to the content.|
|prejoin|.SCLiveStreamRoom-prejoin|Styles applied to the prejoin.|
|conference|.SCLiveStreamRoom-conference|Styles applied to the conference.|
|error|.SCLiveStreamRoom-error|Styles applied to the error elements.|
* @param inProps
*/
export default function LiveStreamRoom(inProps) {
var _a, _b, _c, _d;
//PROPS
const props = useThemeProps({
props: inProps,
name: PREFIX
});
const { id = `live_stream_room_object_${props.liveStreamId ? props.liveStreamId : props.liveStream ? props.liveStream.id : ''}`, liveStreamId = null, liveStream = null, className, showPrejoinTitle = true, showPrejoinDescription = false, startPrejoinContent, endPrejoinContent, presetConnectionDetails, presetPreJoinChoices, LiveStreamVideoConferenceComponentProps = { options: { codec: 'vp8', hq: false } } } = props, rest = __rest(props, ["id", "liveStreamId", "liveStream", "className", "showPrejoinTitle", "showPrejoinDescription", "startPrejoinContent", "endPrejoinContent", "presetConnectionDetails", "presetPreJoinChoices", "LiveStreamVideoConferenceComponentProps"]);
// CONTEXT
const scUserContext = useSCUser();
const scRoutingContext = useSCRouting();
const { preferences, features } = useSCPreferences();
// STATE
const { scLiveStream } = useSCFetchLiveStream({ id: liveStreamId, liveStream });
const [preJoinChoices, setPreJoinChoices] = useState(presetPreJoinChoices);
const [loading, setLoading] = useState(false);
const [error, setError] = useState(null);
const preJoinDefaults = useMemo(() => {
var _a, _b, _c;
return {
username: ((_a = scUserContext.user) === null || _a === void 0 ? void 0 : _a.username) || '',
videoEnabled: ((_b = scLiveStream === null || scLiveStream === void 0 ? void 0 : scLiveStream.settings) === null || _b === void 0 ? void 0 : _b.disableVideo) === false,
audioEnabled: ((_c = scLiveStream === null || scLiveStream === void 0 ? void 0 : scLiveStream.settings) === null || _c === void 0 ? void 0 : _c.muteParticipants) === false
};
}, [scUserContext.user, scLiveStream]);
const [connectionDetails, setConnectionDetails] = useState(presetConnectionDetails);
const liveStreamEnabled = useMemo(() => preferences &&
features &&
features.includes(SCFeatureName.LIVE_STREAM) &&
SCPreferences.CONFIGURATIONS_LIVE_STREAM_ENABLED in preferences &&
preferences[SCPreferences.CONFIGURATIONS_LIVE_STREAM_ENABLED].value, [preferences, features]);
const scPreferencesContext = useContext(SCPreferencesContext);
const appUrl = useMemo(() => scPreferencesContext.preferences && scPreferencesContext.preferences[SCPreferences.CONFIGURATIONS_APP_URL].value, [scPreferencesContext.preferences]);
// INTL
const intl = useIntl();
// MESSAGES
const { enqueueSnackbar } = useSnackbar();
// HANDLERS
/**
* Handle PreJoin Submit
*/
const handlePreJoinSubmit = useCallback((values) => {
if (scLiveStream || !loading) {
setLoading(true);
setError(null);
toggleAttrDisabledPrejoinActions(true);
LiveStreamService.join(scLiveStream.id)
.then((data) => {
setPreJoinChoices(values);
setConnectionDetails(Object.assign(Object.assign({}, data), { participantName: scUserContext.user.username }));
toggleAttrDisabledPrejoinActions(false);
setLoading(false);
})
.catch((error) => {
Logger.error(SCOPE_SC_UI, error);
if (error.response &&
error.response.data &&
typeof error.response.data === 'object' &&
error.response.data.errors &&
error.response.data.errors.length) {
let _msg = (_jsx(FormattedMessage, { id: "ui.liveStreamRoom.connect.error.generic", defaultMessage: "'ui.liveStreamRoom.connect.error.generic", values: { link: (...chunks) => _jsx(Link, Object.assign({ to: '/' }, { children: chunks })) } }));
if (error.response.data.errors[0].code) {
const _error = `ui.liveStreamRoom.connect.error.${camelCase(error.response.data.errors[0].code)}`;
_msg = (_jsx(FormattedMessage, { id: _error, defaultMessage: _error, values: {
link: (...chunks) => (_jsx("a", Object.assign({ style: { color: '#FFF' }, href: error.response.data.errors[0].code === SCLiveStreamConnectionDetailsErrorType.PARTICIPATE_THE_EVENT_TO_JOIN_LIVE_STREAM &&
scLiveStream.event
? scRoutingContext.url(SCRoutes.EVENT_ROUTE_NAME, scLiveStream.event)
: '/' }, { children: chunks })))
} }));
if (error.response.data.errors[0].code === SCLiveStreamConnectionDetailsErrorType.WAITING_HOST_TO_START_LIVE_STREAM ||
error.response.data.errors[0].code === SCLiveStreamConnectionDetailsErrorType.PARTICIPANTS_LIMIT_REACHED) {
setError(_msg);
}
else {
setTimeout(() => toggleAttrDisabledPrejoinActions(false), 10000);
}
enqueueSnackbar(_msg, { variant: 'error', autoHideDuration: 5000 });
}
else {
enqueueSnackbar(_msg, { variant: 'error' });
setError(_msg);
}
}
setLoading(false);
});
}
}, [scUserContext.user, setPreJoinChoices, setConnectionDetails, scLiveStream, setError, loading]);
/**
* Handle disable controls button
*/
const toggleAttrDisabledPrejoinActions = useCallback((disabled) => {
const container = document.querySelector('.lk-prejoin');
if (container) {
const buttons = container.querySelectorAll('button.lk-button');
buttons.forEach((button) => {
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
button.disabled = disabled;
});
}
}, []);
/**
* Handle PreJoin Error
*/
const handlePreJoinError = useCallback((e) => {
console.error(e);
if (e.message !== 'NotAllowedError: Permission denied') {
enqueueSnackbar(intl.formatMessage({
id: 'ui.liveStreamRoom.connect.error.device.permission',
defaultMessage: 'ui.liveStreamRoom.connect.error.device.permission'
}), { variant: 'error', autoHideDuration: 5000 });
}
else if (e.message !== 'NotFoundError: Requested device not found') {
enqueueSnackbar(intl.formatMessage({
id: 'ui.liveStreamRoom.connect.error.device.notFound',
defaultMessage: 'ui.liveStreamRoom.connect.error.device.notFound'
}), { variant: 'error', autoHideDuration: 5000 });
}
}, []);
/**
* User must be authenticated
*/
if (!scLiveStream || !scUserContext.user || !liveStreamEnabled) {
return _jsx(CircularProgress, {});
}
/**
* Renders root object
*/
return (_jsx(Root, Object.assign({ id: id, className: classNames(classes.root, className) }, rest, { children: scLiveStream.closed_at_by_host ? (_jsx(DialogRoot, Object.assign({ open: true, maxWidth: 'md', fullWidth: true }, { children: _jsxs(DialogContent, Object.assign({ className: classes.endConferenceWrap }, { children: [_jsx(Link, Object.assign({ to: scRoutingContext.url(SCRoutes.HOME_ROUTE_NAME, {}), className: classes.logo }, { children: _jsx("img", { src: preferences[SCPreferences.LOGO_NAVBAR_LOGO].value, alt: "logo" }) })), _jsx(Typography, Object.assign({ variant: "h5" }, { children: _jsx(FormattedMessage, { id: "ui.liveStreamRoom.conference.closed", defaultMessage: "ui.liveStreamRoom.conference.closed" }) })), _jsx(Button, Object.assign({ variant: "contained", color: "secondary", component: Link, to: '/', className: classes.btnBackHome }, { children: _jsx(FormattedMessage, { id: "ui.liveStreamRoom.button.backHome", defaultMessage: "ui.liveStreamRoom.button.backHome" }) }))] })) }))) : (_jsx(Box, Object.assign({ className: classes.content, "data-lk-theme": "default" }, { children: connectionDetails === undefined || preJoinChoices === undefined ? (_jsxs(_Fragment, { children: [startPrejoinContent && _jsx(Box, Object.assign({ className: classes.startPrejoinContent }, { children: startPrejoinContent })), (scLiveStream === null || scLiveStream === void 0 ? void 0 : scLiveStream.title) && (_jsx(Typography, Object.assign({ component: 'div', variant: "h4", className: classes.title, alignContent: 'center' }, { children: scLiveStream === null || scLiveStream === void 0 ? void 0 : scLiveStream.title }))), _jsxs(Box, Object.assign({ className: classNames(classes.preJoin, { [classes.preJoinLoading]: loading || error }) }, { children: [_jsx(LiveStreamContext.Provider, Object.assign({ value: { liveStream: scLiveStream } }, { children: _jsx(PreJoin, { defaults: preJoinDefaults, onSubmit: handlePreJoinSubmit, onError: handlePreJoinError, joinLabel: intl.formatMessage({ id: 'ui.liveStreamRoom.preJoin.joinRoom', defaultMessage: 'ui.liveStreamRoom.preJoin.joinRoom' }), micLabel: intl.formatMessage({
id: 'ui.liveStreamRoom.preJoin.microphone',
defaultMessage: 'ui.liveStreamRoom.preJoin.microphone'
}), camLabel: intl.formatMessage({ id: 'ui.liveStreamRoom.preJoin.camera', defaultMessage: 'ui.liveStreamRoom.preJoin.camera' }), userLabel: intl.formatMessage({ id: 'ui.liveStreamRoom.preJoin.username', defaultMessage: 'ui.liveStreamRoom.preJoin.username' }) }) })), loading && (_jsxs(Box, Object.assign({ className: classes.prejoinLoader }, { children: [_jsx(CircularProgress, {}), _jsx(Typography, Object.assign({ component: 'div', variant: "body2" }, { children: _jsx(FormattedMessage, { id: "ui.liveStreamRoom.connecting", defaultMessage: "ui.liveStreamRoom.connecting" }) }))] })))] })), _jsxs(Box, Object.assign({ className: classes.endPrejoinContent }, { children: [scLiveStream && (_jsxs(Stack, Object.assign({ sx: { width: '47%' }, spacing: 1, className: classes.endPrejoinContentBox }, { children: [scLiveStream &&
scUserContext.user &&
scUserContext.user.id !== scLiveStream.host.id &&
scLiveStream &&
(((_a = scLiveStream.settings) === null || _a === void 0 ? void 0 : _a.muteParticipants) || ((_b = scLiveStream.settings) === null || _b === void 0 ? void 0 : _b.disableVideo)) && (_jsxs(Alert, Object.assign({ variant: "filled", severity: "info", className: classes.preJoinAlert }, { children: [_jsx(AlertTitle, { children: _jsx("b", { children: "Info" }) }), ((_c = scLiveStream.settings) === null || _c === void 0 ? void 0 : _c.muteParticipants) && (_jsxs(_Fragment, { children: ["-", ' ', _jsx(FormattedMessage, { id: "ui.liveStreamRoom.hostDisableMicrophone", defaultMessage: "ui.liveStreamRoom.hostDisableMicrophone" }), _jsx("br", {})] })), ((_d = scLiveStream.settings) === null || _d === void 0 ? void 0 : _d.disableVideo) && (_jsxs(_Fragment, { children: ["- ", _jsx(FormattedMessage, { id: "ui.liveStreamRoom.hostDisableVideo", defaultMessage: "ui.liveStreamRoom.hostDisableVideo" })] }))] }))), _jsx(CopyTextField, { className: classes.shareLink, value: `${appUrl}${scRoutingContext.url(SCRoutes.LIVESTREAM_ROUTE_NAME, scLiveStream)}`, label: _jsx(FormattedMessage, { id: "ui.liveStreamRoom.shareLink", defaultMessage: "ui.liveStreamRoom.shareLink" }) }), (scLiveStream === null || scLiveStream === void 0 ? void 0 : scLiveStream.description) && (_jsxs(Alert, Object.assign({ variant: "filled", severity: "info", className: classes.description }, { children: [_jsx(AlertTitle, { children: _jsx("b", { children: _jsx(FormattedMessage, { id: "ui.liveStreamRoom.description", defaultMessage: "ui.liveStreamRoom.description" }) }) }), scLiveStream === null || scLiveStream === void 0 ? void 0 : scLiveStream.description] })))] }))), endPrejoinContent] }))] })) : (_jsx(Box, Object.assign({ className: classes.conference }, { children: _jsx(LiveStreamContext.Provider, Object.assign({ value: { liveStream: scLiveStream } }, { children: _jsx(LiveStreamVideoConference, Object.assign({ connectionDetails: connectionDetails, userChoices: preJoinChoices }, LiveStreamVideoConferenceComponentProps)) })) }))) }))) })));
}