0

112 lines
3.4 KiB
TypeScript
Raw Normal View History

/* eslint-disable react/no-unknown-property */
import React, { useEffect } from 'react';
import { useRecoilValue } from 'recoil';
import { useRouter } from 'next/router';
2023-01-31 23:25:40 -08:00
import { Skeleton } from 'antd';
import {
clientConfigStateAtom,
ClientConfigStore,
isOnlineSelector,
serverStatusState,
2023-01-31 23:25:40 -08:00
appStateAtom,
} from '../../../components/stores/ClientConfigStore';
reafctor: normalize component formatting (#2082) * refactor: move/rename BanUserButton file * refactor: move/rename Chart file * refactor: update generic component filenames to PascalCase * refactor: update config component filenames to PascalCase * refactor: update AdminLayout component filename to PascalCase * refactor: update/move VideoJS component * chore(eslint): disable bad react/require-default-props rule * refactor: normalize ActionButton component * refactor: normalize ActionButtonRow component * refactor: normalize FollowButton component * refactor: normalize NotifyButton component * refactor: normalize ChatActionMessage component * refactor: normalize ChatContainer component * refactor: normalize ChatJoinMessage component * refactor: normalize ChatModerationActionMenu component * refactor: normalize ChatModerationDetailsModal component * refactor: normalize ChatModeratorNotification component * refactor: normalize ChatSocialMessage component * refactor: normalize ChatSystemMessage component * refactor: normalize ChatTextField component * refactor: normalize ChatUserBadge component * refactor: normalize ChatUserMessage component * refactor: normalize ContentHeader component * refactor: normalize OwncastLogo component * refactor: normalize UserDropdown component * chore(eslint): modify react/function-component-definition rule * refactor: normalize CodecSelector component * refactor: update a bunch of functional components using eslint * refactor: update a bunch of functional components using eslint, pt2 * refactor: update a bunch of functional components using eslint, pt3 * refactor: replace all component->component default imports with named imports * refactor: replace all component-stories->component default imports with named imports * refactor: remove default exports from most components * chore(eslint): add eslint config files for the components and pages dirs * fix: use-before-define error in ChatContainer * Fix ChatContainer import * Only process .tsx files in Next builds Co-authored-by: Gabe Kangas <gabek@real-ity.com>
2022-09-07 09:00:28 +02:00
import { Statusbar } from '../../../components/ui/Statusbar/Statusbar';
import { OwncastPlayer } from '../../../components/video/OwncastPlayer/OwncastPlayer';
import { ClientConfig } from '../../../interfaces/client-config.model';
import { ServerStatus } from '../../../interfaces/server-status.model';
2023-01-31 23:25:40 -08:00
import { AppStateOptions } from '../../../components/stores/application-state';
import { Theme } from '../../../components/theme/Theme';
import styles from './VideoEmbed.module.scss';
import { OfflineEmbed } from '../../../components/ui/OfflineEmbed/OfflineEmbed';
2022-05-16 22:55:22 -07:00
export default function VideoEmbed() {
const status = useRecoilValue<ServerStatus>(serverStatusState);
const clientConfig = useRecoilValue<ClientConfig>(clientConfigStateAtom);
2023-01-31 23:25:40 -08:00
const appState = useRecoilValue<AppStateOptions>(appStateAtom);
const { name, summary, offlineMessage, federation } = clientConfig;
const { viewerCount, lastConnectTime, lastDisconnectTime, streamTitle } = status;
const online = useRecoilValue<boolean>(isOnlineSelector);
const { enabled: socialEnabled } = federation;
const router = useRouter();
/**
* router.query isn't initialized until hydration
* (see https://github.com/vercel/next.js/discussions/11484)
* but router.asPath is initialized earlier, so we parse the
* query parameters ourselves
*/
const path = router.asPath.split('?')[1] ?? '';
2023-08-02 13:41:08 -07:00
const query = path.split('&').reduce(
(currQuery, part) => {
const [key, value] = part.split('=');
return { ...currQuery, [key]: value };
},
{} as Record<string, string>,
);
const initiallyMuted = query.initiallyMuted === 'true';
const supportsSocialFollow = socialEnabled && query.supportsSocialFollow !== 'false';
2023-01-31 23:25:40 -08:00
const loadingState = <Skeleton active style={{ padding: '10px' }} paragraph={{ rows: 10 }} />;
// This is a hack to force a specific body background color for just this page.
useEffect(() => {
document.body.classList.add('body-background');
}, []);
2023-01-31 23:25:40 -08:00
const offlineState = (
<OfflineEmbed
streamName={name}
subtitle={offlineMessage || summary}
2024-03-05 22:31:03 -08:00
image="/logo"
supportsFollows={supportsSocialFollow}
/>
2023-01-31 23:25:40 -08:00
);
const onlineState = (
<div className={styles.onlineContainer}>
<style jsx global>
{`
.body-background {
background: var(--theme-color-components-video-status-bar-background);
}
`}
</style>
2023-01-31 23:25:40 -08:00
<OwncastPlayer
source="/hls/stream.m3u8"
online={online}
initiallyMuted={initiallyMuted}
title={streamTitle || name}
/>
<Statusbar
online={online}
lastConnectTime={lastConnectTime}
lastDisconnectTime={lastDisconnectTime}
viewerCount={viewerCount}
/>
</div>
2023-01-31 23:25:40 -08:00
);
const getView = () => {
if (appState.appLoading) {
return loadingState;
}
if (online) {
return onlineState;
}
return offlineState;
};
2022-05-16 22:55:22 -07:00
return (
<>
<ClientConfigStore />
2023-01-31 23:25:40 -08:00
<Theme />
<div className="video-embed">{getView()}</div>
</>
2022-05-16 22:55:22 -07:00
);
}