lbry-desktop/ui/component/livestreamLayout/view.jsx

222 lines
6.9 KiB
React
Raw Normal View History

// @flow
import 'scss/component/_swipeable-drawer.scss';
2022-02-16 09:49:17 -03:00
// $FlowFixMe
import { Global } from '@emotion/react';
import { lazyImport } from 'util/lazyImport';
2021-04-23 15:59:48 -04:00
import { useIsMobile } from 'effects/use-screensize';
import { Menu, MenuList, MenuButton, MenuItem } from '@reach/menu-button';
import FileTitleSection from 'component/fileTitleSection';
import LivestreamLink from 'component/livestreamLink';
import React from 'react';
import { PRIMARY_PLAYER_WRAPPER_CLASS } from 'page/file/view';
import FileRenderInitiator from 'component/fileRenderInitiator';
import LivestreamIframeRender from './iframe-render';
import * as ICONS from 'constants/icons';
import SwipeableDrawer from 'component/swipeableDrawer';
import { DrawerExpandButton } from 'component/swipeableDrawer/view';
import LivestreamMenu from 'component/livestreamChatLayout/livestream-menu';
import Icon from 'component/common/icon';
import CreditAmount from 'component/common/credit-amount';
import { getTipValues } from 'util/livestream';
const LivestreamChatLayout = lazyImport(() => import('component/livestreamChatLayout' /* webpackChunkName: "chat" */));
const VIEW_MODES = {
CHAT: 'chat',
SUPERCHAT: 'sc',
};
type Props = {
activeStreamUri: boolean | string,
claim: ?StreamClaim,
hideComments: boolean,
isCurrentClaimLive: boolean,
release: any,
showLivestream: boolean,
showScheduledInfo: boolean,
uri: string,
superChats: Array<Comment>,
activeViewers?: number,
};
export default function LivestreamLayout(props: Props) {
const {
activeStreamUri,
claim,
hideComments,
isCurrentClaimLive,
release,
showLivestream,
showScheduledInfo,
uri,
superChats,
activeViewers,
} = props;
2021-04-23 15:59:48 -04:00
const isMobile = useIsMobile();
const [showChat, setShowChat] = React.useState(undefined);
const [superchatsHidden, setSuperchatsHidden] = React.useState(false);
const [chatViewMode, setChatViewMode] = React.useState(VIEW_MODES.CHAT);
if (!claim || !claim.signing_channel) return null;
const { name: channelName, claim_id: channelClaimId } = claim.signing_channel;
2021-03-18 13:21:49 -04:00
return (
<>
2022-02-16 09:49:17 -03:00
{!isMobile && <GlobalStyles />}
<div className="section card-stack">
<React.Suspense fallback={null}>
{isMobile && isCurrentClaimLive ? (
<div className={PRIMARY_PLAYER_WRAPPER_CLASS}>
{/* Mobile needs to handle the livestream player like any video player */}
<FileRenderInitiator uri={claim.canonical_url} />
</div>
) : (
<LivestreamIframeRender
channelClaimId={channelClaimId}
release={release}
showLivestream={showLivestream}
showScheduledInfo={showScheduledInfo}
/>
)}
</React.Suspense>
{hideComments && !showScheduledInfo && (
<div className="help--notice">
2021-07-26 22:27:39 +08:00
{channelName
? __('%channel% has disabled chat for this stream. Enjoy the stream!', { channel: channelName })
2021-07-26 22:27:39 +08:00
: __('This channel has disabled chat for this stream. Enjoy the stream!')}
</div>
)}
{!activeStreamUri && !showScheduledInfo && !isCurrentClaimLive && (
2021-03-18 13:21:49 -04:00
<div className="help--notice">
2021-07-26 22:27:39 +08:00
{channelName
? __("%channelName% isn't live right now, but the chat is! Check back later to watch the stream.", {
channelName,
})
: __("This channel isn't live right now, but the chat is! Check back later to watch the stream.")}
2021-03-18 13:21:49 -04:00
</div>
)}
2021-04-23 15:59:48 -04:00
{activeStreamUri && (
<LivestreamLink
title={__("Click here to access the stream that's currently active")}
claimUri={activeStreamUri}
/>
)}
{isMobile && !hideComments && (
<React.Suspense fallback={null}>
<SwipeableDrawer
open={Boolean(showChat)}
toggleDrawer={() => setShowChat(!showChat)}
title={
<ChatModeSelector
superChats={superChats}
chatViewMode={chatViewMode}
setChatViewMode={(mode) => setChatViewMode(mode)}
activeViewers={activeViewers}
/>
}
hasSubtitle={activeViewers}
actions={
<LivestreamMenu
noSuperchats={!superChats || superChats.length === 0}
superchatsHidden={superchatsHidden}
toggleSuperchats={() => setSuperchatsHidden(!superchatsHidden)}
isMobile
/>
}
>
<LivestreamChatLayout
uri={uri}
hideHeader
superchatsHidden={superchatsHidden}
customViewMode={chatViewMode}
setCustomViewMode={(mode) => setChatViewMode(mode)}
/>
</SwipeableDrawer>
2021-04-23 15:59:48 -04:00
<DrawerExpandButton label={__('Open Live Chat')} toggleDrawer={() => setShowChat(!showChat)} />
</React.Suspense>
)}
<FileTitleSection uri={uri} livestream isLive={showLivestream} />
</div>
</>
);
}
const ChatModeSelector = (chatSelectorProps: any) => {
const { superChats, chatViewMode, setChatViewMode, activeViewers } = chatSelectorProps;
const { superChatsFiatAmount, superChatsLBCAmount } = getTipValues(superChats);
const titleProps = { chatViewMode, activeViewers };
if (!superChats) {
return <ChatDrawerTitle {...titleProps} />;
}
return (
<Menu>
<MenuButton>
<div className="swipeable-drawer__menu">
<ChatDrawerTitle {...titleProps} />
<Icon icon={ICONS.DOWN} />
</div>
</MenuButton>
<MenuList className="menu__list--header">
<MenuItem className="menu__link" onSelect={() => setChatViewMode(VIEW_MODES.CHAT)}>
{__('Live Chat')}
</MenuItem>
<MenuItem className="menu__link" onSelect={() => setChatViewMode(VIEW_MODES.SUPERCHAT)}>
<div className="recommended-content__toggles">
<CreditAmount amount={superChatsLBCAmount || 0} size={8} /> /
<CreditAmount amount={superChatsFiatAmount || 0} size={8} isFiat /> {__('Tipped')}
</div>
</MenuItem>
</MenuList>
</Menu>
);
};
const ChatDrawerTitle = (titleProps: any) => {
const { chatViewMode, activeViewers } = titleProps;
return (
<div className="swipeable-drawer__title-menu">
<span className="swipeable-drawer__title">
{chatViewMode === VIEW_MODES.CHAT ? __('Live Chat') : __('HyperChats')}
</span>
{activeViewers && (
<span className="swipeable-drawer__subtitle">{__('%view_count% viewers', { view_count: activeViewers })}</span>
)}
</div>
);
};
2022-02-16 09:49:17 -03:00
const GlobalStyles = () => (
<Global
styles={{
body: {
'scrollbar-width': '0px',
'&::-webkit-scrollbar': {
width: '0px',
height: '0px',
},
},
}}
/>
);