Create stories for layout testing (#2722)
* Inject services with useContext * Extract service for video settings * Create mock factories for services * Create test data for chat history * Add story to visualize different layouts * Fix renaming mistake * Add landscape and portrait viewports * Add landscape stories --------- Co-authored-by: Gabe Kangas <gabek@real-ity.com>
This commit is contained in:
parent
f0f9c2ced1
commit
b38df2fbe3
@ -4,6 +4,68 @@ import '../styles/theme.less';
|
||||
import './preview.scss';
|
||||
import { themes } from '@storybook/theming';
|
||||
import { DocsContainer } from './storybook-theme';
|
||||
import { INITIAL_VIEWPORTS } from '@storybook/addon-viewport';
|
||||
import _ from 'lodash';
|
||||
|
||||
/**
|
||||
* Takes an entry of a viewport (from Object.entries()) and converts it
|
||||
* into two entries, one for landscape and one for portrait.
|
||||
*
|
||||
* @template {string} Key
|
||||
*
|
||||
* @param {[Key, import('@storybook/addon-viewport/dist/ts3.9/models').Viewport]} entry
|
||||
* @returns {Array<[`${Key}${'Portrait' | 'Landscape'}`, import('@storybook/addon-viewport/dist/ts3.9/models').Viewport]>}
|
||||
*/
|
||||
const convertToLandscapeAndPortraitEntries = ([objectKey, viewport]) => {
|
||||
const pixelStringToNumber = str => parseInt(str.split('px')[0]);
|
||||
const dimensions = [viewport.styles.width, viewport.styles.height].map(pixelStringToNumber);
|
||||
const minDimension = Math.min(...dimensions);
|
||||
const maxDimension = Math.max(...dimensions);
|
||||
|
||||
return [
|
||||
[
|
||||
`${objectKey}Portrait`,
|
||||
{
|
||||
...viewport,
|
||||
name: viewport.name + ' (Portrait)',
|
||||
styles: {
|
||||
...viewport.styles,
|
||||
height: maxDimension + 'px',
|
||||
width: minDimension + 'px',
|
||||
},
|
||||
},
|
||||
],
|
||||
[
|
||||
`${objectKey}Landscape`,
|
||||
{
|
||||
...viewport,
|
||||
name: viewport.name + ' (Landscape)',
|
||||
styles: {
|
||||
...viewport.styles,
|
||||
height: minDimension + 'px',
|
||||
width: maxDimension + 'px',
|
||||
},
|
||||
},
|
||||
],
|
||||
];
|
||||
};
|
||||
|
||||
/**
|
||||
* Takes an object and a function f and returns a new object.
|
||||
* f takes the original object's entries (key-value-pairs
|
||||
* from Object.entries) and returns a list of new entries
|
||||
* (also key-value-pairs). These new entries then form the
|
||||
* result.
|
||||
* @template {string | number} OriginalKey
|
||||
* @template {string | number} NewKey
|
||||
* @template OriginalValue
|
||||
* @template OriginalValue
|
||||
*
|
||||
* @param {Record<OriginalKey, OriginalValue>} obj
|
||||
* @param {(entry: [OriginalKey, OriginalValue], index: number, all: Array<[OriginalKey, OriginalValue]>) => Array<[NewKey, NewValue]>} f
|
||||
* @returns {Record<NewKey, NevValue>}
|
||||
*/
|
||||
const flatMapObject = (obj, f) => Object.fromEntries(Object.entries(obj).flatMap(f));
|
||||
|
||||
export const parameters = {
|
||||
fetchMock: {
|
||||
@ -36,4 +98,9 @@ export const parameters = {
|
||||
// Override the default light theme
|
||||
light: { ...themes.normal },
|
||||
},
|
||||
viewport: {
|
||||
// Take a bunch of viewports from the storybook addon and convert them
|
||||
// to portrait + landscape. Keys are appended with 'Landscape' or 'Portrait'.
|
||||
viewports: flatMapObject(INITIAL_VIEWPORTS, convertToLandscapeAndPortraitEntries),
|
||||
},
|
||||
};
|
||||
|
158
web/components/layouts/Main/Main.stories.tsx
Normal file
158
web/components/layouts/Main/Main.stories.tsx
Normal file
@ -0,0 +1,158 @@
|
||||
import { ComponentMeta, ComponentStory } from '@storybook/react';
|
||||
import { MutableSnapshot, RecoilRoot } from 'recoil';
|
||||
import { makeEmptyClientConfig } from '../../../interfaces/client-config.model';
|
||||
import { ServerStatus, makeEmptyServerStatus } from '../../../interfaces/server-status.model';
|
||||
import {
|
||||
accessTokenAtom,
|
||||
appStateAtom,
|
||||
chatMessagesAtom,
|
||||
chatVisibleToggleAtom,
|
||||
clientConfigStateAtom,
|
||||
currentUserAtom,
|
||||
fatalErrorStateAtom,
|
||||
isMobileAtom,
|
||||
isVideoPlayingAtom,
|
||||
serverStatusState,
|
||||
} from '../../stores/ClientConfigStore';
|
||||
import { Main } from './Main';
|
||||
import { ClientConfigServiceContext } from '../../../services/client-config-service';
|
||||
import { ChatServiceContext } from '../../../services/chat-service';
|
||||
import {
|
||||
ServerStatusServiceContext,
|
||||
ServerStatusStaticService,
|
||||
} from '../../../services/status-service';
|
||||
import { clientConfigServiceMockOf } from '../../../services/client-config-service.mock';
|
||||
import chatServiceMockOf from '../../../services/chat-service.mock';
|
||||
import serverStatusServiceMockOf from '../../../services/status-service.mock';
|
||||
import { VideoSettingsServiceContext } from '../../../services/video-settings-service';
|
||||
import videoSettingsServiceMockOf from '../../../services/video-settings-service.mock';
|
||||
import { grootUser, spidermanUser } from '../../../interfaces/user.fixture';
|
||||
import { exampleChatHistory } from '../../../interfaces/chat-message.fixture';
|
||||
|
||||
export default {
|
||||
title: 'owncast/Layout/Main',
|
||||
parameters: {
|
||||
layout: 'fullscreen',
|
||||
},
|
||||
} satisfies ComponentMeta<typeof Main>;
|
||||
|
||||
type StateInitializer = (mutableState: MutableSnapshot) => void;
|
||||
|
||||
const composeStateInitializers =
|
||||
(...fns: Array<StateInitializer>): StateInitializer =>
|
||||
state =>
|
||||
fns.forEach(fn => fn?.(state));
|
||||
|
||||
const defaultClientConfig = {
|
||||
...makeEmptyClientConfig(),
|
||||
logo: 'http://localhost:8080/logo',
|
||||
name: "Spiderman's super serious stream",
|
||||
summary: 'Strong Spidey stops supervillains! Streamed Saturdays & Sundays.',
|
||||
extraPageContent: '<marquee>Spiderman is cool</marquee>',
|
||||
};
|
||||
|
||||
const defaultServerStatus = makeEmptyServerStatus();
|
||||
const onlineServerStatus: ServerStatus = {
|
||||
...defaultServerStatus,
|
||||
online: true,
|
||||
viewerCount: 5,
|
||||
};
|
||||
|
||||
const initializeDefaultState = (mutableState: MutableSnapshot) => {
|
||||
mutableState.set(appStateAtom, {
|
||||
videoAvailable: false,
|
||||
chatAvailable: false,
|
||||
});
|
||||
mutableState.set(clientConfigStateAtom, defaultClientConfig);
|
||||
mutableState.set(chatVisibleToggleAtom, true);
|
||||
mutableState.set(accessTokenAtom, 'token');
|
||||
mutableState.set(currentUserAtom, {
|
||||
...spidermanUser,
|
||||
isModerator: false,
|
||||
});
|
||||
mutableState.set(serverStatusState, defaultServerStatus);
|
||||
mutableState.set(isMobileAtom, false);
|
||||
|
||||
mutableState.set(chatMessagesAtom, exampleChatHistory);
|
||||
mutableState.set(isVideoPlayingAtom, false);
|
||||
mutableState.set(fatalErrorStateAtom, null);
|
||||
};
|
||||
|
||||
const ClientConfigServiceMock = clientConfigServiceMockOf(defaultClientConfig);
|
||||
const ChatServiceMock = chatServiceMockOf(exampleChatHistory, {
|
||||
...grootUser,
|
||||
accessToken: 'some fake token',
|
||||
});
|
||||
const DefaultServerStatusServiceMock = serverStatusServiceMockOf(defaultServerStatus);
|
||||
const OnlineServerStatusServiceMock = serverStatusServiceMockOf(onlineServerStatus);
|
||||
const VideoSettingsServiceMock = videoSettingsServiceMockOf([]);
|
||||
|
||||
const Template: ComponentStory<typeof Main> = ({
|
||||
initializeState,
|
||||
ServerStatusServiceMock = DefaultServerStatusServiceMock,
|
||||
...args
|
||||
}: {
|
||||
initializeState: (mutableState: MutableSnapshot) => void;
|
||||
ServerStatusServiceMock: ServerStatusStaticService;
|
||||
}) => (
|
||||
<RecoilRoot initializeState={composeStateInitializers(initializeDefaultState, initializeState)}>
|
||||
<ClientConfigServiceContext.Provider value={ClientConfigServiceMock}>
|
||||
<ChatServiceContext.Provider value={ChatServiceMock}>
|
||||
<ServerStatusServiceContext.Provider value={ServerStatusServiceMock}>
|
||||
<VideoSettingsServiceContext.Provider value={VideoSettingsServiceMock}>
|
||||
<Main {...args} />
|
||||
</VideoSettingsServiceContext.Provider>
|
||||
</ServerStatusServiceContext.Provider>
|
||||
</ChatServiceContext.Provider>
|
||||
</ClientConfigServiceContext.Provider>
|
||||
</RecoilRoot>
|
||||
);
|
||||
|
||||
export const OfflineDesktop: typeof Template = Template.bind({});
|
||||
|
||||
export const OfflineMobile: typeof Template = Template.bind({});
|
||||
OfflineMobile.args = {
|
||||
initializeState: (mutableState: MutableSnapshot) => {
|
||||
mutableState.set(isMobileAtom, true);
|
||||
},
|
||||
};
|
||||
OfflineMobile.parameters = {
|
||||
viewport: {
|
||||
defaultViewport: 'mobile1',
|
||||
},
|
||||
};
|
||||
|
||||
export const OfflineTablet: typeof Template = Template.bind({});
|
||||
OfflineTablet.parameters = {
|
||||
viewport: {
|
||||
defaultViewport: 'tablet',
|
||||
},
|
||||
};
|
||||
|
||||
export const Online: typeof Template = Template.bind({});
|
||||
Online.args = {
|
||||
ServerStatusServiceMock: OnlineServerStatusServiceMock,
|
||||
};
|
||||
|
||||
export const OnlineMobile: typeof Template = Online.bind({});
|
||||
OnlineMobile.args = {
|
||||
ServerStatusServiceMock: OnlineServerStatusServiceMock,
|
||||
initializeState: (mutableState: MutableSnapshot) => {
|
||||
mutableState.set(isMobileAtom, true);
|
||||
},
|
||||
};
|
||||
OnlineMobile.parameters = {
|
||||
viewport: {
|
||||
defaultViewport: 'mobile1',
|
||||
},
|
||||
};
|
||||
|
||||
export const OnlineTablet: typeof Template = Online.bind({});
|
||||
OnlineTablet.args = {
|
||||
ServerStatusServiceMock: OnlineServerStatusServiceMock,
|
||||
};
|
||||
OnlineTablet.parameters = {
|
||||
viewport: {
|
||||
defaultViewport: 'tablet',
|
||||
},
|
||||
};
|
@ -1,9 +1,9 @@
|
||||
import { FC, useEffect, useState } from 'react';
|
||||
import { FC, useContext, useEffect, useState } from 'react';
|
||||
import { atom, selector, useRecoilState, useSetRecoilState, RecoilEnv } from 'recoil';
|
||||
import { useMachine } from '@xstate/react';
|
||||
import { makeEmptyClientConfig, ClientConfig } from '../../interfaces/client-config.model';
|
||||
import ClientConfigService from '../../services/client-config-service';
|
||||
import ChatService from '../../services/chat-service';
|
||||
import { ClientConfigServiceContext } from '../../services/client-config-service';
|
||||
import { ChatServiceContext } from '../../services/chat-service';
|
||||
import WebsocketService from '../../services/websocket-service';
|
||||
import { ChatMessage } from '../../interfaces/chat-message.model';
|
||||
import { CurrentUser } from '../../interfaces/current-user';
|
||||
@ -24,7 +24,7 @@ import {
|
||||
} from '../../interfaces/socket-events';
|
||||
import { mergeMeta } from '../../utils/helpers';
|
||||
import handleConnectedClientInfoMessage from './eventhandlers/connected-client-info-handler';
|
||||
import ServerStatusService from '../../services/status-service';
|
||||
import { ServerStatusServiceContext } from '../../services/status-service';
|
||||
import handleNameChangeEvent from './eventhandlers/handleNameChangeEvent';
|
||||
import { DisplayableError } from '../../types/displayable-error';
|
||||
|
||||
@ -155,6 +155,10 @@ export const visibleChatMessagesSelector = selector<ChatMessage[]>({
|
||||
});
|
||||
|
||||
export const ClientConfigStore: FC = () => {
|
||||
const ClientConfigService = useContext(ClientConfigServiceContext);
|
||||
const ChatService = useContext(ChatServiceContext);
|
||||
const ServerStatusService = useContext(ServerStatusServiceContext);
|
||||
|
||||
const [appState, appStateSend, appStateService] = useMachine(appStateModel);
|
||||
const [currentUser, setCurrentUser] = useRecoilState(currentUserAtom);
|
||||
const setChatAuthenticated = useSetRecoilState<boolean>(chatAuthenticatedAtom);
|
||||
@ -209,7 +213,7 @@ export const ClientConfigStore: FC = () => {
|
||||
setHasLoadedConfig(true);
|
||||
} catch (error) {
|
||||
setGlobalFatalError('Unable to reach Owncast server', serverConnectivityError);
|
||||
console.error(`ClientConfigService -> getConfig() ERROR: \n${error}`);
|
||||
console.error(`ClientConfigService -> getConfig() ERROR: \n`, error);
|
||||
}
|
||||
};
|
||||
|
||||
@ -228,7 +232,7 @@ export const ClientConfigStore: FC = () => {
|
||||
} catch (error) {
|
||||
sendEvent([AppStateEvent.Fail]);
|
||||
setGlobalFatalError('Unable to reach Owncast server', serverConnectivityError);
|
||||
console.error(`serverStatusState -> getStatus() ERROR: \n${error}`);
|
||||
console.error(`serverStatusState -> getStatus() ERROR: \n`, error);
|
||||
}
|
||||
};
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
import React, { FC, useEffect } from 'react';
|
||||
import React, { FC, useContext, useEffect } from 'react';
|
||||
import { useRecoilState, useRecoilValue } from 'recoil';
|
||||
import { useHotkeys } from 'react-hotkeys-hook';
|
||||
import { VideoJsPlayerOptions } from 'video.js';
|
||||
@ -12,8 +12,8 @@ import PlaybackMetrics from '../metrics/playback';
|
||||
import createVideoSettingsMenuButton from '../settings-menu';
|
||||
import LatencyCompensator from '../latencyCompensator';
|
||||
import styles from './OwncastPlayer.module.scss';
|
||||
import { VideoSettingsServiceContext } from '../../../services/video-settings-service';
|
||||
|
||||
const VIDEO_CONFIG_URL = '/api/video/variants';
|
||||
const PLAYER_VOLUME = 'owncast_volume';
|
||||
const LATENCY_COMPENSATION_ENABLED = 'latencyCompensatorEnabled';
|
||||
|
||||
@ -30,18 +30,6 @@ export type OwncastPlayerProps = {
|
||||
className?: string;
|
||||
};
|
||||
|
||||
async function getVideoSettings() {
|
||||
let qualities = [];
|
||||
|
||||
try {
|
||||
const response = await fetch(VIDEO_CONFIG_URL);
|
||||
qualities = await response.json();
|
||||
} catch (e) {
|
||||
console.error(e);
|
||||
}
|
||||
return qualities;
|
||||
}
|
||||
|
||||
export const OwncastPlayer: FC<OwncastPlayerProps> = ({
|
||||
source,
|
||||
online,
|
||||
@ -49,6 +37,7 @@ export const OwncastPlayer: FC<OwncastPlayerProps> = ({
|
||||
title,
|
||||
className,
|
||||
}) => {
|
||||
const VideoSettingsService = useContext(VideoSettingsServiceContext);
|
||||
const playerRef = React.useRef(null);
|
||||
const [videoPlaying, setVideoPlaying] = useRecoilState<boolean>(isVideoPlayingAtom);
|
||||
const clockSkew = useRecoilValue<Number>(clockSkewAtom);
|
||||
@ -151,7 +140,7 @@ export const OwncastPlayer: FC<OwncastPlayerProps> = ({
|
||||
};
|
||||
|
||||
const createSettings = async (player, videojs) => {
|
||||
const videoQualities = await getVideoSettings();
|
||||
const videoQualities = await VideoSettingsService.getVideoQualities();
|
||||
const menuButton = createVideoSettingsMenuButton(
|
||||
player,
|
||||
videojs,
|
||||
|
62
web/interfaces/chat-message.fixture.ts
Normal file
62
web/interfaces/chat-message.fixture.ts
Normal file
@ -0,0 +1,62 @@
|
||||
import { ChatMessage } from './chat-message.model';
|
||||
import { MessageType } from './socket-events';
|
||||
import { spidermanUser, grootUser } from './user.fixture';
|
||||
import { User } from './user.model';
|
||||
|
||||
export const createMessages = (
|
||||
basicMessages: Array<{ body: string; user: User }>,
|
||||
): Array<ChatMessage> => {
|
||||
const baseDate = new Date(2022, 1, 3).valueOf();
|
||||
return basicMessages.map(
|
||||
({ body, user }, index): ChatMessage => ({
|
||||
body,
|
||||
user,
|
||||
id: index.toString(),
|
||||
type: MessageType.CHAT,
|
||||
timestamp: new Date(baseDate + 1_000 * index),
|
||||
}),
|
||||
);
|
||||
};
|
||||
|
||||
export const exampleChatHistory = createMessages([
|
||||
{
|
||||
body: 'So, how do you like my new suit?',
|
||||
user: spidermanUser,
|
||||
},
|
||||
{
|
||||
body: 'Im am Groot.',
|
||||
user: grootUser,
|
||||
},
|
||||
{
|
||||
body: 'Really? That bad?',
|
||||
user: spidermanUser,
|
||||
},
|
||||
{
|
||||
body: 'Im am Groot!',
|
||||
user: grootUser,
|
||||
},
|
||||
{
|
||||
body: 'But what about the new web slingers?',
|
||||
user: spidermanUser,
|
||||
},
|
||||
{
|
||||
body: 'Im am Groooooooooooooooot.',
|
||||
user: grootUser,
|
||||
},
|
||||
{
|
||||
body: "Ugh, come on, they aren't THAT big!",
|
||||
user: spidermanUser,
|
||||
},
|
||||
{
|
||||
body: 'I am Groot.',
|
||||
user: grootUser,
|
||||
},
|
||||
{
|
||||
body: "Fine then. I don't like your new leaves either!",
|
||||
user: spidermanUser,
|
||||
},
|
||||
{
|
||||
body: 'I AM GROOT!!!!!',
|
||||
user: grootUser,
|
||||
},
|
||||
]);
|
15
web/interfaces/user.fixture.ts
Normal file
15
web/interfaces/user.fixture.ts
Normal file
@ -0,0 +1,15 @@
|
||||
import { User } from './user.model';
|
||||
|
||||
export const createUser = (name: string, color: number, createdAt: Date): User => ({
|
||||
id: name,
|
||||
displayName: name,
|
||||
displayColor: color,
|
||||
createdAt,
|
||||
authenticated: true,
|
||||
nameChangedAt: createdAt,
|
||||
previousNames: [],
|
||||
scopes: [],
|
||||
});
|
||||
|
||||
export const spidermanUser = createUser('Spiderman', 1, new Date(2020, 1, 2));
|
||||
export const grootUser = createUser('Groot', 1, new Date(2020, 2, 3));
|
18
web/services/chat-service.mock.ts
Normal file
18
web/services/chat-service.mock.ts
Normal file
@ -0,0 +1,18 @@
|
||||
import { ChatMessage } from '../interfaces/chat-message.model';
|
||||
import { ChatStaticService, UserRegistrationResponse } from './chat-service';
|
||||
|
||||
export const chatServiceMockOf = (
|
||||
chatHistory: ChatMessage[],
|
||||
userRegistrationResponse: UserRegistrationResponse,
|
||||
): ChatStaticService =>
|
||||
class ChatServiceMock {
|
||||
public static async getChatHistory(): Promise<ChatMessage[]> {
|
||||
return chatHistory;
|
||||
}
|
||||
|
||||
public static async registerUser(): Promise<UserRegistrationResponse> {
|
||||
return userRegistrationResponse;
|
||||
}
|
||||
};
|
||||
|
||||
export default chatServiceMockOf;
|
@ -1,16 +1,22 @@
|
||||
import { createContext } from 'react';
|
||||
import { ChatMessage } from '../interfaces/chat-message.model';
|
||||
import { getUnauthedData } from '../utils/apis';
|
||||
|
||||
const ENDPOINT = `/api/chat`;
|
||||
const URL_CHAT_REGISTRATION = `/api/chat/register`;
|
||||
|
||||
interface UserRegistrationResponse {
|
||||
export interface UserRegistrationResponse {
|
||||
id: string;
|
||||
accessToken: string;
|
||||
displayName: string;
|
||||
displayColor: number;
|
||||
}
|
||||
|
||||
export interface ChatStaticService {
|
||||
getChatHistory(accessToken: string): Promise<ChatMessage[]>;
|
||||
registerUser(username: string): Promise<UserRegistrationResponse>;
|
||||
}
|
||||
|
||||
class ChatService {
|
||||
public static async getChatHistory(accessToken: string): Promise<ChatMessage[]> {
|
||||
const response = await getUnauthedData(`${ENDPOINT}?accessToken=${accessToken}`);
|
||||
@ -31,4 +37,4 @@ class ChatService {
|
||||
}
|
||||
}
|
||||
|
||||
export default ChatService;
|
||||
export const ChatServiceContext = createContext<ChatStaticService>(ChatService);
|
||||
|
11
web/services/client-config-service.mock.ts
Normal file
11
web/services/client-config-service.mock.ts
Normal file
@ -0,0 +1,11 @@
|
||||
import { ClientConfig } from '../interfaces/client-config.model';
|
||||
import { ClientConfigStaticService } from './client-config-service';
|
||||
|
||||
export const clientConfigServiceMockOf = (config: ClientConfig): ClientConfigStaticService =>
|
||||
class ClientConfigServiceMock {
|
||||
public static async getConfig(): Promise<ClientConfig> {
|
||||
return config;
|
||||
}
|
||||
};
|
||||
|
||||
export default clientConfigServiceMockOf;
|
@ -1,7 +1,12 @@
|
||||
import { createContext } from 'react';
|
||||
import { ClientConfig } from '../interfaces/client-config.model';
|
||||
|
||||
const ENDPOINT = `/api/config`;
|
||||
|
||||
export interface ClientConfigStaticService {
|
||||
getConfig(): Promise<ClientConfig>;
|
||||
}
|
||||
|
||||
class ClientConfigService {
|
||||
public static async getConfig(): Promise<ClientConfig> {
|
||||
const response = await fetch(ENDPOINT);
|
||||
@ -10,4 +15,5 @@ class ClientConfigService {
|
||||
}
|
||||
}
|
||||
|
||||
export default ClientConfigService;
|
||||
export const ClientConfigServiceContext =
|
||||
createContext<ClientConfigStaticService>(ClientConfigService);
|
||||
|
13
web/services/status-service.mock.ts
Normal file
13
web/services/status-service.mock.ts
Normal file
@ -0,0 +1,13 @@
|
||||
import { ServerStatus } from '../interfaces/server-status.model';
|
||||
import { ServerStatusStaticService } from './status-service';
|
||||
|
||||
export const serverStatusServiceMockOf = (
|
||||
serverStatus: ServerStatus,
|
||||
): ServerStatusStaticService =>
|
||||
class ServerStatusServiceMock {
|
||||
public static async getStatus(): Promise<ServerStatus> {
|
||||
return serverStatus;
|
||||
}
|
||||
};
|
||||
|
||||
export default serverStatusServiceMockOf;
|
@ -1,7 +1,12 @@
|
||||
import { createContext } from 'react';
|
||||
import { ServerStatus } from '../interfaces/server-status.model';
|
||||
|
||||
const ENDPOINT = `/api/status`;
|
||||
|
||||
export interface ServerStatusStaticService {
|
||||
getStatus(): Promise<ServerStatus>;
|
||||
}
|
||||
|
||||
class ServerStatusService {
|
||||
public static async getStatus(): Promise<ServerStatus> {
|
||||
const response = await fetch(ENDPOINT);
|
||||
@ -10,4 +15,5 @@ class ServerStatusService {
|
||||
}
|
||||
}
|
||||
|
||||
export default ServerStatusService;
|
||||
export const ServerStatusServiceContext =
|
||||
createContext<ServerStatusStaticService>(ServerStatusService);
|
||||
|
12
web/services/video-settings-service.mock.ts
Normal file
12
web/services/video-settings-service.mock.ts
Normal file
@ -0,0 +1,12 @@
|
||||
import { VideoSettingsStaticService, VideoQuality } from './video-settings-service';
|
||||
|
||||
export const videoSettingsServiceMockOf = (
|
||||
videoQualities: Array<VideoQuality>,
|
||||
): VideoSettingsStaticService =>
|
||||
class VideoSettingsServiceMock {
|
||||
public static async getVideoQualities(): Promise<Array<VideoQuality>> {
|
||||
return videoQualities;
|
||||
}
|
||||
};
|
||||
|
||||
export default videoSettingsServiceMockOf;
|
36
web/services/video-settings-service.ts
Normal file
36
web/services/video-settings-service.ts
Normal file
@ -0,0 +1,36 @@
|
||||
import { createContext } from 'react';
|
||||
|
||||
export type VideoQuality = {
|
||||
index: number;
|
||||
/**
|
||||
* This property is not just for display or so
|
||||
* but it holds information
|
||||
*
|
||||
* @example '1.2Mbps@24fps'
|
||||
*/
|
||||
name: string;
|
||||
};
|
||||
|
||||
export interface VideoSettingsStaticService {
|
||||
getVideoQualities(): Promise<Array<VideoQuality>>;
|
||||
}
|
||||
|
||||
class VideoSettingsService {
|
||||
private static readonly VIDEO_CONFIG_URL = '/api/video/variants';
|
||||
|
||||
public static async getVideoQualities(): Promise<Array<VideoQuality>> {
|
||||
let qualities: Array<VideoQuality> = [];
|
||||
|
||||
try {
|
||||
const response = await fetch(VideoSettingsService.VIDEO_CONFIG_URL);
|
||||
qualities = await response.json();
|
||||
console.log(qualities);
|
||||
} catch (e) {
|
||||
console.error(e);
|
||||
}
|
||||
return qualities;
|
||||
}
|
||||
}
|
||||
|
||||
export const VideoSettingsServiceContext =
|
||||
createContext<VideoSettingsStaticService>(VideoSettingsService);
|
Loading…
Reference in New Issue
Block a user