@100mslive/roomkit-react 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -0
- package/dist/Accordion/Accordion.d.ts +964 -0
- package/dist/Accordion/index.d.ts +979 -0
- package/dist/ActiveSpeakerView-H3VYXANB.js +39 -0
- package/dist/ActiveSpeakerView-H3VYXANB.js.map +7 -0
- package/dist/ActiveSpeakerView-REZLWPPI.css +11 -0
- package/dist/ActiveSpeakerView-REZLWPPI.css.map +7 -0
- package/dist/AudioLevel/AudioLevel.d.ts +8 -0
- package/dist/AudioLevel/index.d.ts +1 -0
- package/dist/Avatar/Avatar.d.ts +487 -0
- package/dist/Avatar/getAvatarBg.d.ts +7 -0
- package/dist/Avatar/index.d.ts +1 -0
- package/dist/Button/Button.d.ts +488 -0
- package/dist/Button/index.d.ts +1 -0
- package/dist/Checkbox/Checkbox.d.ts +958 -0
- package/dist/Checkbox/index.d.ts +1 -0
- package/dist/Collapsible/Collapsible.d.ts +1434 -0
- package/dist/Collapsible/index.d.ts +1 -0
- package/dist/Divider/Divider.d.ts +958 -0
- package/dist/Divider/index.d.ts +1 -0
- package/dist/Dropdown/Dropdown.d.ts +5728 -0
- package/dist/Dropdown/index.d.ts +1 -0
- package/dist/Fieldset/Fieldset.d.ts +477 -0
- package/dist/Fieldset/index.d.ts +1 -0
- package/dist/Footer/Footer.d.ts +3834 -0
- package/dist/Footer/index.d.ts +1 -0
- package/dist/HLSView-3RDXRV7Y.js +689 -0
- package/dist/HLSView-3RDXRV7Y.js.map +7 -0
- package/dist/HLSView-6BVBCQM7.css +11 -0
- package/dist/HLSView-6BVBCQM7.css.map +7 -0
- package/dist/IconButton/IconButton.d.ts +479 -0
- package/dist/IconButton/index.d.ts +1 -0
- package/dist/Input/Input.d.ts +2917 -0
- package/dist/Input/index.d.ts +1 -0
- package/dist/Label/Label.d.ts +479 -0
- package/dist/Label/index.d.ts +1 -0
- package/dist/Layout/Box.d.ts +477 -0
- package/dist/Layout/Flex.d.ts +482 -0
- package/dist/Layout/index.d.ts +2 -0
- package/dist/Link/Link.d.ts +489 -0
- package/dist/Link/index.d.ts +2 -0
- package/dist/Loading/Loading.d.ts +14 -0
- package/dist/Loading/index.d.ts +1 -0
- package/dist/Modal/Dialog.d.ts +3826 -0
- package/dist/Modal/DialogContent.d.ts +3343 -0
- package/dist/Modal/index.d.ts +1 -0
- package/dist/Pagination/StyledPagination.d.ts +1918 -0
- package/dist/Pagination/index.d.ts +1 -0
- package/dist/PinnedTrackView-453PELNU.js +70 -0
- package/dist/PinnedTrackView-453PELNU.js.map +7 -0
- package/dist/PinnedTrackView-QQ5FDXJX.css +11 -0
- package/dist/PinnedTrackView-QQ5FDXJX.css.map +7 -0
- package/dist/Popover/index.d.ts +1436 -0
- package/dist/Progress/index.d.ts +957 -0
- package/dist/QRCode/QRCode.d.ts +3 -0
- package/dist/QRCode/index.d.ts +1 -0
- package/dist/RadioGroup/RadioGroup.d.ts +1435 -0
- package/dist/RadioGroup/index.d.ts +1 -0
- package/dist/ReactSelect/ReactSelect.d.ts +4778 -0
- package/dist/ReactSelect/index.d.ts +1 -0
- package/dist/Select/Select.d.ts +1437 -0
- package/dist/Select/index.d.ts +1 -0
- package/dist/Slider/Slider.d.ts +488 -0
- package/dist/Slider/index.d.ts +1 -0
- package/dist/Stats/Stats.d.ts +19 -0
- package/dist/Stats/StyledStats.d.ts +2872 -0
- package/dist/Stats/formatBytes.d.ts +1 -0
- package/dist/Stats/index.d.ts +1 -0
- package/dist/Switch/Switch.d.ts +482 -0
- package/dist/Switch/index.d.ts +1 -0
- package/dist/Tabs/Tabs.d.ts +1912 -0
- package/dist/Tabs/index.d.ts +1 -0
- package/dist/Text/Text.d.ts +605 -0
- package/dist/Text/index.d.ts +1 -0
- package/dist/Theme/ThemeProvider.d.ts +397 -0
- package/dist/Theme/base.config.d.ts +417 -0
- package/dist/Theme/index.d.ts +3 -0
- package/dist/Theme/stitches.config.d.ts +4702 -0
- package/dist/Theme/themes.d.ts +64 -0
- package/dist/Theme/useSSR.d.ts +6 -0
- package/dist/TileMenu/StyledMenuTile.d.ts +3351 -0
- package/dist/TileMenu/TileMenu.d.ts +6 -0
- package/dist/TileMenu/index.d.ts +1 -0
- package/dist/Toast/Toast.d.ts +2881 -0
- package/dist/Toast/index.d.ts +1 -0
- package/dist/Tooltip/Tooltip.d.ts +14 -0
- package/dist/Tooltip/index.d.ts +1 -0
- package/dist/Video/Video.d.ts +499 -0
- package/dist/Video/index.d.ts +1 -0
- package/dist/VideoList/StyledVideoList.d.ts +1438 -0
- package/dist/VideoList/index.d.ts +2 -0
- package/dist/VideoList/videoListUtils.d.ts +1 -0
- package/dist/VideoTile/StyledVideoTile.d.ts +4309 -0
- package/dist/VideoTile/index.d.ts +1 -0
- package/dist/VirtualBackground-LHYBWUT5.js +158 -0
- package/dist/VirtualBackground-LHYBWUT5.js.map +7 -0
- package/dist/chunk-7YUYZ64D.js +6843 -0
- package/dist/chunk-7YUYZ64D.js.map +7 -0
- package/dist/chunk-HCAGFNXW.js +8270 -0
- package/dist/chunk-HCAGFNXW.js.map +7 -0
- package/dist/chunk-KYYP6ZVK.js +907 -0
- package/dist/chunk-KYYP6ZVK.js.map +7 -0
- package/dist/chunk-ULXGBUSC.js +65 -0
- package/dist/chunk-ULXGBUSC.js.map +7 -0
- package/dist/chunk-XRJXE6UO.js +243 -0
- package/dist/chunk-XRJXE6UO.js.map +7 -0
- package/dist/conference-IDNRO4WK.js +3812 -0
- package/dist/conference-IDNRO4WK.js.map +7 -0
- package/dist/conference-KN6OKGDU.css +11 -0
- package/dist/conference-KN6OKGDU.css.map +7 -0
- package/dist/fixtures/chats.d.ts +4 -0
- package/dist/fixtures/peers.d.ts +3 -0
- package/dist/fixtures/tracks.d.ts +2 -0
- package/dist/index.cjs.css +11 -0
- package/dist/index.cjs.css.map +7 -0
- package/dist/index.cjs.js +22987 -0
- package/dist/index.cjs.js.map +7 -0
- package/dist/index.css +11 -0
- package/dist/index.css.map +7 -0
- package/dist/index.d.ts +38 -0
- package/dist/index.js +133 -0
- package/dist/index.js.map +7 -0
- package/dist/meta.cjs.json +13085 -0
- package/dist/meta.esbuild.json +13693 -0
- package/dist/transcription-BTSB7FZH.js +356 -0
- package/dist/transcription-BTSB7FZH.js.map +7 -0
- package/dist/utils/animations.d.ts +92 -0
- package/dist/utils/index.d.ts +2 -0
- package/dist/utils/styles.d.ts +21 -0
- package/package.json +116 -0
- package/src/Accordion/Accordion.stories.tsx +50 -0
- package/src/Accordion/Accordion.tsx +88 -0
- package/src/Accordion/index.ts +8 -0
- package/src/AudioLevel/AudioLevel.tsx +34 -0
- package/src/AudioLevel/index.ts +1 -0
- package/src/Avatar/Avatar.stories.tsx +33 -0
- package/src/Avatar/Avatar.tsx +55 -0
- package/src/Avatar/getAvatarBg.ts +46 -0
- package/src/Avatar/index.ts +1 -0
- package/src/Button/Button.mdx +43 -0
- package/src/Button/Button.stories.tsx +52 -0
- package/src/Button/Button.tsx +144 -0
- package/src/Button/index.tsx +1 -0
- package/src/Chat/Chat.mdx +39 -0
- package/src/Chat/Chat.stories.tsx +39 -0
- package/src/Checkbox/Checkbox.stories.tsx +61 -0
- package/src/Checkbox/Checkbox.tsx +33 -0
- package/src/Checkbox/index.tsx +1 -0
- package/src/Collapsible/Collapsible.tsx +34 -0
- package/src/Collapsible/index.tsx +1 -0
- package/src/Divider/Divider.tsx +45 -0
- package/src/Divider/HorizontalDivider.stories.tsx +34 -0
- package/src/Divider/VerticalDivider.stories.tsx +40 -0
- package/src/Divider/index.ts +1 -0
- package/src/Dropdown/Dropdown.stories.tsx +94 -0
- package/src/Dropdown/Dropdown.tsx +142 -0
- package/src/Dropdown/index.tsx +1 -0
- package/src/Fieldset/Fieldset.stories.tsx +29 -0
- package/src/Fieldset/Fieldset.tsx +11 -0
- package/src/Fieldset/index.tsx +1 -0
- package/src/Footer/Footer.stories.tsx +61 -0
- package/src/Footer/Footer.tsx +47 -0
- package/src/Footer/index.tsx +1 -0
- package/src/IconButton/IconButton.tsx +45 -0
- package/src/IconButton/index.tsx +1 -0
- package/src/Icons/Icons.stories.mdx +10 -0
- package/src/Icons/IconsList.jsx +17 -0
- package/src/Input/Input.stories.tsx +25 -0
- package/src/Input/Input.tsx +105 -0
- package/src/Input/PasswordInput.stories.tsx +53 -0
- package/src/Input/index.tsx +1 -0
- package/src/Introduction/Integrating.stories.mdx +100 -0
- package/src/Introduction/Introduction.stories.mdx +9 -0
- package/src/Label/Label.tsx +8 -0
- package/src/Label/index.ts +1 -0
- package/src/Layout/Box.tsx +3 -0
- package/src/Layout/Flex.tsx +76 -0
- package/src/Layout/index.tsx +2 -0
- package/src/Link/Link.stories.tsx +18 -0
- package/src/Link/Link.tsx +54 -0
- package/src/Link/index.tsx +2 -0
- package/src/Loading/Loading.mdx +15 -0
- package/src/Loading/Loading.stories.tsx +37 -0
- package/src/Loading/Loading.tsx +33 -0
- package/src/Loading/index.ts +1 -0
- package/src/Modal/Dialog.mdx +19 -0
- package/src/Modal/Dialog.stories.tsx +68 -0
- package/src/Modal/Dialog.tsx +26 -0
- package/src/Modal/DialogContent.tsx +63 -0
- package/src/Modal/index.ts +1 -0
- package/src/Pagination/StyledPagination.stories.tsx +80 -0
- package/src/Pagination/StyledPagination.tsx +68 -0
- package/src/Pagination/index.tsx +1 -0
- package/src/Popover/Popover.mdx +9 -0
- package/src/Popover/Popover.stories.tsx +95 -0
- package/src/Popover/index.tsx +26 -0
- package/src/Prebuilt/App.jsx +273 -0
- package/src/Prebuilt/AppContext.jsx +21 -0
- package/src/Prebuilt/IconButton.jsx +19 -0
- package/src/Prebuilt/Prebuilt.stories.tsx +20 -0
- package/src/Prebuilt/common/PeersSorter.js +89 -0
- package/src/Prebuilt/common/constants.js +208 -0
- package/src/Prebuilt/common/hooks.js +47 -0
- package/src/Prebuilt/common/roles.js +4 -0
- package/src/Prebuilt/common/useSortedPeers.js +28 -0
- package/src/Prebuilt/common/utils.js +91 -0
- package/src/Prebuilt/components/AppData/AppData.jsx +189 -0
- package/src/Prebuilt/components/AppData/useAppConfig.js +7 -0
- package/src/Prebuilt/components/AppData/useAppLayout.js +6 -0
- package/src/Prebuilt/components/AppData/useChatState.js +18 -0
- package/src/Prebuilt/components/AppData/useSidepane.js +49 -0
- package/src/Prebuilt/components/AppData/useUISettings.js +164 -0
- package/src/Prebuilt/components/AudioLevel/BeamSpeakerLabelsLogging.jsx +16 -0
- package/src/Prebuilt/components/AudioVideoToggle.jsx +67 -0
- package/src/Prebuilt/components/AuthToken.jsx +133 -0
- package/src/Prebuilt/components/BottomActionSheet/BottomActionSheet.jsx +96 -0
- package/src/Prebuilt/components/BottomActionSheet/BottomActionSheet.stories.tsx +46 -0
- package/src/Prebuilt/components/Chat/Chat.jsx +155 -0
- package/src/Prebuilt/components/Chat/ChatBody.jsx +370 -0
- package/src/Prebuilt/components/Chat/ChatFooter.jsx +150 -0
- package/src/Prebuilt/components/Chat/ChatHeader.jsx +67 -0
- package/src/Prebuilt/components/Chat/ChatSelector.jsx +162 -0
- package/src/Prebuilt/components/Chat/useEmojiPickerStyles.js +30 -0
- package/src/Prebuilt/components/Chat/useUnreadCount.js +17 -0
- package/src/Prebuilt/components/Connection/ConnectionIndicator.jsx +80 -0
- package/src/Prebuilt/components/Connection/TileConnection.jsx +40 -0
- package/src/Prebuilt/components/Connection/connectionQualityUtils.js +39 -0
- package/src/Prebuilt/components/EmojiReaction.jsx +138 -0
- package/src/Prebuilt/components/ErrorBoundary.jsx +105 -0
- package/src/Prebuilt/components/FirstPersonDisplay.jsx +50 -0
- package/src/Prebuilt/components/Footer/ChatToggle.jsx +27 -0
- package/src/Prebuilt/components/Footer/ConferencingFooter.jsx +104 -0
- package/src/Prebuilt/components/Footer/StreamingFooter.jsx +71 -0
- package/src/Prebuilt/components/Footer.jsx +8 -0
- package/src/Prebuilt/components/FullPageProgress.jsx +11 -0
- package/src/Prebuilt/components/GoLiveButton.jsx +45 -0
- package/src/Prebuilt/components/HMSVideo/Controls.jsx +21 -0
- package/src/Prebuilt/components/HMSVideo/FullscreenButton.jsx +18 -0
- package/src/Prebuilt/components/HMSVideo/HLSAutoplayBlockedPrompt.jsx +35 -0
- package/src/Prebuilt/components/HMSVideo/HLSQualitySelector.jsx +82 -0
- package/src/Prebuilt/components/HMSVideo/HMSVIdeoUtils.js +27 -0
- package/src/Prebuilt/components/HMSVideo/HMSVideo.jsx +11 -0
- package/src/Prebuilt/components/HMSVideo/PlayButton.jsx +13 -0
- package/src/Prebuilt/components/HMSVideo/VideoProgress.jsx +77 -0
- package/src/Prebuilt/components/HMSVideo/VideoTime.jsx +31 -0
- package/src/Prebuilt/components/HMSVideo/VolumeControl.jsx +39 -0
- package/src/Prebuilt/components/HMSVideo/index.js +19 -0
- package/src/Prebuilt/components/Header/AdditionalRoomState.jsx +217 -0
- package/src/Prebuilt/components/Header/AmbientMusic.jsx +88 -0
- package/src/Prebuilt/components/Header/ConferencingHeader.jsx +29 -0
- package/src/Prebuilt/components/Header/Header.jsx +8 -0
- package/src/Prebuilt/components/Header/HeaderComponents.jsx +41 -0
- package/src/Prebuilt/components/Header/ParticipantFilter.jsx +91 -0
- package/src/Prebuilt/components/Header/ParticipantList.jsx +337 -0
- package/src/Prebuilt/components/Header/StreamActions.jsx +225 -0
- package/src/Prebuilt/components/Header/StreamingHeader.jsx +55 -0
- package/src/Prebuilt/components/Header/index.jsx +1 -0
- package/src/Prebuilt/components/HlsStatsOverlay.jsx +101 -0
- package/src/Prebuilt/components/Image.jsx +7 -0
- package/src/Prebuilt/components/Input/KeyboardInputManager.js +107 -0
- package/src/Prebuilt/components/LeaveRoom.jsx +202 -0
- package/src/Prebuilt/components/MetaActions.jsx +45 -0
- package/src/Prebuilt/components/MoreSettings/BulkRoleChangeModal.jsx +139 -0
- package/src/Prebuilt/components/MoreSettings/ChangeNameModal.jsx +92 -0
- package/src/Prebuilt/components/MoreSettings/ChangeSelfRole.jsx +67 -0
- package/src/Prebuilt/components/MoreSettings/EmbedUrl.jsx +106 -0
- package/src/Prebuilt/components/MoreSettings/FullScreenItem.jsx +29 -0
- package/src/Prebuilt/components/MoreSettings/MoreSettings.jsx +225 -0
- package/src/Prebuilt/components/MoreSettings/MuteAllModal.jsx +90 -0
- package/src/Prebuilt/components/Notifications/AutoplayBlockedModal.jsx +38 -0
- package/src/Prebuilt/components/Notifications/InitErrorModal.jsx +39 -0
- package/src/Prebuilt/components/Notifications/MessageNotifications.jsx +25 -0
- package/src/Prebuilt/components/Notifications/Notifications.jsx +151 -0
- package/src/Prebuilt/components/Notifications/PeerNotifications.jsx +45 -0
- package/src/Prebuilt/components/Notifications/PermissionErrorModal.jsx +67 -0
- package/src/Prebuilt/components/Notifications/ReconnectNotifications.jsx +69 -0
- package/src/Prebuilt/components/Notifications/TrackBulkUnmuteModal.jsx +51 -0
- package/src/Prebuilt/components/Notifications/TrackNotifications.jsx +19 -0
- package/src/Prebuilt/components/Notifications/TrackUnmuteModal.jsx +49 -0
- package/src/Prebuilt/components/Notifications/index.jsx +1 -0
- package/src/Prebuilt/components/PIP/PIPComponent.jsx +82 -0
- package/src/Prebuilt/components/PIP/PIPManager.js +296 -0
- package/src/Prebuilt/components/PIP/SetupMediaSession.js +60 -0
- package/src/Prebuilt/components/PIP/index.jsx +11 -0
- package/src/Prebuilt/components/PIP/pip.test.js +72 -0
- package/src/Prebuilt/components/PIP/pipUtils.js +183 -0
- package/src/Prebuilt/components/Pagination.jsx +29 -0
- package/src/Prebuilt/components/Playlist/Playlist.jsx +129 -0
- package/src/Prebuilt/components/Playlist/PlaylistControls.jsx +172 -0
- package/src/Prebuilt/components/Playlist/PlaylistItem.jsx +51 -0
- package/src/Prebuilt/components/Playlist/VideoPlayer.jsx +95 -0
- package/src/Prebuilt/components/PostLeave.jsx +76 -0
- package/src/Prebuilt/components/Preview/PreviewContainer.jsx +51 -0
- package/src/Prebuilt/components/Preview/PreviewJoin.jsx +196 -0
- package/src/Prebuilt/components/Preview/PreviewName.jsx +37 -0
- package/src/Prebuilt/components/RoleChangeModal.jsx +185 -0
- package/src/Prebuilt/components/RoleChangeRequestModal.jsx +26 -0
- package/src/Prebuilt/components/ScreenShare.jsx +45 -0
- package/src/Prebuilt/components/ScreenshareDisplay.jsx +45 -0
- package/src/Prebuilt/components/ScreenshareHintModal.jsx +37 -0
- package/src/Prebuilt/components/ScreenshareTile.jsx +91 -0
- package/src/Prebuilt/components/Settings/DeviceSettings.jsx +220 -0
- package/src/Prebuilt/components/Settings/LayoutSettings.jsx +91 -0
- package/src/Prebuilt/components/Settings/NotificationSettings.jsx +61 -0
- package/src/Prebuilt/components/Settings/SettingsModal.jsx +171 -0
- package/src/Prebuilt/components/Settings/StartRecording.jsx +130 -0
- package/src/Prebuilt/components/Settings/SwitchWithLabel.jsx +40 -0
- package/src/Prebuilt/components/Settings/common.js +15 -0
- package/src/Prebuilt/components/ShareMenuIcon.jsx +26 -0
- package/src/Prebuilt/components/StatsForNerds.jsx +250 -0
- package/src/Prebuilt/components/Streaming/Common.jsx +132 -0
- package/src/Prebuilt/components/Streaming/HLSStreaming.jsx +226 -0
- package/src/Prebuilt/components/Streaming/RTMPIcon.jsx +24 -0
- package/src/Prebuilt/components/Streaming/RTMPStreaming.jsx +336 -0
- package/src/Prebuilt/components/Streaming/ResolutionInput.jsx +88 -0
- package/src/Prebuilt/components/Streaming/StreamingLanding.jsx +76 -0
- package/src/Prebuilt/components/TileMenu.jsx +275 -0
- package/src/Prebuilt/components/Toast/Toast.jsx +17 -0
- package/src/Prebuilt/components/Toast/ToastBatcher.js +57 -0
- package/src/Prebuilt/components/Toast/ToastConfig.jsx +134 -0
- package/src/Prebuilt/components/Toast/ToastContainer.jsx +30 -0
- package/src/Prebuilt/components/Toast/ToastManager.js +44 -0
- package/src/Prebuilt/components/VideoList.jsx +101 -0
- package/src/Prebuilt/components/VideoTile.jsx +177 -0
- package/src/Prebuilt/components/conference.jsx +164 -0
- package/src/Prebuilt/components/gridView.jsx +85 -0
- package/src/Prebuilt/components/hooks/useDropdownList.jsx +23 -0
- package/src/Prebuilt/components/hooks/useDropdownSelection.jsx +6 -0
- package/src/Prebuilt/components/hooks/useFeatures.js +22 -0
- package/src/Prebuilt/components/hooks/useFullscreen.js +43 -0
- package/src/Prebuilt/components/hooks/useMetadata.jsx +52 -0
- package/src/Prebuilt/components/hooks/useNavigation.js +19 -0
- package/src/Prebuilt/components/hooks/usePlaylist.js +25 -0
- package/src/Prebuilt/components/hooks/usePlaylistMusic.js +35 -0
- package/src/Prebuilt/components/hooks/useScreenshareAudio.js +28 -0
- package/src/Prebuilt/components/hooks/useSetPinnedMessage.js +38 -0
- package/src/Prebuilt/components/hooks/useSkipPreview.jsx +20 -0
- package/src/Prebuilt/components/hooks/useUserPreferences.jsx +25 -0
- package/src/Prebuilt/components/init/Init.jsx +58 -0
- package/src/Prebuilt/components/init/initUtils.js +90 -0
- package/src/Prebuilt/components/pdfAnnotator/pdfErrorView.jsx +29 -0
- package/src/Prebuilt/components/pdfAnnotator/pdfFileOptions.jsx +108 -0
- package/src/Prebuilt/components/pdfAnnotator/pdfHeader.jsx +31 -0
- package/src/Prebuilt/components/pdfAnnotator/pdfInfo.jsx +32 -0
- package/src/Prebuilt/components/pdfAnnotator/shareScreenOptions.jsx +233 -0
- package/src/Prebuilt/components/pdfAnnotator/submitPdf.jsx +89 -0
- package/src/Prebuilt/components/pdfAnnotator/uploadedFile.jsx +85 -0
- package/src/Prebuilt/components/peerTileUtils.jsx +27 -0
- package/src/Prebuilt/hms.js +7 -0
- package/src/Prebuilt/images/first_person.png +0 -0
- package/src/Prebuilt/index.d.ts +15 -0
- package/src/Prebuilt/index.js +2 -0
- package/src/Prebuilt/layouts/ActiveSpeakerView.jsx +34 -0
- package/src/Prebuilt/layouts/EmbedView.jsx +141 -0
- package/src/Prebuilt/layouts/HLSView.jsx +290 -0
- package/src/Prebuilt/layouts/InsetView.jsx +222 -0
- package/src/Prebuilt/layouts/NonPublisherView.jsx +51 -0
- package/src/Prebuilt/layouts/PDFView.jsx +122 -0
- package/src/Prebuilt/layouts/PinnedTrackView.jsx +59 -0
- package/src/Prebuilt/layouts/SidePane.jsx +51 -0
- package/src/Prebuilt/layouts/WaitingView.jsx +51 -0
- package/src/Prebuilt/layouts/WhiteboardView.jsx +66 -0
- package/src/Prebuilt/layouts/mainGridView.jsx +98 -0
- package/src/Prebuilt/layouts/mainView.jsx +113 -0
- package/src/Prebuilt/layouts/screenShareView.jsx +185 -0
- package/src/Prebuilt/plugins/FlyingEmoji.jsx +132 -0
- package/src/Prebuilt/plugins/RemoteStopScreenshare.jsx +18 -0
- package/src/Prebuilt/plugins/VirtualBackground/VirtualBackground.jsx +90 -0
- package/src/Prebuilt/plugins/VirtualBackground/vbutils.js +66 -0
- package/src/Prebuilt/plugins/confetti.jsx +60 -0
- package/src/Prebuilt/plugins/transcription/Transcriber.js +216 -0
- package/src/Prebuilt/plugins/transcription/TranscriptionButton.jsx +138 -0
- package/src/Prebuilt/plugins/transcription/index.jsx +1 -0
- package/src/Prebuilt/plugins/whiteboard/PusherCommunicationProvider.js +110 -0
- package/src/Prebuilt/plugins/whiteboard/README.md +29 -0
- package/src/Prebuilt/plugins/whiteboard/ToggleWhiteboard.jsx +43 -0
- package/src/Prebuilt/plugins/whiteboard/Whiteboard.css +12 -0
- package/src/Prebuilt/plugins/whiteboard/Whiteboard.jsx +11 -0
- package/src/Prebuilt/plugins/whiteboard/WhiteboardEvents.js +8 -0
- package/src/Prebuilt/plugins/whiteboard/index.js +3 -0
- package/src/Prebuilt/plugins/whiteboard/useMultiplayerState.js +212 -0
- package/src/Prebuilt/plugins/whiteboard/useWhiteboardMetadata.js +47 -0
- package/src/Prebuilt/primitives/DialogContent.jsx +280 -0
- package/src/Prebuilt/primitives/DropdownTrigger.jsx +46 -0
- package/src/Prebuilt/services/FeatureFlags.jsx +47 -0
- package/src/Prebuilt/services/tokenService.js +49 -0
- package/src/Progress/index.tsx +17 -0
- package/src/QRCode/QRCode.mdx +9 -0
- package/src/QRCode/QRCode.stories.tsx +29 -0
- package/src/QRCode/QRCode.tsx +6 -0
- package/src/QRCode/index.tsx +1 -0
- package/src/RadioGroup/RadioGroup.stories.tsx +32 -0
- package/src/RadioGroup/RadioGroup.tsx +33 -0
- package/src/RadioGroup/index.tsx +1 -0
- package/src/ReactSelect/ReactSelect.stories.tsx +83 -0
- package/src/ReactSelect/ReactSelect.tsx +97 -0
- package/src/ReactSelect/index.ts +1 -0
- package/src/Select/Select.stories.tsx +33 -0
- package/src/Select/Select.tsx +63 -0
- package/src/Select/index.ts +1 -0
- package/src/Slider/Slider.stories.tsx +21 -0
- package/src/Slider/Slider.tsx +70 -0
- package/src/Slider/index.ts +1 -0
- package/src/Stats/Stats.tsx +211 -0
- package/src/Stats/StyledStats.tsx +57 -0
- package/src/Stats/formatBytes.ts +19 -0
- package/src/Stats/index.tsx +1 -0
- package/src/Switch/Switch.mdx +11 -0
- package/src/Switch/Switch.stories.tsx +46 -0
- package/src/Switch/Switch.tsx +52 -0
- package/src/Switch/index.ts +1 -0
- package/src/Tabs/Tabs.stories.tsx +77 -0
- package/src/Tabs/Tabs.tsx +41 -0
- package/src/Tabs/index.tsx +1 -0
- package/src/Text/Text.stories.tsx +21 -0
- package/src/Text/Text.tsx +149 -0
- package/src/Text/index.tsx +1 -0
- package/src/Theme/Theme.stories.mdx +8 -0
- package/src/Theme/ThemeProvider.tsx +104 -0
- package/src/Theme/ThemeStory.jsx +58 -0
- package/src/Theme/base.config.ts +264 -0
- package/src/Theme/index.tsx +3 -0
- package/src/Theme/stitches.config.ts +100 -0
- package/src/Theme/themes.ts +70 -0
- package/src/Theme/useSSR.tsx +24 -0
- package/src/TileMenu/StyledMenuTile.tsx +101 -0
- package/src/TileMenu/TileMenu.tsx +96 -0
- package/src/TileMenu/index.tsx +1 -0
- package/src/Toast/AppToast.stories.tsx +56 -0
- package/src/Toast/Toast.mdx +19 -0
- package/src/Toast/Toast.stories.tsx +57 -0
- package/src/Toast/Toast.tsx +168 -0
- package/src/Toast/index.tsx +1 -0
- package/src/Tooltip/Tooltip.stories.tsx +62 -0
- package/src/Tooltip/Tooltip.tsx +79 -0
- package/src/Tooltip/index.ts +1 -0
- package/src/Video/UseVideo.mdx +22 -0
- package/src/Video/UseVideo.stories.tsx +26 -0
- package/src/Video/Video.mdx +24 -0
- package/src/Video/Video.stories.tsx +27 -0
- package/src/Video/Video.tsx +61 -0
- package/src/Video/index.tsx +1 -0
- package/src/VideoList/StyledVideoList.tsx +39 -0
- package/src/VideoList/VideoList.stories.tsx +92 -0
- package/src/VideoList/index.tsx +2 -0
- package/src/VideoList/videoListUtils.tsx +20 -0
- package/src/VideoTile/StyledVideoTile.tsx +148 -0
- package/src/VideoTile/VideoTile.mdx +28 -0
- package/src/VideoTile/VideoTile.stories.tsx +32 -0
- package/src/VideoTile/index.tsx +1 -0
- package/src/fixtures/chats.ts +25 -0
- package/src/fixtures/peers.ts +24 -0
- package/src/fixtures/tracks.ts +11 -0
- package/src/index.ts +38 -0
- package/src/store/SetupFakeStore.ts +33 -0
- package/src/store/StorybookSDK.ts +229 -0
- package/src/utils/animations.ts +90 -0
- package/src/utils/index.ts +2 -0
- package/src/utils/styles.ts +22 -0
@@ -0,0 +1,296 @@
|
|
1
|
+
import * as workerTimers from 'worker-timers';
|
2
|
+
import { drawVideoElementsOnCanvas, dummyChangeInCanvas, resetPIPCanvasColors } from './pipUtils';
|
3
|
+
import { isIOS, isMacOS, isSafari } from '../../common/constants';
|
4
|
+
const MAX_NUMBER_OF_TILES_IN_PIP = 4;
|
5
|
+
const DEFAULT_FPS = 30;
|
6
|
+
const DEFAULT_CANVAS_WIDTH = 480;
|
7
|
+
const DEFAULT_CANVAS_HEIGHT = 320;
|
8
|
+
const LEAVE_EVENT_NAME = 'leavepictureinpicture';
|
9
|
+
|
10
|
+
const PIPStates = {
|
11
|
+
starting: 'starting',
|
12
|
+
started: 'started',
|
13
|
+
stopping: 'stopping',
|
14
|
+
stopped: 'stopped',
|
15
|
+
};
|
16
|
+
|
17
|
+
/**
|
18
|
+
* video elements are stitched together as a canvas which is converted to
|
19
|
+
* another video element converted to PIP.
|
20
|
+
* The task is split into two parts -
|
21
|
+
* 1. a function which gets hit when the input changes and updates the tracks
|
22
|
+
* to show
|
23
|
+
* 2. an independent loop which updates the canvas periodically based on the current
|
24
|
+
* tracks which should be shown.
|
25
|
+
*/
|
26
|
+
class PipManager {
|
27
|
+
constructor() {
|
28
|
+
this.reset();
|
29
|
+
}
|
30
|
+
|
31
|
+
/**
|
32
|
+
* @private
|
33
|
+
*/
|
34
|
+
reset() {
|
35
|
+
console.debug('resetting PIP state');
|
36
|
+
resetPIPCanvasColors();
|
37
|
+
this.canvas = null; // where stitching will take place
|
38
|
+
this.pipVideo = null; // the element which will be sent in PIP
|
39
|
+
this.timeoutRef = null; // setTimeout reference so it can be cancelled
|
40
|
+
this.hmsActions = null; // for attaching detaching
|
41
|
+
this.videoElements = []; // for attaching tracks
|
42
|
+
this.tracksToShow = [];
|
43
|
+
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
44
|
+
this.onStateChange = () => {}; // for user of this class to listen to changes
|
45
|
+
this.state = PIPStates.stopped;
|
46
|
+
}
|
47
|
+
|
48
|
+
/**
|
49
|
+
* check if PIP is supported in this browser env
|
50
|
+
*/
|
51
|
+
isSupported() {
|
52
|
+
return !!document.pictureInPictureEnabled && !isIOS && !(isMacOS && isSafari);
|
53
|
+
}
|
54
|
+
|
55
|
+
/**
|
56
|
+
* check if pip is currently turned on
|
57
|
+
*/
|
58
|
+
isOn() {
|
59
|
+
return !!document.pictureInPictureElement;
|
60
|
+
}
|
61
|
+
|
62
|
+
/**
|
63
|
+
* request browser to start pip and start the render loop updating the pip
|
64
|
+
* video element with peer tracks.
|
65
|
+
* @param hmsActions
|
66
|
+
* @param onStateChangeFn {function(bool):void} callback called to notify change in pip state
|
67
|
+
*/
|
68
|
+
async start(hmsActions, onStateChangeFn) {
|
69
|
+
if (!this.isSupported()) {
|
70
|
+
throw new Error('pip is not supported on this browser');
|
71
|
+
}
|
72
|
+
console.debug('starting PIP, current state', this.state);
|
73
|
+
if (this.state === PIPStates.started) {
|
74
|
+
await this.stop(); // if anything is already running
|
75
|
+
} else if (this.state === PIPStates.starting) {
|
76
|
+
return; // ignore double clicks
|
77
|
+
}
|
78
|
+
this.state = PIPStates.starting;
|
79
|
+
try {
|
80
|
+
await this.init(hmsActions, onStateChangeFn);
|
81
|
+
// when user closes pip, call internal stop
|
82
|
+
this.pipVideo.addEventListener(LEAVE_EVENT_NAME, this.stop);
|
83
|
+
this.renderLoop();
|
84
|
+
if (!this.isOn()) {
|
85
|
+
await this.requestPIP();
|
86
|
+
}
|
87
|
+
console.debug('pip started');
|
88
|
+
this.state = PIPStates.started;
|
89
|
+
this.onStateChange(true);
|
90
|
+
} catch (err) {
|
91
|
+
console.error('error in request pip', err);
|
92
|
+
this.state = PIPStates.stopped;
|
93
|
+
}
|
94
|
+
}
|
95
|
+
|
96
|
+
stop = async () => {
|
97
|
+
if (this.state === PIPStates.stopped) {
|
98
|
+
return;
|
99
|
+
}
|
100
|
+
this.state = PIPStates.stopping;
|
101
|
+
this.pipVideo?.removeEventListener(LEAVE_EVENT_NAME, this.stop);
|
102
|
+
if (this.timeoutRef) {
|
103
|
+
workerTimers.clearTimeout(this.timeoutRef);
|
104
|
+
}
|
105
|
+
if (this.isOn()) {
|
106
|
+
this.exitPIP();
|
107
|
+
}
|
108
|
+
// detach all to avoid bandwidth consumption
|
109
|
+
await this.detachOldAttachNewTracks(this.tracksToShow, []);
|
110
|
+
this.onStateChange(false); // notify parent about this
|
111
|
+
this.reset(); // cleanup
|
112
|
+
this.state = PIPStates.stopped;
|
113
|
+
};
|
114
|
+
|
115
|
+
/**
|
116
|
+
* @param peers {Array} All Remote Peers present in call.
|
117
|
+
* @param tracksMap {Object} map of track id to track
|
118
|
+
* */
|
119
|
+
async updatePeersAndTracks(peers, tracksMap) {
|
120
|
+
if (!this.canvas) {
|
121
|
+
return;
|
122
|
+
}
|
123
|
+
const newTracksToShowUnordered = this.pickTracksToShow(peers, tracksMap);
|
124
|
+
const currentTracksShowing = this.tracksToShow;
|
125
|
+
this.tracksToShow = this.orderNewTracksToShow(newTracksToShowUnordered, currentTracksShowing);
|
126
|
+
try {
|
127
|
+
await this.detachOldAttachNewTracks(currentTracksShowing, this.tracksToShow, tracksMap);
|
128
|
+
} catch (error) {
|
129
|
+
console.error('error in detaching/attaching tracks', error);
|
130
|
+
}
|
131
|
+
}
|
132
|
+
|
133
|
+
// ------- Private function --------------
|
134
|
+
|
135
|
+
/**
|
136
|
+
* @private
|
137
|
+
*/
|
138
|
+
async init(hmsActions, onStateChangeFn) {
|
139
|
+
await this.initMediaElements();
|
140
|
+
this.hmsActions = hmsActions;
|
141
|
+
this.onStateChange = onStateChangeFn;
|
142
|
+
}
|
143
|
+
|
144
|
+
async initMediaElements() {
|
145
|
+
if (!this.canvas) {
|
146
|
+
const { canvas, pipVideo } = this.initializeCanvasAndVideoElement();
|
147
|
+
this.canvas = canvas; // where stitching will take place
|
148
|
+
this.pipVideo = pipVideo; // the element which will be sent in PIP
|
149
|
+
this.videoElements = this.initializeVideoElements(); // for attaching tracks
|
150
|
+
const videoPlayPromise = this.pipVideo.play();
|
151
|
+
dummyChangeInCanvas(this.canvas);
|
152
|
+
await videoPlayPromise;
|
153
|
+
}
|
154
|
+
}
|
155
|
+
|
156
|
+
initializeCanvasAndVideoElement() {
|
157
|
+
const canvas = document.createElement('canvas');
|
158
|
+
canvas.width = DEFAULT_CANVAS_WIDTH;
|
159
|
+
canvas.height = DEFAULT_CANVAS_HEIGHT;
|
160
|
+
const pipVideo = document.createElement('video');
|
161
|
+
pipVideo.width = DEFAULT_CANVAS_WIDTH;
|
162
|
+
pipVideo.height = DEFAULT_CANVAS_HEIGHT;
|
163
|
+
pipVideo.muted = true;
|
164
|
+
pipVideo.srcObject = canvas.captureStream();
|
165
|
+
return { canvas, pipVideo };
|
166
|
+
}
|
167
|
+
|
168
|
+
initializeVideoElements() {
|
169
|
+
let videoElements = [];
|
170
|
+
for (let i = 0; i < MAX_NUMBER_OF_TILES_IN_PIP; i++) {
|
171
|
+
const videoElement = document.createElement('video');
|
172
|
+
videoElement.autoplay = true;
|
173
|
+
videoElement.playsinline = true;
|
174
|
+
videoElements.push(videoElement);
|
175
|
+
}
|
176
|
+
return videoElements;
|
177
|
+
}
|
178
|
+
|
179
|
+
/**
|
180
|
+
* render loop is responsible for rendering the video elements on canvas/pip.
|
181
|
+
* in each loop current video elements are stitched and painted on canvas
|
182
|
+
*/
|
183
|
+
renderLoop() {
|
184
|
+
const delay = 1000 / DEFAULT_FPS;
|
185
|
+
this.timeoutRef = workerTimers.setTimeout(() => {
|
186
|
+
if (this.state === PIPStates.stopping || this.state === PIPStates.stopped) {
|
187
|
+
return;
|
188
|
+
}
|
189
|
+
if (this.state === PIPStates.started) {
|
190
|
+
drawVideoElementsOnCanvas(this.videoElements, this.canvas);
|
191
|
+
}
|
192
|
+
this.renderLoop();
|
193
|
+
}, delay);
|
194
|
+
}
|
195
|
+
|
196
|
+
async requestPIP() {
|
197
|
+
try {
|
198
|
+
if (this.isOn()) {
|
199
|
+
this.exitPIP(); // is this really needed?
|
200
|
+
}
|
201
|
+
await this.pipVideo.requestPictureInPicture();
|
202
|
+
} catch (error) {
|
203
|
+
console.error('error in requestpip', error, 'state', this.state);
|
204
|
+
throw error;
|
205
|
+
}
|
206
|
+
}
|
207
|
+
|
208
|
+
exitPIP() {
|
209
|
+
document.exitPictureInPicture();
|
210
|
+
}
|
211
|
+
|
212
|
+
/**
|
213
|
+
* Logic - pick only enabled video tracks
|
214
|
+
* @param peers {Array<any>}
|
215
|
+
* @param tracksMap {Record<string, any>}
|
216
|
+
*/
|
217
|
+
pickTracksToShow(peers, tracksMap) {
|
218
|
+
const tracksToShow = [];
|
219
|
+
for (const peer of peers) {
|
220
|
+
if (tracksToShow.length === MAX_NUMBER_OF_TILES_IN_PIP) {
|
221
|
+
break;
|
222
|
+
} else if (peer.videoTrack && tracksMap[peer.videoTrack]?.enabled) {
|
223
|
+
tracksToShow.push(peer.videoTrack);
|
224
|
+
}
|
225
|
+
}
|
226
|
+
return tracksToShow;
|
227
|
+
}
|
228
|
+
|
229
|
+
/**
|
230
|
+
* there has to be a smart reordering of new tracks based on currently showing
|
231
|
+
* ones to reduce unnecessary displacement. If someone was showing up both
|
232
|
+
* earlier and now, it's a better UX to keep their position same instead
|
233
|
+
* of letting it change.
|
234
|
+
* The returned array is a shuffled version of newTracks with position of
|
235
|
+
* tracks present in the old tracks intact.
|
236
|
+
* eg. old = [1,5,9,3], new = [3,8,2,9], result = [8,2,9,3]
|
237
|
+
* @param oldTracks {Array}
|
238
|
+
* @param newTracks {Array}
|
239
|
+
* @return {Array}
|
240
|
+
*/
|
241
|
+
orderNewTracksToShow(newTracks, oldTracks) {
|
242
|
+
const betterNewTracks = [];
|
243
|
+
const leftOvers = [];
|
244
|
+
// put the common ones in right position
|
245
|
+
newTracks.forEach(track => {
|
246
|
+
const oldPosition = oldTracks.indexOf(track);
|
247
|
+
if (oldPosition !== -1 && oldPosition < newTracks.length) {
|
248
|
+
// if track is there currently and we can put it on the same position
|
249
|
+
betterNewTracks[oldPosition] = track;
|
250
|
+
} else {
|
251
|
+
leftOvers.push(track);
|
252
|
+
}
|
253
|
+
});
|
254
|
+
// put the left overs in remaining empty positions
|
255
|
+
for (let i = 0; i < newTracks.length; i++) {
|
256
|
+
if (!betterNewTracks[i]) {
|
257
|
+
betterNewTracks[i] = leftOvers.shift();
|
258
|
+
}
|
259
|
+
}
|
260
|
+
return betterNewTracks;
|
261
|
+
}
|
262
|
+
|
263
|
+
/**
|
264
|
+
* call detach for tracks which no longer need to be shown and attach for
|
265
|
+
* new ones which are to be shown now.
|
266
|
+
* Note: oldTracks and newTracks are not necessarily of same length
|
267
|
+
* @param oldTracks {Array<String>}
|
268
|
+
* @param newTracks {Array<String>}
|
269
|
+
* @param tracksMap {Record<String, any>}
|
270
|
+
*/
|
271
|
+
// eslint-disable-next-line complexity
|
272
|
+
async detachOldAttachNewTracks(oldTracks, newTracks, tracksMap = null) {
|
273
|
+
const numTracks = Math.max(oldTracks.length, newTracks.length);
|
274
|
+
for (let i = 0; i < numTracks; i++) {
|
275
|
+
if (oldTracks[i] === newTracks[i]) {
|
276
|
+
continue; // it would already have been attached previously
|
277
|
+
} else if (oldTracks[i]) {
|
278
|
+
// old track is there but not equal to new track, detach
|
279
|
+
// no need to call detach if we know for sure track is no longer in store
|
280
|
+
if (!tracksMap || tracksMap[oldTracks[i]]) {
|
281
|
+
await this.hmsActions.detachVideo(oldTracks[i], this.videoElements[i]);
|
282
|
+
}
|
283
|
+
if (this.videoElements[i]) {
|
284
|
+
// even if old track got removed from the room, element needs to be cleaned up
|
285
|
+
this.videoElements[i].srcObject = null;
|
286
|
+
}
|
287
|
+
}
|
288
|
+
if (newTracks[i]) {
|
289
|
+
await this.hmsActions.attachVideo(newTracks[i], this.videoElements[i]);
|
290
|
+
}
|
291
|
+
}
|
292
|
+
}
|
293
|
+
}
|
294
|
+
|
295
|
+
export const PictureInPicture = new PipManager();
|
296
|
+
// PictureInPicture.initMediaElements().catch(console.error); // for safari, init early
|
@@ -0,0 +1,60 @@
|
|
1
|
+
import { selectIsLocalAudioEnabled, selectIsLocalVideoEnabled } from '@100mslive/react-sdk';
|
2
|
+
|
3
|
+
/**
|
4
|
+
* Media Session API allows for handling control actions on top of pip
|
5
|
+
* https://web.dev/media-session/#video-conferencing-actions
|
6
|
+
*/
|
7
|
+
class SetupMediaSession {
|
8
|
+
setup = (actions, store) => {
|
9
|
+
this.actions = actions;
|
10
|
+
this.store = store;
|
11
|
+
this.initState();
|
12
|
+
this.setUpHandlers();
|
13
|
+
};
|
14
|
+
|
15
|
+
initState = () => {
|
16
|
+
const isMicActive = this.store.getState(selectIsLocalAudioEnabled);
|
17
|
+
const isCamActive = this.store.getState(selectIsLocalVideoEnabled);
|
18
|
+
navigator.mediaSession?.setMicrophoneActive?.(isMicActive);
|
19
|
+
navigator.mediaSession?.setCameraActive?.(isCamActive);
|
20
|
+
|
21
|
+
this.store.subscribe(isMicActive => {
|
22
|
+
navigator.mediaSession?.setMicrophoneActive?.(isMicActive);
|
23
|
+
}, selectIsLocalAudioEnabled);
|
24
|
+
|
25
|
+
this.store.subscribe(isCamActive => {
|
26
|
+
navigator.mediaSession?.setCameraActive?.(isCamActive);
|
27
|
+
}, selectIsLocalVideoEnabled);
|
28
|
+
};
|
29
|
+
|
30
|
+
toggleMic = async () => {
|
31
|
+
console.log('toggle mic clicked in pip');
|
32
|
+
const current = this.store.getState(selectIsLocalAudioEnabled);
|
33
|
+
await this.actions.setLocalAudioEnabled(!current);
|
34
|
+
};
|
35
|
+
|
36
|
+
toggleCam = async () => {
|
37
|
+
console.log('toggle cam clicked in pip');
|
38
|
+
const current = this.store.getState(selectIsLocalVideoEnabled);
|
39
|
+
await this.actions.setLocalVideoEnabled(!current);
|
40
|
+
};
|
41
|
+
|
42
|
+
leave = () => {
|
43
|
+
console.log('leave called from pip');
|
44
|
+
this.actions.leave();
|
45
|
+
};
|
46
|
+
|
47
|
+
setUpHandlers = () => {
|
48
|
+
if (navigator.mediaSession) {
|
49
|
+
try {
|
50
|
+
navigator.mediaSession.setActionHandler('togglemicrophone', this.toggleMic);
|
51
|
+
navigator.mediaSession.setActionHandler('togglecamera', this.toggleCam);
|
52
|
+
navigator.mediaSession.setActionHandler('hangup', this.leave);
|
53
|
+
} catch (err) {
|
54
|
+
console.error('error in setting media session handlers', err);
|
55
|
+
}
|
56
|
+
}
|
57
|
+
};
|
58
|
+
}
|
59
|
+
|
60
|
+
export const MediaSession = new SetupMediaSession();
|
@@ -0,0 +1,11 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import PIPComponent from './PIPComponent';
|
3
|
+
import { usePinnedTrack } from '../AppData/useUISettings';
|
4
|
+
|
5
|
+
export const PIP = () => {
|
6
|
+
const pinnedTrack = usePinnedTrack();
|
7
|
+
|
8
|
+
return (
|
9
|
+
<PIPComponent peers={pinnedTrack && pinnedTrack.enabled ? [pinnedTrack.peerId] : undefined} showLocalPeer={true} />
|
10
|
+
);
|
11
|
+
};
|
@@ -0,0 +1,72 @@
|
|
1
|
+
import { PictureInPicture } from './PIPManager';
|
2
|
+
|
3
|
+
describe('pip manager tests', () => {
|
4
|
+
/**
|
5
|
+
* Ensure that if a track is showing in both before and after, it's at the
|
6
|
+
* same position.
|
7
|
+
*/
|
8
|
+
test('merging old and new tracks to show avoids shuffling', () => {
|
9
|
+
const makeTestData = (oldArr, newArr, result) => {
|
10
|
+
return { oldArr, newArr, result };
|
11
|
+
};
|
12
|
+
|
13
|
+
const examples = [
|
14
|
+
makeTestData([1, 5, 9, 3], [3, 8, 2, 9], [8, 2, 9, 3]),
|
15
|
+
makeTestData([1, 3, 5], [5, 3, 1], [1, 3, 5]),
|
16
|
+
makeTestData([2, 7, 9, 4], [9, 2, 4, 6], [2, 6, 9, 4]),
|
17
|
+
makeTestData([1, 2, 3], [4, 5, 6], [4, 5, 6]),
|
18
|
+
makeTestData([1, 2], [4, 5, 2, 1], [1, 2, 4, 5]),
|
19
|
+
makeTestData([4, 1, 2, 3], [1, 3], [3, 1]),
|
20
|
+
makeTestData([4, 5, 1, 3], [1, 3], [1, 3]),
|
21
|
+
makeTestData([], [1, 3], [1, 3]),
|
22
|
+
makeTestData([1, 3], [], []),
|
23
|
+
makeTestData([1], [4, 2, 1, 3], [1, 4, 2, 3]),
|
24
|
+
makeTestData([4, 2, 1, 3], [1], [1]),
|
25
|
+
makeTestData([], [], []),
|
26
|
+
];
|
27
|
+
|
28
|
+
examples.forEach(example => {
|
29
|
+
expect(PictureInPicture.orderNewTracksToShow(example.newArr, example.oldArr)).toEqual(example.result);
|
30
|
+
});
|
31
|
+
});
|
32
|
+
|
33
|
+
/**
|
34
|
+
* It's very important that detach is properly called for old tracks to guarantee
|
35
|
+
* no unnecessary bandwidth consumption.
|
36
|
+
* For simplicity's sake currently, detach/attach is done if position of
|
37
|
+
* the track changes as well. That is earlier it was showing on third position
|
38
|
+
* but not it's on second position(because there are only two tracks left to show).
|
39
|
+
*/
|
40
|
+
test('attach and detach are called properly after tracks in view changes', async () => {
|
41
|
+
const makeTestData = (oldTracks, newTracks, detachFor, attachFor) => {
|
42
|
+
return { oldTracks, newTracks, detachFor, attachFor };
|
43
|
+
};
|
44
|
+
|
45
|
+
let attachCalledFor = [];
|
46
|
+
let detachCalledFor = [];
|
47
|
+
PictureInPicture.hmsActions = {
|
48
|
+
attachVideo: jest.fn(track => attachCalledFor.push(track)),
|
49
|
+
detachVideo: jest.fn(track => detachCalledFor.push(track)),
|
50
|
+
};
|
51
|
+
|
52
|
+
const examples = [
|
53
|
+
makeTestData([1, 2], [3, 4], [1, 2], [3, 4]),
|
54
|
+
makeTestData([1, 2], [1, 2], [], []),
|
55
|
+
makeTestData([1, 5, 9, 3], [8, 2, 9, 3], [1, 5], [8, 2]),
|
56
|
+
makeTestData([1, 5, 9], [6, 5, 9], [1], [6]),
|
57
|
+
makeTestData([], [1, 3], [], [1, 3]),
|
58
|
+
makeTestData([1, 3], [], [1, 3], []),
|
59
|
+
makeTestData([1], [4, 2, 1, 3], [1], [4, 2, 1, 3]),
|
60
|
+
makeTestData([4, 2, 1, 3], [1], [4, 2, 1, 3], [1]),
|
61
|
+
makeTestData([], [], [], []),
|
62
|
+
];
|
63
|
+
|
64
|
+
for (let example of examples) {
|
65
|
+
attachCalledFor = [];
|
66
|
+
detachCalledFor = [];
|
67
|
+
await PictureInPicture.detachOldAttachNewTracks(example.oldTracks, example.newTracks);
|
68
|
+
expect(attachCalledFor).toEqual(example.attachFor);
|
69
|
+
expect(detachCalledFor).toEqual(example.detachFor);
|
70
|
+
}
|
71
|
+
});
|
72
|
+
});
|
@@ -0,0 +1,183 @@
|
|
1
|
+
let CANVAS_FILL_COLOR;
|
2
|
+
let CANVAS_STROKE_COLOR;
|
3
|
+
|
4
|
+
function setPIPCanvasColors() {
|
5
|
+
if (!CANVAS_FILL_COLOR) {
|
6
|
+
CANVAS_FILL_COLOR = window
|
7
|
+
.getComputedStyle(document.documentElement)
|
8
|
+
.getPropertyValue('--hms-ui-colors-surfaceLight');
|
9
|
+
}
|
10
|
+
if (!CANVAS_STROKE_COLOR) {
|
11
|
+
CANVAS_STROKE_COLOR = window
|
12
|
+
.getComputedStyle(document.documentElement)
|
13
|
+
.getPropertyValue('--hms-ui-colors-borderLight');
|
14
|
+
}
|
15
|
+
}
|
16
|
+
|
17
|
+
export function resetPIPCanvasColors() {
|
18
|
+
CANVAS_FILL_COLOR = '';
|
19
|
+
CANVAS_STROKE_COLOR = '';
|
20
|
+
}
|
21
|
+
/**
|
22
|
+
* no tile - blank canvas, black image
|
23
|
+
* 1 tile - takes full space on canvas
|
24
|
+
* 2 tile - stack two video adjacent to each other
|
25
|
+
* 3 tile - two rows first row has two tile second row has one tile centered.
|
26
|
+
* 4 tiles - two rows two columns - all equal size
|
27
|
+
* All videos will respect their aspect ratios.
|
28
|
+
*/
|
29
|
+
export function drawVideoElementsOnCanvas(videoElements, canvas) {
|
30
|
+
let videoTiles = videoElements.filter(videoElement => videoElement.srcObject !== null);
|
31
|
+
|
32
|
+
const ctx = canvas.getContext('2d');
|
33
|
+
setPIPCanvasColors();
|
34
|
+
ctx.fillStyle = CANVAS_FILL_COLOR;
|
35
|
+
ctx.clearRect(0, 0, canvas.width, canvas.height);
|
36
|
+
|
37
|
+
if (videoTiles.length === 0) {
|
38
|
+
// no tile to render, render black image
|
39
|
+
ctx.fillRect(0, 0, canvas.width, canvas.height);
|
40
|
+
return;
|
41
|
+
}
|
42
|
+
|
43
|
+
fillGridTiles(videoTiles.slice(0, 4), ctx, canvas);
|
44
|
+
}
|
45
|
+
|
46
|
+
// this is to send some data for stream and resolve video element's play for a
|
47
|
+
// video element rendering this canvas' capture stream
|
48
|
+
export function dummyChangeInCanvas(canvas) {
|
49
|
+
const ctx = canvas.getContext('2d');
|
50
|
+
setPIPCanvasColors();
|
51
|
+
ctx.fillStyle = CANVAS_FILL_COLOR;
|
52
|
+
ctx.fillRect(0, 0, canvas.width, canvas.height);
|
53
|
+
}
|
54
|
+
|
55
|
+
/**
|
56
|
+
* Imagine the canvas as a grid with passed in number of rows and columns. Go
|
57
|
+
* over the tiles in the grid in order while drawing the videoElements upon them.
|
58
|
+
*/
|
59
|
+
function fillGridTiles(videoElements, ctx, canvas) {
|
60
|
+
const offset = 8;
|
61
|
+
canvas.width = 480;
|
62
|
+
canvas.height = 320;
|
63
|
+
|
64
|
+
ctx.fillStyle = CANVAS_FILL_COLOR;
|
65
|
+
ctx.fillRect(0, 0, canvas.width, canvas.height);
|
66
|
+
// Show borders only when there is atleast one video
|
67
|
+
if (videoElements.length > 0) {
|
68
|
+
ctx.strokeStyle = CANVAS_STROKE_COLOR;
|
69
|
+
ctx.lineWidth = offset / 2;
|
70
|
+
ctx.strokeRect(0, 0, canvas.width, canvas.height);
|
71
|
+
}
|
72
|
+
|
73
|
+
if (videoElements.length === 1) {
|
74
|
+
const video = videoElements[0];
|
75
|
+
const { width, height } = getRenderDimensions(video, canvas.width - offset, canvas.height - offset);
|
76
|
+
/**
|
77
|
+
* The x and y offset are to center the video tile horizontally and vertically
|
78
|
+
* width and height are the aspect ratio constrained video tile dimensions
|
79
|
+
*/
|
80
|
+
const xOffset = (canvas.width - width) / 2;
|
81
|
+
const yOffset = (canvas.height - height) / 2;
|
82
|
+
ctx.drawImage(video, xOffset, yOffset, width, height);
|
83
|
+
}
|
84
|
+
|
85
|
+
if (videoElements.length === 2) {
|
86
|
+
videoElements.forEach((video, index) => {
|
87
|
+
const { width, height } = getRenderDimensions(
|
88
|
+
video,
|
89
|
+
canvas.width / 2 - offset, // This will be the max available width for each tile
|
90
|
+
canvas.height - offset,
|
91
|
+
);
|
92
|
+
/**
|
93
|
+
* (canvas.width / 2 - width) / 2 This is to center width wise within in the box
|
94
|
+
* (canvas.width / 2) * index This is the start offset
|
95
|
+
* for 1st element it is 0, for second it will be canvas.width/2 which starts from the center
|
96
|
+
*/
|
97
|
+
const xOffset = (canvas.width / 2 - width) / 2 + (canvas.width / 2) * index;
|
98
|
+
/**
|
99
|
+
* (canvas.height - height) / 2 This is to center height wise
|
100
|
+
*/
|
101
|
+
const yOffset = (canvas.height - height) / 2;
|
102
|
+
|
103
|
+
ctx.drawImage(video, xOffset, yOffset, width, height);
|
104
|
+
});
|
105
|
+
/**
|
106
|
+
* Draw a border between tiles
|
107
|
+
*/
|
108
|
+
const path = new Path2D();
|
109
|
+
path.moveTo(canvas.width / 2, 0);
|
110
|
+
path.lineTo(canvas.width / 2, canvas.height);
|
111
|
+
ctx.stroke(path);
|
112
|
+
}
|
113
|
+
|
114
|
+
if (videoElements.length === 3) {
|
115
|
+
videoElements.forEach((video, index) => {
|
116
|
+
const { width, height } = getRenderDimensions(video, canvas.width / 2 - offset, canvas.height / 2 - offset);
|
117
|
+
/**
|
118
|
+
* for first two tiles, xOffset is similar to the 2 tiles calculation with only difference being the height. it is half now.
|
119
|
+
*/
|
120
|
+
const xOffset =
|
121
|
+
index < 2 ? (canvas.width / 2 - width) / 2 + (canvas.width / 2) * index : canvas.width / 2 - width / 2;
|
122
|
+
const yOffset = (index < 2 ? 0 : canvas.height / 2) + (canvas.height / 2 - height) / 2;
|
123
|
+
|
124
|
+
ctx.drawImage(video, xOffset, yOffset, width, height);
|
125
|
+
});
|
126
|
+
/**
|
127
|
+
* Draw borders between tiles
|
128
|
+
*/
|
129
|
+
const path = new Path2D();
|
130
|
+
path.moveTo(canvas.width / 2, 0);
|
131
|
+
path.lineTo(canvas.width / 2, canvas.height / 2);
|
132
|
+
ctx.stroke(path);
|
133
|
+
path.moveTo(0, canvas.height / 2);
|
134
|
+
path.lineTo(canvas.width, canvas.height / 2);
|
135
|
+
ctx.stroke(path);
|
136
|
+
}
|
137
|
+
if (videoElements.length === 4) {
|
138
|
+
videoElements.forEach((video, index) => {
|
139
|
+
const { width, height } = getRenderDimensions(video, canvas.width / 2 - offset, canvas.height / 2 - offset);
|
140
|
+
/**
|
141
|
+
* Similar to two tiles repeat after 2 tiles
|
142
|
+
* (canvas.width / 2 - width) / 2 is to center horizontally
|
143
|
+
*/
|
144
|
+
const xOffset = (canvas.width / 2 - width) / 2 + (canvas.width / 2) * (index % 2);
|
145
|
+
/**
|
146
|
+
* Similar to two tiles with the yOffset being height/2 for the 3rd and 4th tiles
|
147
|
+
* (canvas.height / 2 - height) / 2 is to center vertically
|
148
|
+
*/
|
149
|
+
const yOffset = (index < 2 ? 0 : canvas.height / 2) + (canvas.height / 2 - height) / 2;
|
150
|
+
|
151
|
+
ctx.drawImage(video, xOffset, yOffset, width, height);
|
152
|
+
});
|
153
|
+
/**
|
154
|
+
* Draw borders between tiles
|
155
|
+
*/
|
156
|
+
const path = new Path2D();
|
157
|
+
path.moveTo(canvas.width / 2, 0);
|
158
|
+
path.lineTo(canvas.width / 2, canvas.height);
|
159
|
+
ctx.stroke(path);
|
160
|
+
path.moveTo(0, canvas.height / 2);
|
161
|
+
path.lineTo(canvas.width, canvas.height / 2);
|
162
|
+
ctx.stroke(path);
|
163
|
+
}
|
164
|
+
}
|
165
|
+
|
166
|
+
/**
|
167
|
+
* Restrict the dimensions within the available dimension with aspect ratio
|
168
|
+
* constraint applied
|
169
|
+
* @param {HTMLVideoElement} video
|
170
|
+
* @param {number} width
|
171
|
+
* @param {number} height
|
172
|
+
* @returns { width: number, height: number }
|
173
|
+
*/
|
174
|
+
function getRenderDimensions(video, width, height) {
|
175
|
+
let finalWidth = (video.videoWidth / video.videoHeight) * height;
|
176
|
+
let finalHeight = height;
|
177
|
+
|
178
|
+
if (finalWidth > width) {
|
179
|
+
finalWidth = width;
|
180
|
+
finalHeight = (video.videoHeight / video.videoWidth) * width;
|
181
|
+
}
|
182
|
+
return { width: finalWidth, height: finalHeight };
|
183
|
+
}
|
@@ -0,0 +1,29 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import { ChevronLeftIcon, ChevronRightIcon } from '@100mslive/react-icons';
|
3
|
+
import { StyledPagination } from '../../Pagination';
|
4
|
+
|
5
|
+
export const Pagination = ({ page, setPage, numPages }) => {
|
6
|
+
const disableLeft = page === 0;
|
7
|
+
const disableRight = page === numPages - 1;
|
8
|
+
const nextPage = () => {
|
9
|
+
setPage(Math.min(page + 1, numPages - 1));
|
10
|
+
};
|
11
|
+
const prevPage = () => {
|
12
|
+
setPage(Math.max(page - 1, 0));
|
13
|
+
};
|
14
|
+
return (
|
15
|
+
<StyledPagination.Root>
|
16
|
+
<StyledPagination.Chevron disabled={disableLeft} onClick={prevPage}>
|
17
|
+
<ChevronLeftIcon width={16} height={16} style={{ cursor: disableLeft ? 'not-allowed' : 'pointer' }} />
|
18
|
+
</StyledPagination.Chevron>
|
19
|
+
<StyledPagination.Dots>
|
20
|
+
{[...Array(numPages)].map((_, i) => (
|
21
|
+
<StyledPagination.Dot key={i} active={page === i} onClick={() => setPage(i)} />
|
22
|
+
))}
|
23
|
+
</StyledPagination.Dots>
|
24
|
+
<StyledPagination.Chevron disabled={disableRight} onClick={nextPage}>
|
25
|
+
<ChevronRightIcon width={16} height={16} style={{ cursor: disableRight ? 'not-allowed' : 'pointer' }} />
|
26
|
+
</StyledPagination.Chevron>
|
27
|
+
</StyledPagination.Root>
|
28
|
+
);
|
29
|
+
};
|