diff --git a/frontend/package-lock.json b/frontend/package-lock.json index a7bf6997f..d322a1759 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -13,7 +13,6 @@ "@equinor/eds-core-react": "^0.36.1", "@equinor/eds-icons": "^0.21.0", "@equinor/eds-tokens": "^0.9.2", - "@livekit/components-react": "^2.4.3", "@livekit/components-styles": "^1.0.12", "@microsoft/applicationinsights-web": "^3.1.2", "@microsoft/signalr": "^8.0.0", @@ -32,6 +31,7 @@ "react-dom": "^18.2.0", "react-error-boundary": "^4.0.13", "react-modal": "^3.15.1", + "react-player": "^2.16.0", "react-router-dom": "^6.22.3", "react-scripts": "^5.0.1", "styled-components": "^6.1.8", @@ -2774,44 +2774,6 @@ "version": "2.0.4", "license": "MIT" }, - "node_modules/@livekit/components-core": { - "version": "0.11.2", - "resolved": "https://registry.npmjs.org/@livekit/components-core/-/components-core-0.11.2.tgz", - "integrity": "sha512-rXQ1OvyGe9gY8BCpH5FTr4Il17/sS/ecJQbG3PoOXAkQVl5JP965eqUPyKXZTdxNKlVLef00AygrO2pPArwOTA==", - "dependencies": { - "@floating-ui/dom": "1.6.8", - "loglevel": "1.9.1", - "rxjs": "7.8.1" - }, - "engines": { - "node": ">=18" - }, - "peerDependencies": { - "@livekit/protocol": "^1.16.0", - "livekit-client": "^2.4.0", - "tslib": "^2.6.2" - } - }, - "node_modules/@livekit/components-react": { - "version": "2.4.3", - "resolved": "https://registry.npmjs.org/@livekit/components-react/-/components-react-2.4.3.tgz", - "integrity": "sha512-XhCvwFvNjhBJcoQHIY4Hk6MBp7mM9q0n0i7sN/xK3fB1DSjkxIkpc7lh/+Pjqdu6F6OJT3MjwNFYnftqy6kcmw==", - "dependencies": { - "@livekit/components-core": "0.11.2", - "clsx": "2.1.1", - "usehooks-ts": "3.1.0" - }, - "engines": { - "node": ">=18" - }, - "peerDependencies": { - "@livekit/protocol": "^1.16.0", - "livekit-client": "^2.4.0", - "react": ">=18", - "react-dom": ">=18", - "tslib": "^2.6.2" - } - }, "node_modules/@livekit/components-styles": { "version": "1.0.12", "resolved": "https://registry.npmjs.org/@livekit/components-styles/-/components-styles-1.0.12.tgz", @@ -2820,15 +2782,6 @@ "node": ">=18" } }, - "node_modules/@livekit/protocol": { - "version": "1.21.0", - "resolved": "https://registry.npmjs.org/@livekit/protocol/-/protocol-1.21.0.tgz", - "integrity": "sha512-3TohFPNZy1axTuoDLU6mA1rwuP4VawgehvX52OoLJnU+fNQYfmMJqz8k7NSh79jG5I8Og77YYhT905Omrhli2A==", - "peer": true, - "dependencies": { - "@bufbuild/protobuf": "^1.7.2" - } - }, "node_modules/@microsoft/applicationinsights-analytics-js": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/@microsoft/applicationinsights-analytics-js/-/applicationinsights-analytics-js-3.1.2.tgz", @@ -5259,14 +5212,6 @@ "node": ">=0.10.0" } }, - "node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "engines": { - "node": ">=6" - } - }, "node_modules/co": { "version": "4.6.0", "license": "MIT", @@ -9923,6 +9868,11 @@ "@bufbuild/protobuf": "^1.7.2" } }, + "node_modules/load-script": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/load-script/-/load-script-1.0.0.tgz", + "integrity": "sha512-kPEjMFtZvwL9TaZo0uZ2ml+Ye9HUMmPwbYRJ324qF9tqMejwykJ5ggTyvzmrbBeapCAbk98BSbTeovHEEP1uCA==" + }, "node_modules/loader-runner": { "version": "4.3.0", "license": "MIT", @@ -10101,6 +10051,11 @@ "node": ">= 4.0.0" } }, + "node_modules/memoize-one": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/memoize-one/-/memoize-one-5.2.1.tgz", + "integrity": "sha512-zYiwtZUcYyXKo/np96AGZAckk+FWWsUdJ3cHGGmld7+AhvcWmQyGCYUh1hc4Q/pkOhb65dQR/pqCyK0cOaHz4Q==" + }, "node_modules/merge-descriptors": { "version": "1.0.1", "license": "MIT" @@ -12408,6 +12363,11 @@ "version": "6.0.11", "license": "MIT" }, + "node_modules/react-fast-compare": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/react-fast-compare/-/react-fast-compare-3.2.2.tgz", + "integrity": "sha512-nsO+KSNgo1SbJqJEYRE9ERzo7YtYbou/OqjSQKxV7jcKox7+usiUVZOAC+XnDOABXggQTno0Y1CpVnuWEc1boQ==" + }, "node_modules/react-is": { "version": "18.2.0", "license": "MIT" @@ -12433,6 +12393,21 @@ "react-dom": "^0.14.0 || ^15.0.0 || ^16 || ^17 || ^18" } }, + "node_modules/react-player": { + "version": "2.16.0", + "resolved": "https://registry.npmjs.org/react-player/-/react-player-2.16.0.tgz", + "integrity": "sha512-mAIPHfioD7yxO0GNYVFD1303QFtI3lyyQZLY229UEAp/a10cSW+hPcakg0Keq8uWJxT2OiT/4Gt+Lc9bD6bJmQ==", + "dependencies": { + "deepmerge": "^4.0.0", + "load-script": "^1.0.0", + "memoize-one": "^5.1.1", + "prop-types": "^15.7.2", + "react-fast-compare": "^3.0.1" + }, + "peerDependencies": { + "react": ">=16.6.0" + } + }, "node_modules/react-refresh": { "version": "0.11.0", "license": "MIT", @@ -13865,6 +13840,7 @@ "version": "7.8.1", "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.1.tgz", "integrity": "sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==", + "optional": true, "dependencies": { "tslib": "^2.1.0" } @@ -15496,20 +15472,6 @@ "resolved": "https://registry.npmjs.org/url-toolkit/-/url-toolkit-2.2.5.tgz", "integrity": "sha512-mtN6xk+Nac+oyJ/PrI7tzfmomRVNFIWKUbG8jdYFt52hxbiReFAXIjYskvu64/dvuW71IcB7lV8l0HvZMac6Jg==" }, - "node_modules/usehooks-ts": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/usehooks-ts/-/usehooks-ts-3.1.0.tgz", - "integrity": "sha512-bBIa7yUyPhE1BCc0GmR96VU/15l/9gP1Ch5mYdLcFBaFGQsdmXkvjV0TtOqW1yUd6VjIwDunm+flSciCQXujiw==", - "dependencies": { - "lodash.debounce": "^4.0.8" - }, - "engines": { - "node": ">=16.15.0" - }, - "peerDependencies": { - "react": "^16.8.0 || ^17 || ^18" - } - }, "node_modules/util-deprecate": { "version": "1.0.2", "license": "MIT" diff --git a/frontend/package.json b/frontend/package.json index c76e982e1..9b61c2255 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -9,7 +9,6 @@ "@equinor/eds-core-react": "^0.36.1", "@equinor/eds-icons": "^0.21.0", "@equinor/eds-tokens": "^0.9.2", - "@livekit/components-react": "^2.4.3", "@livekit/components-styles": "^1.0.12", "@microsoft/applicationinsights-web": "^3.1.2", "@microsoft/signalr": "^8.0.0", @@ -28,6 +27,7 @@ "react-dom": "^18.2.0", "react-error-boundary": "^4.0.13", "react-modal": "^3.15.1", + "react-player": "^2.16.0", "react-router-dom": "^6.22.3", "react-scripts": "^5.0.1", "styled-components": "^6.1.8", diff --git a/frontend/src/api/ApiCaller.tsx b/frontend/src/api/ApiCaller.tsx index 6d7308b2b..dba00b248 100644 --- a/frontend/src/api/ApiCaller.tsx +++ b/frontend/src/api/ApiCaller.tsx @@ -1,7 +1,6 @@ import { config } from 'config' import { Mission } from 'models/Mission' import { Robot } from 'models/Robot' -import { VideoStream } from 'models/VideoStream' import { filterRobots } from 'utils/filtersAndSorts' import { MissionRunQueryParameters } from 'models/MissionRunQueryParameters' import { MissionDefinitionQueryParameters } from 'models/MissionDefinitionQueryParameters' @@ -264,12 +263,6 @@ export class BackendAPICaller { return result.content } - static async getVideoStreamsByRobotId(robotId: string): Promise { - const path: string = 'robots/' + robotId + '/video-streams' - const result = await BackendAPICaller.GET(path).catch(BackendAPICaller.handleError('GET', path)) - return result.content - } - static async getPlantInfo(): Promise { const path: string = 'mission-loader/plants' const result = await BackendAPICaller.GET(path).catch(BackendAPICaller.handleError('GET', path)) diff --git a/frontend/src/components/Contexts/MediaStreamContext.tsx b/frontend/src/components/Contexts/MediaStreamContext.tsx index cc92da144..1c1251964 100644 --- a/frontend/src/components/Contexts/MediaStreamContext.tsx +++ b/frontend/src/components/Contexts/MediaStreamContext.tsx @@ -2,26 +2,7 @@ import { createContext, FC, useContext, useEffect, useState } from 'react' import { SignalREventLabels, useSignalRContext } from './SignalRContext' import { useRobotContext } from './RobotContext' import { RemoteParticipant, RemoteTrack, RemoteTrackPublication, Room, RoomEvent } from 'livekit-client' - -export enum MediaType { - Video, - Audio, -} - -export enum MediaConnectionType { - LiveKit, -} - -type MediaStreamConfig = { - url: string - streamId: string - authToken: string - mediaType: MediaType - robotId: string - connectionType: MediaConnectionType -} - -type MediaStreamConfigAndTracks = MediaStreamConfig & { streams: MediaStreamTrack[] } +import { MediaConnectionType, MediaStreamConfig, MediaStreamConfigAndTracks } from 'models/VideoStream' type MediaStreamDictionaryType = { [robotId: string]: MediaStreamConfigAndTracks diff --git a/frontend/src/components/Pages/MissionPage/MissionPage.tsx b/frontend/src/components/Pages/MissionPage/MissionPage.tsx index dc34eb2c8..088548b05 100644 --- a/frontend/src/components/Pages/MissionPage/MissionPage.tsx +++ b/frontend/src/components/Pages/MissionPage/MissionPage.tsx @@ -1,7 +1,6 @@ import { TaskTable } from 'components/Pages/MissionPage/TaskOverview/TaskTable' import { VideoStreamWindow } from 'components/Pages/MissionPage/VideoStream/VideoStreamWindow' import { Mission } from 'models/Mission' -import { VideoStream } from 'models/VideoStream' import { useEffect, useState } from 'react' import { useParams } from 'react-router-dom' import styled from 'styled-components' @@ -15,6 +14,7 @@ import { AlertType, useAlertContext } from 'components/Contexts/AlertContext' import { useLanguageContext } from 'components/Contexts/LanguageContext' import { FailedRequestAlertContent, FailedRequestAlertListContent } from 'components/Alerts/FailedRequestAlert' import { AlertCategory } from 'components/Alerts/AlertsBanner' +import { useMediaStreamContext } from 'components/Contexts/MediaStreamContext' const StyledMissionPage = styled.div` display: flex; @@ -41,9 +41,10 @@ export const MissionPage = () => { const { missionId } = useParams() const { TranslateText } = useLanguageContext() const { setAlert, setListAlert } = useAlertContext() - const [videoStreams, setVideoStreams] = useState([]) + const [videoMediaStreams, setVideoMediaStreams] = useState([]) const [selectedMission, setSelectedMission] = useState() const { registerEvent, connectionReady } = useSignalRContext() + const { mediaStreams } = useMediaStreamContext() useEffect(() => { if (connectionReady) { @@ -56,18 +57,17 @@ export const MissionPage = () => { }, [connectionReady]) useEffect(() => { - const updateVideoStreams = (mission: Mission) => - BackendAPICaller.getVideoStreamsByRobotId(mission.robot.id) - .then((streams) => setVideoStreams(streams)) - .catch((e) => { - console.warn(`Failed to get video stream with robot ID ${mission.robot.id}`) - }) + if (selectedMission && Object(mediaStreams).keys.includes(selectedMission?.robot.id)) { + const mediaStreamConfig = mediaStreams[selectedMission.robot.id] + if (mediaStreamConfig.streams.length > 0) setVideoMediaStreams(mediaStreamConfig.streams) + } + }, [selectedMission, mediaStreams]) + useEffect(() => { if (missionId) BackendAPICaller.getMissionRunById(missionId) .then((mission) => { setSelectedMission(mission) - updateVideoStreams(mission) }) .catch((e) => { setAlert( @@ -101,7 +101,7 @@ export const MissionPage = () => { - {videoStreams.length > 0 && } + {videoMediaStreams.length > 0 && } )} diff --git a/frontend/src/components/Pages/MissionPage/VideoStream/FullScreenVideo.tsx b/frontend/src/components/Pages/MissionPage/VideoStream/FullScreenVideo.tsx index bda3442ec..caf206f32 100644 --- a/frontend/src/components/Pages/MissionPage/VideoStream/FullScreenVideo.tsx +++ b/frontend/src/components/Pages/MissionPage/VideoStream/FullScreenVideo.tsx @@ -1,10 +1,8 @@ -import { VideoPlayerOvenPlayer, isValidOvenPlayerType } from './VideoPlayerOvenPlayer' -import { VideoPlayerSimple } from './VideoPlayerSimple' -import { VideoStream } from 'models/VideoStream' import styled from 'styled-components' import { Typography, Button, Icon } from '@equinor/eds-core-react' import { tokens } from '@equinor/eds-tokens' import { Icons } from 'utils/icons' +import { VideoPlayerSimpleStream } from './VideoPlayerSimpleStream' const FullscreenExitButton = styled(Button)` position: absolute; @@ -28,26 +26,17 @@ const FullScreenCard = styled.div` padding: 1rem; ` -// Styles for rotation -const FullScreenCardRotated = styled.div` - transform: rotate(270deg); - padding: 1rem; -` -const RotateText = styled.div` - writing-mode: vertical-rl; -` - -const PositionText = styled.div` - display: flex; - flex-direction: row-reverse; -` - interface IFullScreenVideoStreamCardProps { - videoStream: VideoStream + videoStream: MediaStream + videoStreamName: string toggleFullScreenMode: VoidFunction } -export const FullScreenVideoStreamCard = ({ videoStream, toggleFullScreenMode }: IFullScreenVideoStreamCardProps) => { +export const FullScreenVideoStreamCard = ({ + videoStream, + videoStreamName, + toggleFullScreenMode, +}: IFullScreenVideoStreamCardProps) => { const cardWidth = () => { const availableInnerHeight = window.innerHeight - 9 * 16 const availableInnerWidth = window.innerWidth - 2 * 16 @@ -57,15 +46,6 @@ export const FullScreenVideoStreamCard = ({ videoStream, toggleFullScreenMode }: Math.min(coverageFactor * availableInnerWidth, aspectRatio * coverageFactor * availableInnerHeight) ) } - const rotatedCardWidth = () => { - const availableInnerHeight = window.innerHeight - 7.5 * 16 - const availableInnerWidth = window.innerWidth + 0.5 * 16 - const coverageFactor = 0.9 - const aspectRatio = 9 / 16 - return Math.round( - Math.min(coverageFactor * availableInnerHeight, aspectRatio * coverageFactor * availableInnerWidth) - ) - } const fullScreenExitButton = (shouldRotate270Clockwise: boolean) => { if (shouldRotate270Clockwise) { @@ -82,33 +62,11 @@ export const FullScreenVideoStreamCard = ({ videoStream, toggleFullScreenMode }: ) } - if (isValidOvenPlayerType(videoStream)) { - if (videoStream.shouldRotate270Clockwise) { - return ( - - - - {videoStream.name} - - - - {fullScreenExitButton(true)} - - ) - } - return ( - - {videoStream.name} - - {fullScreenExitButton(false)} - - ) - } // Rotated stream is not supported for simpleplayer return ( - {videoStream.name} - + {videoStreamName} + {fullScreenExitButton(false)} ) diff --git a/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerOvenPlayer.tsx b/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerOvenPlayer.tsx deleted file mode 100644 index 9c5fccb18..000000000 --- a/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerOvenPlayer.tsx +++ /dev/null @@ -1,47 +0,0 @@ -import { useEffect } from 'react' - -// import OvenPlayer from 'ovenplayer' - -// Styles -import 'video.js/dist/video-js.css' -import { VideoStream } from 'models/VideoStream' - -interface IVideoPlayerProps { - videoStream: VideoStream -} - -// TODO: Video player is not used at the moment, commented out for now -export const VideoPlayerOvenPlayer = ({ videoStream }: IVideoPlayerProps) => { - useEffect(() => { - // const aspectRatio = videoStream.shouldRotate270Clockwise ? '9:16' : '16:9' - switch (videoStream.type) { - case 'webrtc': - case 'hls': - case 'llhls': - case 'dash': - case 'lldash': - case 'mp4': - // const player = OvenPlayer.create(videoStream.id, { - // aspectRatio: aspectRatio, - // controls: false, - // mute: true, - // autoStart: true, - // expandFullScreenUI: false, - // sources: [ - // { - // label: videoStream.name, - // type: videoStream.type, - // file: videoStream.url, - // }, - // ], - // }) - } - }, [videoStream.id, videoStream.name, videoStream.shouldRotate270Clockwise, videoStream.type, videoStream.url]) - - return
-} - -export const isValidOvenPlayerType = (videoStream: VideoStream) => { - const validTypes = ['webrtc', 'hls', 'llhls', 'dash', 'lldash', 'mp4'] - return validTypes.includes(videoStream.type) -} diff --git a/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerSimple.tsx b/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerSimple.tsx deleted file mode 100644 index 8da14bfdb..000000000 --- a/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerSimple.tsx +++ /dev/null @@ -1,9 +0,0 @@ -import { VideoStream } from 'models/VideoStream' - -interface IVideoPlayerProps { - videoStream: VideoStream -} - -export const VideoPlayerSimple = ({ videoStream }: IVideoPlayerProps) => { - return {videoStream.name -} diff --git a/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerSimpleStream.tsx b/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerSimpleStream.tsx new file mode 100644 index 000000000..62cc41c4c --- /dev/null +++ b/frontend/src/components/Pages/MissionPage/VideoStream/VideoPlayerSimpleStream.tsx @@ -0,0 +1,10 @@ +import ReactPlayer from 'react-player/lazy' + +interface IVideoPlayerProps { + videoStream: MediaStream + videoStreamName: string +} + +export const VideoPlayerSimpleStream = ({ videoStream, videoStreamName }: IVideoPlayerProps) => ( + +) diff --git a/frontend/src/components/Pages/MissionPage/VideoStream/VideoStreamCards.tsx b/frontend/src/components/Pages/MissionPage/VideoStream/VideoStreamCards.tsx index 02d9d7d79..48bd9fa96 100644 --- a/frontend/src/components/Pages/MissionPage/VideoStream/VideoStreamCards.tsx +++ b/frontend/src/components/Pages/MissionPage/VideoStream/VideoStreamCards.tsx @@ -1,10 +1,8 @@ import { Card, Typography, Icon, Button } from '@equinor/eds-core-react' import { tokens } from '@equinor/eds-tokens' -import { VideoPlayerOvenPlayer, isValidOvenPlayerType } from './VideoPlayerOvenPlayer' -import { VideoPlayerSimple } from './VideoPlayerSimple' -import { VideoStream } from 'models/VideoStream' import styled from 'styled-components' import { Icons } from 'utils/icons' +import { VideoPlayerSimpleStream } from './VideoPlayerSimpleStream' const FullscreenButton = styled(Button)` position: absolute; @@ -25,25 +23,19 @@ const StyledVideoSection = styled.div` height: 10rem; ` -const StyledVideoSectionRotated = styled.div` - height: 10rem; - width: 10rem; -` - -const Rotate = styled.div` - transform: rotate(270deg); - position: relative; - left: 4rem; - bottom: 4rem; -` - interface IVideoStreamCardProps { - videoStream: VideoStream + videoStream: MediaStream + videoStreamName: string toggleFullScreenMode: VoidFunction setFullScreenStream: Function } -export const VideoStreamCard = ({ videoStream, toggleFullScreenMode, setFullScreenStream }: IVideoStreamCardProps) => { +export const VideoStreamCard = ({ + videoStream, + videoStreamName, + toggleFullScreenMode, + setFullScreenStream, +}: IVideoStreamCardProps) => { const turnOnFullScreen = () => { setFullScreenStream(videoStream) toggleFullScreenMode() @@ -55,40 +47,15 @@ export const VideoStreamCard = ({ videoStream, toggleFullScreenMode, setFullScre ) - const getVideoPlayer = () => { - if (isValidOvenPlayerType(videoStream)) { - if (videoStream.shouldRotate270Clockwise) { - return ( - - - - - {fullScreenButton} - - ) - } else { - return ( - - - {fullScreenButton} - - ) - } - } else { - // Rotated stream is not supported for simpleplayer - return ( + return ( + +
- + {fullScreenButton} - ) - } - } - - return ( - -
{getVideoPlayer()}
- {videoStream.name} +
+ {videoStreamName}
) } diff --git a/frontend/src/components/Pages/MissionPage/VideoStream/VideoStreamWindow.tsx b/frontend/src/components/Pages/MissionPage/VideoStream/VideoStreamWindow.tsx index aa7a5c96e..ed5bc9215 100644 --- a/frontend/src/components/Pages/MissionPage/VideoStream/VideoStreamWindow.tsx +++ b/frontend/src/components/Pages/MissionPage/VideoStream/VideoStreamWindow.tsx @@ -1,7 +1,6 @@ import { useState } from 'react' import { Typography } from '@equinor/eds-core-react' import { FullScreenVideoStreamCard } from './FullScreenVideo' -import { VideoStream } from 'models/VideoStream' import { VideoStreamCard } from './VideoStreamCards' import styled from 'styled-components' import ReactModal from 'react-modal' @@ -24,26 +23,29 @@ const VideoFullScreen = styled(ReactModal)` ` interface VideoStreamWindowProps { - videoStreams: VideoStream[] + videoStreams: MediaStreamTrack[] } export const VideoStreamWindow = ({ videoStreams }: VideoStreamWindowProps) => { const { TranslateText } = useLanguageContext() const [fullScreenMode, setFullScreenMode] = useState(false) - const [fullScreenStream, setFullScreenStream] = useState() + const [fullScreenStream, setFullScreenStream] = useState() const toggleFullScreenMode = () => { setFullScreenMode(!fullScreenMode) } - const updateFullScreenStream = (videoStream: VideoStream) => { + const updateFullScreenStream = (videoStream: MediaStream) => { setFullScreenStream(videoStream) toggleFullScreenMode() } + const videoStreamName = 'test' // TODO: decide if we even want a name for it + const videoCards = videoStreams.map((videoStream, index) => ( @@ -57,7 +59,11 @@ export const VideoStreamWindow = ({ videoStreams }: VideoStreamWindowProps) => { {fullScreenMode === false && videoCards} {videoStream && ( - {FullScreenVideoStreamCard({ videoStream, toggleFullScreenMode })} + )} diff --git a/frontend/src/models/Robot.ts b/frontend/src/models/Robot.ts index 2a28c15b2..2ba0830f2 100644 --- a/frontend/src/models/Robot.ts +++ b/frontend/src/models/Robot.ts @@ -3,7 +3,6 @@ import { BatteryStatus } from './Battery' import { Installation, placeholderInstallation } from './Installation' import { Pose } from './Pose' import { RobotModel, placeholderRobotModel } from './RobotModel' -import { VideoStream } from './VideoStream' export enum RobotStatus { Available = 'Available', @@ -37,7 +36,6 @@ export interface Robot { host?: string logs?: string port?: number - videoStreams?: VideoStream[] isarUri?: string currentArea?: Area flotillaStatus?: RobotFlotillaStatus diff --git a/frontend/src/models/VideoStream.ts b/frontend/src/models/VideoStream.ts index 88b441a21..e8fd0f67d 100644 --- a/frontend/src/models/VideoStream.ts +++ b/frontend/src/models/VideoStream.ts @@ -1,8 +1,19 @@ -export interface VideoStream { - id: string - name: string - robotId?: string +export enum MediaType { + Video, + Audio, +} + +export enum MediaConnectionType { + LiveKit, +} + +export type MediaStreamConfig = { url: string - type: string - shouldRotate270Clockwise: boolean + streamId: string + authToken: string + mediaType: MediaType + robotId: string + connectionType: MediaConnectionType } + +export type MediaStreamConfigAndTracks = MediaStreamConfig & { streams: MediaStreamTrack[] }