@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.
Files changed (459) hide show
  1. package/README.md +1 -0
  2. package/dist/Accordion/Accordion.d.ts +964 -0
  3. package/dist/Accordion/index.d.ts +979 -0
  4. package/dist/ActiveSpeakerView-H3VYXANB.js +39 -0
  5. package/dist/ActiveSpeakerView-H3VYXANB.js.map +7 -0
  6. package/dist/ActiveSpeakerView-REZLWPPI.css +11 -0
  7. package/dist/ActiveSpeakerView-REZLWPPI.css.map +7 -0
  8. package/dist/AudioLevel/AudioLevel.d.ts +8 -0
  9. package/dist/AudioLevel/index.d.ts +1 -0
  10. package/dist/Avatar/Avatar.d.ts +487 -0
  11. package/dist/Avatar/getAvatarBg.d.ts +7 -0
  12. package/dist/Avatar/index.d.ts +1 -0
  13. package/dist/Button/Button.d.ts +488 -0
  14. package/dist/Button/index.d.ts +1 -0
  15. package/dist/Checkbox/Checkbox.d.ts +958 -0
  16. package/dist/Checkbox/index.d.ts +1 -0
  17. package/dist/Collapsible/Collapsible.d.ts +1434 -0
  18. package/dist/Collapsible/index.d.ts +1 -0
  19. package/dist/Divider/Divider.d.ts +958 -0
  20. package/dist/Divider/index.d.ts +1 -0
  21. package/dist/Dropdown/Dropdown.d.ts +5728 -0
  22. package/dist/Dropdown/index.d.ts +1 -0
  23. package/dist/Fieldset/Fieldset.d.ts +477 -0
  24. package/dist/Fieldset/index.d.ts +1 -0
  25. package/dist/Footer/Footer.d.ts +3834 -0
  26. package/dist/Footer/index.d.ts +1 -0
  27. package/dist/HLSView-3RDXRV7Y.js +689 -0
  28. package/dist/HLSView-3RDXRV7Y.js.map +7 -0
  29. package/dist/HLSView-6BVBCQM7.css +11 -0
  30. package/dist/HLSView-6BVBCQM7.css.map +7 -0
  31. package/dist/IconButton/IconButton.d.ts +479 -0
  32. package/dist/IconButton/index.d.ts +1 -0
  33. package/dist/Input/Input.d.ts +2917 -0
  34. package/dist/Input/index.d.ts +1 -0
  35. package/dist/Label/Label.d.ts +479 -0
  36. package/dist/Label/index.d.ts +1 -0
  37. package/dist/Layout/Box.d.ts +477 -0
  38. package/dist/Layout/Flex.d.ts +482 -0
  39. package/dist/Layout/index.d.ts +2 -0
  40. package/dist/Link/Link.d.ts +489 -0
  41. package/dist/Link/index.d.ts +2 -0
  42. package/dist/Loading/Loading.d.ts +14 -0
  43. package/dist/Loading/index.d.ts +1 -0
  44. package/dist/Modal/Dialog.d.ts +3826 -0
  45. package/dist/Modal/DialogContent.d.ts +3343 -0
  46. package/dist/Modal/index.d.ts +1 -0
  47. package/dist/Pagination/StyledPagination.d.ts +1918 -0
  48. package/dist/Pagination/index.d.ts +1 -0
  49. package/dist/PinnedTrackView-453PELNU.js +70 -0
  50. package/dist/PinnedTrackView-453PELNU.js.map +7 -0
  51. package/dist/PinnedTrackView-QQ5FDXJX.css +11 -0
  52. package/dist/PinnedTrackView-QQ5FDXJX.css.map +7 -0
  53. package/dist/Popover/index.d.ts +1436 -0
  54. package/dist/Progress/index.d.ts +957 -0
  55. package/dist/QRCode/QRCode.d.ts +3 -0
  56. package/dist/QRCode/index.d.ts +1 -0
  57. package/dist/RadioGroup/RadioGroup.d.ts +1435 -0
  58. package/dist/RadioGroup/index.d.ts +1 -0
  59. package/dist/ReactSelect/ReactSelect.d.ts +4778 -0
  60. package/dist/ReactSelect/index.d.ts +1 -0
  61. package/dist/Select/Select.d.ts +1437 -0
  62. package/dist/Select/index.d.ts +1 -0
  63. package/dist/Slider/Slider.d.ts +488 -0
  64. package/dist/Slider/index.d.ts +1 -0
  65. package/dist/Stats/Stats.d.ts +19 -0
  66. package/dist/Stats/StyledStats.d.ts +2872 -0
  67. package/dist/Stats/formatBytes.d.ts +1 -0
  68. package/dist/Stats/index.d.ts +1 -0
  69. package/dist/Switch/Switch.d.ts +482 -0
  70. package/dist/Switch/index.d.ts +1 -0
  71. package/dist/Tabs/Tabs.d.ts +1912 -0
  72. package/dist/Tabs/index.d.ts +1 -0
  73. package/dist/Text/Text.d.ts +605 -0
  74. package/dist/Text/index.d.ts +1 -0
  75. package/dist/Theme/ThemeProvider.d.ts +397 -0
  76. package/dist/Theme/base.config.d.ts +417 -0
  77. package/dist/Theme/index.d.ts +3 -0
  78. package/dist/Theme/stitches.config.d.ts +4702 -0
  79. package/dist/Theme/themes.d.ts +64 -0
  80. package/dist/Theme/useSSR.d.ts +6 -0
  81. package/dist/TileMenu/StyledMenuTile.d.ts +3351 -0
  82. package/dist/TileMenu/TileMenu.d.ts +6 -0
  83. package/dist/TileMenu/index.d.ts +1 -0
  84. package/dist/Toast/Toast.d.ts +2881 -0
  85. package/dist/Toast/index.d.ts +1 -0
  86. package/dist/Tooltip/Tooltip.d.ts +14 -0
  87. package/dist/Tooltip/index.d.ts +1 -0
  88. package/dist/Video/Video.d.ts +499 -0
  89. package/dist/Video/index.d.ts +1 -0
  90. package/dist/VideoList/StyledVideoList.d.ts +1438 -0
  91. package/dist/VideoList/index.d.ts +2 -0
  92. package/dist/VideoList/videoListUtils.d.ts +1 -0
  93. package/dist/VideoTile/StyledVideoTile.d.ts +4309 -0
  94. package/dist/VideoTile/index.d.ts +1 -0
  95. package/dist/VirtualBackground-LHYBWUT5.js +158 -0
  96. package/dist/VirtualBackground-LHYBWUT5.js.map +7 -0
  97. package/dist/chunk-7YUYZ64D.js +6843 -0
  98. package/dist/chunk-7YUYZ64D.js.map +7 -0
  99. package/dist/chunk-HCAGFNXW.js +8270 -0
  100. package/dist/chunk-HCAGFNXW.js.map +7 -0
  101. package/dist/chunk-KYYP6ZVK.js +907 -0
  102. package/dist/chunk-KYYP6ZVK.js.map +7 -0
  103. package/dist/chunk-ULXGBUSC.js +65 -0
  104. package/dist/chunk-ULXGBUSC.js.map +7 -0
  105. package/dist/chunk-XRJXE6UO.js +243 -0
  106. package/dist/chunk-XRJXE6UO.js.map +7 -0
  107. package/dist/conference-IDNRO4WK.js +3812 -0
  108. package/dist/conference-IDNRO4WK.js.map +7 -0
  109. package/dist/conference-KN6OKGDU.css +11 -0
  110. package/dist/conference-KN6OKGDU.css.map +7 -0
  111. package/dist/fixtures/chats.d.ts +4 -0
  112. package/dist/fixtures/peers.d.ts +3 -0
  113. package/dist/fixtures/tracks.d.ts +2 -0
  114. package/dist/index.cjs.css +11 -0
  115. package/dist/index.cjs.css.map +7 -0
  116. package/dist/index.cjs.js +22987 -0
  117. package/dist/index.cjs.js.map +7 -0
  118. package/dist/index.css +11 -0
  119. package/dist/index.css.map +7 -0
  120. package/dist/index.d.ts +38 -0
  121. package/dist/index.js +133 -0
  122. package/dist/index.js.map +7 -0
  123. package/dist/meta.cjs.json +13085 -0
  124. package/dist/meta.esbuild.json +13693 -0
  125. package/dist/transcription-BTSB7FZH.js +356 -0
  126. package/dist/transcription-BTSB7FZH.js.map +7 -0
  127. package/dist/utils/animations.d.ts +92 -0
  128. package/dist/utils/index.d.ts +2 -0
  129. package/dist/utils/styles.d.ts +21 -0
  130. package/package.json +116 -0
  131. package/src/Accordion/Accordion.stories.tsx +50 -0
  132. package/src/Accordion/Accordion.tsx +88 -0
  133. package/src/Accordion/index.ts +8 -0
  134. package/src/AudioLevel/AudioLevel.tsx +34 -0
  135. package/src/AudioLevel/index.ts +1 -0
  136. package/src/Avatar/Avatar.stories.tsx +33 -0
  137. package/src/Avatar/Avatar.tsx +55 -0
  138. package/src/Avatar/getAvatarBg.ts +46 -0
  139. package/src/Avatar/index.ts +1 -0
  140. package/src/Button/Button.mdx +43 -0
  141. package/src/Button/Button.stories.tsx +52 -0
  142. package/src/Button/Button.tsx +144 -0
  143. package/src/Button/index.tsx +1 -0
  144. package/src/Chat/Chat.mdx +39 -0
  145. package/src/Chat/Chat.stories.tsx +39 -0
  146. package/src/Checkbox/Checkbox.stories.tsx +61 -0
  147. package/src/Checkbox/Checkbox.tsx +33 -0
  148. package/src/Checkbox/index.tsx +1 -0
  149. package/src/Collapsible/Collapsible.tsx +34 -0
  150. package/src/Collapsible/index.tsx +1 -0
  151. package/src/Divider/Divider.tsx +45 -0
  152. package/src/Divider/HorizontalDivider.stories.tsx +34 -0
  153. package/src/Divider/VerticalDivider.stories.tsx +40 -0
  154. package/src/Divider/index.ts +1 -0
  155. package/src/Dropdown/Dropdown.stories.tsx +94 -0
  156. package/src/Dropdown/Dropdown.tsx +142 -0
  157. package/src/Dropdown/index.tsx +1 -0
  158. package/src/Fieldset/Fieldset.stories.tsx +29 -0
  159. package/src/Fieldset/Fieldset.tsx +11 -0
  160. package/src/Fieldset/index.tsx +1 -0
  161. package/src/Footer/Footer.stories.tsx +61 -0
  162. package/src/Footer/Footer.tsx +47 -0
  163. package/src/Footer/index.tsx +1 -0
  164. package/src/IconButton/IconButton.tsx +45 -0
  165. package/src/IconButton/index.tsx +1 -0
  166. package/src/Icons/Icons.stories.mdx +10 -0
  167. package/src/Icons/IconsList.jsx +17 -0
  168. package/src/Input/Input.stories.tsx +25 -0
  169. package/src/Input/Input.tsx +105 -0
  170. package/src/Input/PasswordInput.stories.tsx +53 -0
  171. package/src/Input/index.tsx +1 -0
  172. package/src/Introduction/Integrating.stories.mdx +100 -0
  173. package/src/Introduction/Introduction.stories.mdx +9 -0
  174. package/src/Label/Label.tsx +8 -0
  175. package/src/Label/index.ts +1 -0
  176. package/src/Layout/Box.tsx +3 -0
  177. package/src/Layout/Flex.tsx +76 -0
  178. package/src/Layout/index.tsx +2 -0
  179. package/src/Link/Link.stories.tsx +18 -0
  180. package/src/Link/Link.tsx +54 -0
  181. package/src/Link/index.tsx +2 -0
  182. package/src/Loading/Loading.mdx +15 -0
  183. package/src/Loading/Loading.stories.tsx +37 -0
  184. package/src/Loading/Loading.tsx +33 -0
  185. package/src/Loading/index.ts +1 -0
  186. package/src/Modal/Dialog.mdx +19 -0
  187. package/src/Modal/Dialog.stories.tsx +68 -0
  188. package/src/Modal/Dialog.tsx +26 -0
  189. package/src/Modal/DialogContent.tsx +63 -0
  190. package/src/Modal/index.ts +1 -0
  191. package/src/Pagination/StyledPagination.stories.tsx +80 -0
  192. package/src/Pagination/StyledPagination.tsx +68 -0
  193. package/src/Pagination/index.tsx +1 -0
  194. package/src/Popover/Popover.mdx +9 -0
  195. package/src/Popover/Popover.stories.tsx +95 -0
  196. package/src/Popover/index.tsx +26 -0
  197. package/src/Prebuilt/App.jsx +273 -0
  198. package/src/Prebuilt/AppContext.jsx +21 -0
  199. package/src/Prebuilt/IconButton.jsx +19 -0
  200. package/src/Prebuilt/Prebuilt.stories.tsx +20 -0
  201. package/src/Prebuilt/common/PeersSorter.js +89 -0
  202. package/src/Prebuilt/common/constants.js +208 -0
  203. package/src/Prebuilt/common/hooks.js +47 -0
  204. package/src/Prebuilt/common/roles.js +4 -0
  205. package/src/Prebuilt/common/useSortedPeers.js +28 -0
  206. package/src/Prebuilt/common/utils.js +91 -0
  207. package/src/Prebuilt/components/AppData/AppData.jsx +189 -0
  208. package/src/Prebuilt/components/AppData/useAppConfig.js +7 -0
  209. package/src/Prebuilt/components/AppData/useAppLayout.js +6 -0
  210. package/src/Prebuilt/components/AppData/useChatState.js +18 -0
  211. package/src/Prebuilt/components/AppData/useSidepane.js +49 -0
  212. package/src/Prebuilt/components/AppData/useUISettings.js +164 -0
  213. package/src/Prebuilt/components/AudioLevel/BeamSpeakerLabelsLogging.jsx +16 -0
  214. package/src/Prebuilt/components/AudioVideoToggle.jsx +67 -0
  215. package/src/Prebuilt/components/AuthToken.jsx +133 -0
  216. package/src/Prebuilt/components/BottomActionSheet/BottomActionSheet.jsx +96 -0
  217. package/src/Prebuilt/components/BottomActionSheet/BottomActionSheet.stories.tsx +46 -0
  218. package/src/Prebuilt/components/Chat/Chat.jsx +155 -0
  219. package/src/Prebuilt/components/Chat/ChatBody.jsx +370 -0
  220. package/src/Prebuilt/components/Chat/ChatFooter.jsx +150 -0
  221. package/src/Prebuilt/components/Chat/ChatHeader.jsx +67 -0
  222. package/src/Prebuilt/components/Chat/ChatSelector.jsx +162 -0
  223. package/src/Prebuilt/components/Chat/useEmojiPickerStyles.js +30 -0
  224. package/src/Prebuilt/components/Chat/useUnreadCount.js +17 -0
  225. package/src/Prebuilt/components/Connection/ConnectionIndicator.jsx +80 -0
  226. package/src/Prebuilt/components/Connection/TileConnection.jsx +40 -0
  227. package/src/Prebuilt/components/Connection/connectionQualityUtils.js +39 -0
  228. package/src/Prebuilt/components/EmojiReaction.jsx +138 -0
  229. package/src/Prebuilt/components/ErrorBoundary.jsx +105 -0
  230. package/src/Prebuilt/components/FirstPersonDisplay.jsx +50 -0
  231. package/src/Prebuilt/components/Footer/ChatToggle.jsx +27 -0
  232. package/src/Prebuilt/components/Footer/ConferencingFooter.jsx +104 -0
  233. package/src/Prebuilt/components/Footer/StreamingFooter.jsx +71 -0
  234. package/src/Prebuilt/components/Footer.jsx +8 -0
  235. package/src/Prebuilt/components/FullPageProgress.jsx +11 -0
  236. package/src/Prebuilt/components/GoLiveButton.jsx +45 -0
  237. package/src/Prebuilt/components/HMSVideo/Controls.jsx +21 -0
  238. package/src/Prebuilt/components/HMSVideo/FullscreenButton.jsx +18 -0
  239. package/src/Prebuilt/components/HMSVideo/HLSAutoplayBlockedPrompt.jsx +35 -0
  240. package/src/Prebuilt/components/HMSVideo/HLSQualitySelector.jsx +82 -0
  241. package/src/Prebuilt/components/HMSVideo/HMSVIdeoUtils.js +27 -0
  242. package/src/Prebuilt/components/HMSVideo/HMSVideo.jsx +11 -0
  243. package/src/Prebuilt/components/HMSVideo/PlayButton.jsx +13 -0
  244. package/src/Prebuilt/components/HMSVideo/VideoProgress.jsx +77 -0
  245. package/src/Prebuilt/components/HMSVideo/VideoTime.jsx +31 -0
  246. package/src/Prebuilt/components/HMSVideo/VolumeControl.jsx +39 -0
  247. package/src/Prebuilt/components/HMSVideo/index.js +19 -0
  248. package/src/Prebuilt/components/Header/AdditionalRoomState.jsx +217 -0
  249. package/src/Prebuilt/components/Header/AmbientMusic.jsx +88 -0
  250. package/src/Prebuilt/components/Header/ConferencingHeader.jsx +29 -0
  251. package/src/Prebuilt/components/Header/Header.jsx +8 -0
  252. package/src/Prebuilt/components/Header/HeaderComponents.jsx +41 -0
  253. package/src/Prebuilt/components/Header/ParticipantFilter.jsx +91 -0
  254. package/src/Prebuilt/components/Header/ParticipantList.jsx +337 -0
  255. package/src/Prebuilt/components/Header/StreamActions.jsx +225 -0
  256. package/src/Prebuilt/components/Header/StreamingHeader.jsx +55 -0
  257. package/src/Prebuilt/components/Header/index.jsx +1 -0
  258. package/src/Prebuilt/components/HlsStatsOverlay.jsx +101 -0
  259. package/src/Prebuilt/components/Image.jsx +7 -0
  260. package/src/Prebuilt/components/Input/KeyboardInputManager.js +107 -0
  261. package/src/Prebuilt/components/LeaveRoom.jsx +202 -0
  262. package/src/Prebuilt/components/MetaActions.jsx +45 -0
  263. package/src/Prebuilt/components/MoreSettings/BulkRoleChangeModal.jsx +139 -0
  264. package/src/Prebuilt/components/MoreSettings/ChangeNameModal.jsx +92 -0
  265. package/src/Prebuilt/components/MoreSettings/ChangeSelfRole.jsx +67 -0
  266. package/src/Prebuilt/components/MoreSettings/EmbedUrl.jsx +106 -0
  267. package/src/Prebuilt/components/MoreSettings/FullScreenItem.jsx +29 -0
  268. package/src/Prebuilt/components/MoreSettings/MoreSettings.jsx +225 -0
  269. package/src/Prebuilt/components/MoreSettings/MuteAllModal.jsx +90 -0
  270. package/src/Prebuilt/components/Notifications/AutoplayBlockedModal.jsx +38 -0
  271. package/src/Prebuilt/components/Notifications/InitErrorModal.jsx +39 -0
  272. package/src/Prebuilt/components/Notifications/MessageNotifications.jsx +25 -0
  273. package/src/Prebuilt/components/Notifications/Notifications.jsx +151 -0
  274. package/src/Prebuilt/components/Notifications/PeerNotifications.jsx +45 -0
  275. package/src/Prebuilt/components/Notifications/PermissionErrorModal.jsx +67 -0
  276. package/src/Prebuilt/components/Notifications/ReconnectNotifications.jsx +69 -0
  277. package/src/Prebuilt/components/Notifications/TrackBulkUnmuteModal.jsx +51 -0
  278. package/src/Prebuilt/components/Notifications/TrackNotifications.jsx +19 -0
  279. package/src/Prebuilt/components/Notifications/TrackUnmuteModal.jsx +49 -0
  280. package/src/Prebuilt/components/Notifications/index.jsx +1 -0
  281. package/src/Prebuilt/components/PIP/PIPComponent.jsx +82 -0
  282. package/src/Prebuilt/components/PIP/PIPManager.js +296 -0
  283. package/src/Prebuilt/components/PIP/SetupMediaSession.js +60 -0
  284. package/src/Prebuilt/components/PIP/index.jsx +11 -0
  285. package/src/Prebuilt/components/PIP/pip.test.js +72 -0
  286. package/src/Prebuilt/components/PIP/pipUtils.js +183 -0
  287. package/src/Prebuilt/components/Pagination.jsx +29 -0
  288. package/src/Prebuilt/components/Playlist/Playlist.jsx +129 -0
  289. package/src/Prebuilt/components/Playlist/PlaylistControls.jsx +172 -0
  290. package/src/Prebuilt/components/Playlist/PlaylistItem.jsx +51 -0
  291. package/src/Prebuilt/components/Playlist/VideoPlayer.jsx +95 -0
  292. package/src/Prebuilt/components/PostLeave.jsx +76 -0
  293. package/src/Prebuilt/components/Preview/PreviewContainer.jsx +51 -0
  294. package/src/Prebuilt/components/Preview/PreviewJoin.jsx +196 -0
  295. package/src/Prebuilt/components/Preview/PreviewName.jsx +37 -0
  296. package/src/Prebuilt/components/RoleChangeModal.jsx +185 -0
  297. package/src/Prebuilt/components/RoleChangeRequestModal.jsx +26 -0
  298. package/src/Prebuilt/components/ScreenShare.jsx +45 -0
  299. package/src/Prebuilt/components/ScreenshareDisplay.jsx +45 -0
  300. package/src/Prebuilt/components/ScreenshareHintModal.jsx +37 -0
  301. package/src/Prebuilt/components/ScreenshareTile.jsx +91 -0
  302. package/src/Prebuilt/components/Settings/DeviceSettings.jsx +220 -0
  303. package/src/Prebuilt/components/Settings/LayoutSettings.jsx +91 -0
  304. package/src/Prebuilt/components/Settings/NotificationSettings.jsx +61 -0
  305. package/src/Prebuilt/components/Settings/SettingsModal.jsx +171 -0
  306. package/src/Prebuilt/components/Settings/StartRecording.jsx +130 -0
  307. package/src/Prebuilt/components/Settings/SwitchWithLabel.jsx +40 -0
  308. package/src/Prebuilt/components/Settings/common.js +15 -0
  309. package/src/Prebuilt/components/ShareMenuIcon.jsx +26 -0
  310. package/src/Prebuilt/components/StatsForNerds.jsx +250 -0
  311. package/src/Prebuilt/components/Streaming/Common.jsx +132 -0
  312. package/src/Prebuilt/components/Streaming/HLSStreaming.jsx +226 -0
  313. package/src/Prebuilt/components/Streaming/RTMPIcon.jsx +24 -0
  314. package/src/Prebuilt/components/Streaming/RTMPStreaming.jsx +336 -0
  315. package/src/Prebuilt/components/Streaming/ResolutionInput.jsx +88 -0
  316. package/src/Prebuilt/components/Streaming/StreamingLanding.jsx +76 -0
  317. package/src/Prebuilt/components/TileMenu.jsx +275 -0
  318. package/src/Prebuilt/components/Toast/Toast.jsx +17 -0
  319. package/src/Prebuilt/components/Toast/ToastBatcher.js +57 -0
  320. package/src/Prebuilt/components/Toast/ToastConfig.jsx +134 -0
  321. package/src/Prebuilt/components/Toast/ToastContainer.jsx +30 -0
  322. package/src/Prebuilt/components/Toast/ToastManager.js +44 -0
  323. package/src/Prebuilt/components/VideoList.jsx +101 -0
  324. package/src/Prebuilt/components/VideoTile.jsx +177 -0
  325. package/src/Prebuilt/components/conference.jsx +164 -0
  326. package/src/Prebuilt/components/gridView.jsx +85 -0
  327. package/src/Prebuilt/components/hooks/useDropdownList.jsx +23 -0
  328. package/src/Prebuilt/components/hooks/useDropdownSelection.jsx +6 -0
  329. package/src/Prebuilt/components/hooks/useFeatures.js +22 -0
  330. package/src/Prebuilt/components/hooks/useFullscreen.js +43 -0
  331. package/src/Prebuilt/components/hooks/useMetadata.jsx +52 -0
  332. package/src/Prebuilt/components/hooks/useNavigation.js +19 -0
  333. package/src/Prebuilt/components/hooks/usePlaylist.js +25 -0
  334. package/src/Prebuilt/components/hooks/usePlaylistMusic.js +35 -0
  335. package/src/Prebuilt/components/hooks/useScreenshareAudio.js +28 -0
  336. package/src/Prebuilt/components/hooks/useSetPinnedMessage.js +38 -0
  337. package/src/Prebuilt/components/hooks/useSkipPreview.jsx +20 -0
  338. package/src/Prebuilt/components/hooks/useUserPreferences.jsx +25 -0
  339. package/src/Prebuilt/components/init/Init.jsx +58 -0
  340. package/src/Prebuilt/components/init/initUtils.js +90 -0
  341. package/src/Prebuilt/components/pdfAnnotator/pdfErrorView.jsx +29 -0
  342. package/src/Prebuilt/components/pdfAnnotator/pdfFileOptions.jsx +108 -0
  343. package/src/Prebuilt/components/pdfAnnotator/pdfHeader.jsx +31 -0
  344. package/src/Prebuilt/components/pdfAnnotator/pdfInfo.jsx +32 -0
  345. package/src/Prebuilt/components/pdfAnnotator/shareScreenOptions.jsx +233 -0
  346. package/src/Prebuilt/components/pdfAnnotator/submitPdf.jsx +89 -0
  347. package/src/Prebuilt/components/pdfAnnotator/uploadedFile.jsx +85 -0
  348. package/src/Prebuilt/components/peerTileUtils.jsx +27 -0
  349. package/src/Prebuilt/hms.js +7 -0
  350. package/src/Prebuilt/images/first_person.png +0 -0
  351. package/src/Prebuilt/index.d.ts +15 -0
  352. package/src/Prebuilt/index.js +2 -0
  353. package/src/Prebuilt/layouts/ActiveSpeakerView.jsx +34 -0
  354. package/src/Prebuilt/layouts/EmbedView.jsx +141 -0
  355. package/src/Prebuilt/layouts/HLSView.jsx +290 -0
  356. package/src/Prebuilt/layouts/InsetView.jsx +222 -0
  357. package/src/Prebuilt/layouts/NonPublisherView.jsx +51 -0
  358. package/src/Prebuilt/layouts/PDFView.jsx +122 -0
  359. package/src/Prebuilt/layouts/PinnedTrackView.jsx +59 -0
  360. package/src/Prebuilt/layouts/SidePane.jsx +51 -0
  361. package/src/Prebuilt/layouts/WaitingView.jsx +51 -0
  362. package/src/Prebuilt/layouts/WhiteboardView.jsx +66 -0
  363. package/src/Prebuilt/layouts/mainGridView.jsx +98 -0
  364. package/src/Prebuilt/layouts/mainView.jsx +113 -0
  365. package/src/Prebuilt/layouts/screenShareView.jsx +185 -0
  366. package/src/Prebuilt/plugins/FlyingEmoji.jsx +132 -0
  367. package/src/Prebuilt/plugins/RemoteStopScreenshare.jsx +18 -0
  368. package/src/Prebuilt/plugins/VirtualBackground/VirtualBackground.jsx +90 -0
  369. package/src/Prebuilt/plugins/VirtualBackground/vbutils.js +66 -0
  370. package/src/Prebuilt/plugins/confetti.jsx +60 -0
  371. package/src/Prebuilt/plugins/transcription/Transcriber.js +216 -0
  372. package/src/Prebuilt/plugins/transcription/TranscriptionButton.jsx +138 -0
  373. package/src/Prebuilt/plugins/transcription/index.jsx +1 -0
  374. package/src/Prebuilt/plugins/whiteboard/PusherCommunicationProvider.js +110 -0
  375. package/src/Prebuilt/plugins/whiteboard/README.md +29 -0
  376. package/src/Prebuilt/plugins/whiteboard/ToggleWhiteboard.jsx +43 -0
  377. package/src/Prebuilt/plugins/whiteboard/Whiteboard.css +12 -0
  378. package/src/Prebuilt/plugins/whiteboard/Whiteboard.jsx +11 -0
  379. package/src/Prebuilt/plugins/whiteboard/WhiteboardEvents.js +8 -0
  380. package/src/Prebuilt/plugins/whiteboard/index.js +3 -0
  381. package/src/Prebuilt/plugins/whiteboard/useMultiplayerState.js +212 -0
  382. package/src/Prebuilt/plugins/whiteboard/useWhiteboardMetadata.js +47 -0
  383. package/src/Prebuilt/primitives/DialogContent.jsx +280 -0
  384. package/src/Prebuilt/primitives/DropdownTrigger.jsx +46 -0
  385. package/src/Prebuilt/services/FeatureFlags.jsx +47 -0
  386. package/src/Prebuilt/services/tokenService.js +49 -0
  387. package/src/Progress/index.tsx +17 -0
  388. package/src/QRCode/QRCode.mdx +9 -0
  389. package/src/QRCode/QRCode.stories.tsx +29 -0
  390. package/src/QRCode/QRCode.tsx +6 -0
  391. package/src/QRCode/index.tsx +1 -0
  392. package/src/RadioGroup/RadioGroup.stories.tsx +32 -0
  393. package/src/RadioGroup/RadioGroup.tsx +33 -0
  394. package/src/RadioGroup/index.tsx +1 -0
  395. package/src/ReactSelect/ReactSelect.stories.tsx +83 -0
  396. package/src/ReactSelect/ReactSelect.tsx +97 -0
  397. package/src/ReactSelect/index.ts +1 -0
  398. package/src/Select/Select.stories.tsx +33 -0
  399. package/src/Select/Select.tsx +63 -0
  400. package/src/Select/index.ts +1 -0
  401. package/src/Slider/Slider.stories.tsx +21 -0
  402. package/src/Slider/Slider.tsx +70 -0
  403. package/src/Slider/index.ts +1 -0
  404. package/src/Stats/Stats.tsx +211 -0
  405. package/src/Stats/StyledStats.tsx +57 -0
  406. package/src/Stats/formatBytes.ts +19 -0
  407. package/src/Stats/index.tsx +1 -0
  408. package/src/Switch/Switch.mdx +11 -0
  409. package/src/Switch/Switch.stories.tsx +46 -0
  410. package/src/Switch/Switch.tsx +52 -0
  411. package/src/Switch/index.ts +1 -0
  412. package/src/Tabs/Tabs.stories.tsx +77 -0
  413. package/src/Tabs/Tabs.tsx +41 -0
  414. package/src/Tabs/index.tsx +1 -0
  415. package/src/Text/Text.stories.tsx +21 -0
  416. package/src/Text/Text.tsx +149 -0
  417. package/src/Text/index.tsx +1 -0
  418. package/src/Theme/Theme.stories.mdx +8 -0
  419. package/src/Theme/ThemeProvider.tsx +104 -0
  420. package/src/Theme/ThemeStory.jsx +58 -0
  421. package/src/Theme/base.config.ts +264 -0
  422. package/src/Theme/index.tsx +3 -0
  423. package/src/Theme/stitches.config.ts +100 -0
  424. package/src/Theme/themes.ts +70 -0
  425. package/src/Theme/useSSR.tsx +24 -0
  426. package/src/TileMenu/StyledMenuTile.tsx +101 -0
  427. package/src/TileMenu/TileMenu.tsx +96 -0
  428. package/src/TileMenu/index.tsx +1 -0
  429. package/src/Toast/AppToast.stories.tsx +56 -0
  430. package/src/Toast/Toast.mdx +19 -0
  431. package/src/Toast/Toast.stories.tsx +57 -0
  432. package/src/Toast/Toast.tsx +168 -0
  433. package/src/Toast/index.tsx +1 -0
  434. package/src/Tooltip/Tooltip.stories.tsx +62 -0
  435. package/src/Tooltip/Tooltip.tsx +79 -0
  436. package/src/Tooltip/index.ts +1 -0
  437. package/src/Video/UseVideo.mdx +22 -0
  438. package/src/Video/UseVideo.stories.tsx +26 -0
  439. package/src/Video/Video.mdx +24 -0
  440. package/src/Video/Video.stories.tsx +27 -0
  441. package/src/Video/Video.tsx +61 -0
  442. package/src/Video/index.tsx +1 -0
  443. package/src/VideoList/StyledVideoList.tsx +39 -0
  444. package/src/VideoList/VideoList.stories.tsx +92 -0
  445. package/src/VideoList/index.tsx +2 -0
  446. package/src/VideoList/videoListUtils.tsx +20 -0
  447. package/src/VideoTile/StyledVideoTile.tsx +148 -0
  448. package/src/VideoTile/VideoTile.mdx +28 -0
  449. package/src/VideoTile/VideoTile.stories.tsx +32 -0
  450. package/src/VideoTile/index.tsx +1 -0
  451. package/src/fixtures/chats.ts +25 -0
  452. package/src/fixtures/peers.ts +24 -0
  453. package/src/fixtures/tracks.ts +11 -0
  454. package/src/index.ts +38 -0
  455. package/src/store/SetupFakeStore.ts +33 -0
  456. package/src/store/StorybookSDK.ts +229 -0
  457. package/src/utils/animations.ts +90 -0
  458. package/src/utils/index.ts +2 -0
  459. 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
+ };