2021-03-01 18:07:10 -08:00
|
|
|
// @flow
|
2022-01-14 17:24:16 -03:00
|
|
|
import { formatLbryChannelName } from 'util/url';
|
2021-10-18 23:54:59 +08:00
|
|
|
import { lazyImport } from 'util/lazyImport';
|
2022-03-16 14:35:27 -03:00
|
|
|
import { LIVESTREAM_STARTS_SOON_BUFFER, LIVESTREAM_STARTED_RECENTLY_BUFFER } from 'constants/livestream';
|
2021-03-15 10:32:51 -04:00
|
|
|
import analytics from 'analytics';
|
2022-01-14 17:24:16 -03:00
|
|
|
import LivestreamLayout from 'component/livestreamLayout';
|
2021-12-16 15:59:13 -06:00
|
|
|
import moment from 'moment';
|
2022-01-14 17:24:16 -03:00
|
|
|
import Page from 'component/page';
|
|
|
|
import React from 'react';
|
2022-03-16 18:39:09 -03:00
|
|
|
import { useIsMobile } from 'effects/use-screensize';
|
2022-03-16 14:35:27 -03:00
|
|
|
import useFetchLiveStatus from 'effects/use-fetch-live';
|
2021-03-01 18:07:10 -08:00
|
|
|
|
2022-01-14 17:24:16 -03:00
|
|
|
const LivestreamChatLayout = lazyImport(() => import('component/livestreamChatLayout' /* webpackChunkName: "chat" */));
|
2021-10-18 23:54:59 +08:00
|
|
|
|
2021-03-01 18:07:10 -08:00
|
|
|
type Props = {
|
2022-01-14 17:24:16 -03:00
|
|
|
activeLivestreamForChannel: any,
|
|
|
|
activeLivestreamInitialized: boolean,
|
2021-12-09 17:23:23 +08:00
|
|
|
channelClaimId: ?string,
|
2021-06-03 11:55:16 -04:00
|
|
|
chatDisabled: boolean,
|
2022-01-14 17:24:16 -03:00
|
|
|
claim: StreamClaim,
|
|
|
|
isAuthenticated: boolean,
|
|
|
|
uri: string,
|
2022-03-15 13:28:55 -03:00
|
|
|
socketConnected: boolean,
|
2022-03-15 13:18:08 -03:00
|
|
|
doSetPrimaryUri: (uri: ?string) => void,
|
2022-01-14 17:24:16 -03:00
|
|
|
doCommentSocketConnect: (string, string, string) => void,
|
2022-01-06 11:49:49 -06:00
|
|
|
doFetchChannelLiveStatus: (string) => void,
|
2022-01-14 17:24:16 -03:00
|
|
|
doUserSetReferrer: (string) => void,
|
2021-03-01 18:07:10 -08:00
|
|
|
};
|
|
|
|
|
2022-03-21 11:11:26 -03:00
|
|
|
export const LivestreamContext = React.createContext<any>();
|
2022-03-16 18:39:09 -03:00
|
|
|
|
2021-03-15 10:32:51 -04:00
|
|
|
export default function LivestreamPage(props: Props) {
|
2021-12-16 15:59:13 -06:00
|
|
|
const {
|
2022-01-14 17:24:16 -03:00
|
|
|
activeLivestreamForChannel,
|
|
|
|
activeLivestreamInitialized,
|
2021-12-16 15:59:13 -06:00
|
|
|
channelClaimId,
|
|
|
|
chatDisabled,
|
2022-01-14 17:24:16 -03:00
|
|
|
claim,
|
|
|
|
isAuthenticated,
|
|
|
|
uri,
|
2022-03-15 13:28:55 -03:00
|
|
|
socketConnected,
|
2022-03-15 13:18:08 -03:00
|
|
|
doSetPrimaryUri,
|
2021-12-16 15:59:13 -06:00
|
|
|
doCommentSocketConnect,
|
2022-01-06 11:49:49 -06:00
|
|
|
doFetchChannelLiveStatus,
|
2022-01-14 17:24:16 -03:00
|
|
|
doUserSetReferrer,
|
2021-12-16 15:59:13 -06:00
|
|
|
} = props;
|
2021-03-29 19:05:18 -04:00
|
|
|
|
2022-03-16 18:39:09 -03:00
|
|
|
const isMobile = useIsMobile();
|
|
|
|
|
2022-01-14 17:24:16 -03:00
|
|
|
const [activeStreamUri, setActiveStreamUri] = React.useState(false);
|
|
|
|
const [showLivestream, setShowLivestream] = React.useState(false);
|
|
|
|
const [showScheduledInfo, setShowScheduledInfo] = React.useState(false);
|
|
|
|
const [hideComments, setHideComments] = React.useState(false);
|
2022-03-16 18:39:09 -03:00
|
|
|
const [layountRendered, setLayountRendered] = React.useState(chatDisabled || isMobile);
|
2022-01-14 17:24:16 -03:00
|
|
|
|
|
|
|
const isInitialized = Boolean(activeLivestreamForChannel) || activeLivestreamInitialized;
|
|
|
|
const isChannelBroadcasting = Boolean(activeLivestreamForChannel);
|
|
|
|
const claimId = claim && claim.claim_id;
|
|
|
|
const isCurrentClaimLive = isChannelBroadcasting && activeLivestreamForChannel.claimId === claimId;
|
|
|
|
const livestreamChannelId = channelClaimId || '';
|
|
|
|
|
|
|
|
// $FlowFixMe
|
|
|
|
const release = moment.unix(claim.value.release_time);
|
|
|
|
const stringifiedClaim = JSON.stringify(claim);
|
|
|
|
|
2021-11-24 08:03:21 -08:00
|
|
|
React.useEffect(() => {
|
|
|
|
// TODO: This should not be needed one we unify the livestream player (?)
|
|
|
|
analytics.playerLoadedEvent('livestream', false);
|
|
|
|
}, []);
|
|
|
|
|
2022-03-16 08:35:58 -03:00
|
|
|
const { signing_channel: channelClaim } = claim || {};
|
|
|
|
const { canonical_url: channelUrl } = channelClaim || {};
|
|
|
|
|
|
|
|
// On livestream page, only connect, fileRenderFloating will handle disconnect.
|
|
|
|
// (either by leaving page with floating player off, or by closing the player)
|
2021-03-15 10:32:51 -04:00
|
|
|
React.useEffect(() => {
|
2022-03-15 13:28:55 -03:00
|
|
|
if (!claim || socketConnected) return;
|
2022-01-14 17:24:16 -03:00
|
|
|
|
|
|
|
const { claim_id: claimId, signing_channel: channelClaim } = claim;
|
2022-03-16 08:35:58 -03:00
|
|
|
const channelName = channelClaim && formatLbryChannelName(channelUrl);
|
2022-01-14 17:24:16 -03:00
|
|
|
|
|
|
|
if (claimId && channelName) {
|
|
|
|
doCommentSocketConnect(uri, channelName, claimId);
|
2021-03-29 19:05:18 -04:00
|
|
|
}
|
2021-03-15 10:32:51 -04:00
|
|
|
|
2022-03-15 13:28:55 -03:00
|
|
|
// only listen to socketConnected on initial mount
|
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
2022-03-16 08:35:58 -03:00
|
|
|
}, [channelUrl, claim, doCommentSocketConnect, uri]);
|
2021-12-16 15:59:13 -06:00
|
|
|
|
2022-03-15 22:45:16 -04:00
|
|
|
const claimReleaseStartingSoonStatic = () =>
|
|
|
|
release.isBetween(moment(), moment().add(LIVESTREAM_STARTS_SOON_BUFFER, 'minutes'));
|
|
|
|
|
|
|
|
const claimReleaseStartedRecentlyStatic = () =>
|
|
|
|
release.isBetween(moment().subtract(LIVESTREAM_STARTED_RECENTLY_BUFFER, 'minutes'), moment());
|
|
|
|
|
|
|
|
// Find out current channels status + active live claim every 30 seconds (or 15 if not live)
|
|
|
|
const fasterPoll = !isCurrentClaimLive && (claimReleaseStartingSoonStatic() || claimReleaseStartedRecentlyStatic());
|
|
|
|
|
2022-03-16 14:35:27 -03:00
|
|
|
useFetchLiveStatus(livestreamChannelId, doFetchChannelLiveStatus, fasterPoll);
|
2021-12-16 15:59:13 -06:00
|
|
|
|
|
|
|
React.useEffect(() => {
|
2021-12-22 10:12:44 -06:00
|
|
|
setActiveStreamUri(!isCurrentClaimLive && isChannelBroadcasting ? activeLivestreamForChannel.claimUri : false);
|
2021-12-16 15:59:13 -06:00
|
|
|
}, [isCurrentClaimLive, isChannelBroadcasting]); // eslint-disable-line react-hooks/exhaustive-deps
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
if (!isInitialized) return;
|
|
|
|
|
|
|
|
const claimReleaseInFuture = () => release.isAfter();
|
|
|
|
const claimReleaseInPast = () => release.isBefore();
|
|
|
|
|
|
|
|
const claimReleaseStartingSoon = () =>
|
|
|
|
release.isBetween(moment(), moment().add(LIVESTREAM_STARTS_SOON_BUFFER, 'minutes'));
|
|
|
|
|
|
|
|
const claimReleaseStartedRecently = () =>
|
|
|
|
release.isBetween(moment().subtract(LIVESTREAM_STARTED_RECENTLY_BUFFER, 'minutes'), moment());
|
|
|
|
|
|
|
|
const checkShowLivestream = () =>
|
2022-03-15 22:45:16 -04:00
|
|
|
isChannelBroadcasting &&
|
|
|
|
isCurrentClaimLive &&
|
|
|
|
(claimReleaseInPast() || claimReleaseStartingSoon() || claimReleaseInFuture());
|
2021-12-16 15:59:13 -06:00
|
|
|
|
|
|
|
const checkShowScheduledInfo = () =>
|
|
|
|
(!isChannelBroadcasting && (claimReleaseInFuture() || claimReleaseStartedRecently())) ||
|
|
|
|
(isChannelBroadcasting &&
|
|
|
|
((!isCurrentClaimLive && (claimReleaseInFuture() || claimReleaseStartedRecently())) ||
|
|
|
|
(isCurrentClaimLive && claimReleaseInFuture() && !claimReleaseStartingSoon())));
|
|
|
|
|
|
|
|
const checkCommentsDisabled = () => chatDisabled || (claimReleaseInFuture() && !claimReleaseStartingSoon());
|
|
|
|
|
|
|
|
const calculateStreamReleaseState = () => {
|
|
|
|
setShowLivestream(checkShowLivestream());
|
|
|
|
setShowScheduledInfo(checkShowScheduledInfo());
|
|
|
|
setHideComments(checkCommentsDisabled());
|
|
|
|
};
|
|
|
|
|
|
|
|
calculateStreamReleaseState();
|
2022-03-15 22:45:16 -04:00
|
|
|
const intervalId = setInterval(calculateStreamReleaseState, 5000);
|
2021-12-16 15:59:13 -06:00
|
|
|
|
|
|
|
if (isCurrentClaimLive && claimReleaseInPast() && isChannelBroadcasting === true) {
|
|
|
|
clearInterval(intervalId);
|
|
|
|
}
|
|
|
|
|
|
|
|
return () => clearInterval(intervalId);
|
|
|
|
}, [chatDisabled, isChannelBroadcasting, release, isCurrentClaimLive, isInitialized]);
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
if (uri && stringifiedClaim) {
|
|
|
|
const jsonClaim = JSON.parse(stringifiedClaim);
|
2021-03-15 10:32:51 -04:00
|
|
|
if (!isAuthenticated) {
|
|
|
|
const uri = jsonClaim.signing_channel && jsonClaim.signing_channel.permanent_url;
|
2022-01-14 17:24:16 -03:00
|
|
|
if (uri) doUserSetReferrer(uri.replace('lbry://', ''));
|
2021-03-01 18:07:10 -08:00
|
|
|
}
|
|
|
|
}
|
2022-01-14 17:24:16 -03:00
|
|
|
}, [uri, stringifiedClaim, isAuthenticated, doUserSetReferrer]);
|
2021-03-01 18:07:10 -08:00
|
|
|
|
|
|
|
React.useEffect(() => {
|
2022-03-16 18:39:09 -03:00
|
|
|
if (!layountRendered) return;
|
|
|
|
|
2022-03-15 13:18:08 -03:00
|
|
|
doSetPrimaryUri(uri);
|
2022-02-07 13:51:07 -03:00
|
|
|
|
2022-03-15 13:18:08 -03:00
|
|
|
return () => doSetPrimaryUri(null);
|
2022-03-16 18:39:09 -03:00
|
|
|
}, [doSetPrimaryUri, layountRendered, uri]);
|
2021-03-01 18:07:10 -08:00
|
|
|
|
|
|
|
return (
|
2021-12-16 15:59:13 -06:00
|
|
|
<Page
|
2022-02-11 19:50:55 +01:00
|
|
|
className="file-page scheduledLivestream-wrapper"
|
2021-12-16 15:59:13 -06:00
|
|
|
noFooter
|
|
|
|
livestream
|
|
|
|
chatDisabled={hideComments}
|
|
|
|
rightSide={
|
|
|
|
!hideComments &&
|
|
|
|
isInitialized && (
|
|
|
|
<React.Suspense fallback={null}>
|
2022-03-16 18:39:09 -03:00
|
|
|
<LivestreamChatLayout uri={uri} setLayountRendered={setLayountRendered} />
|
2021-12-16 15:59:13 -06:00
|
|
|
</React.Suspense>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
>
|
2022-03-14 12:15:30 -07:00
|
|
|
{isInitialized && (
|
2022-03-21 11:11:26 -03:00
|
|
|
<LivestreamContext.Provider value={{ livestreamPage: true, layountRendered }}>
|
2022-03-16 18:39:09 -03:00
|
|
|
<LivestreamLayout
|
|
|
|
uri={uri}
|
|
|
|
hideComments={hideComments}
|
|
|
|
release={release}
|
|
|
|
isCurrentClaimLive={isCurrentClaimLive}
|
|
|
|
showLivestream={showLivestream}
|
|
|
|
showScheduledInfo={showScheduledInfo}
|
|
|
|
activeStreamUri={activeStreamUri}
|
|
|
|
/>
|
2022-03-21 11:11:26 -03:00
|
|
|
</LivestreamContext.Provider>
|
2021-12-16 15:59:13 -06:00
|
|
|
)}
|
|
|
|
</Page>
|
2021-03-01 18:07:10 -08:00
|
|
|
);
|
|
|
|
}
|