@streamplace/components 0.8.9 → 0.8.13
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/dist/components/chat/chat-box.d.ts +8 -0
- package/dist/components/chat/chat-box.d.ts.map +1 -0
- package/dist/components/chat/chat-box.js +19 -2
- package/dist/components/chat/chat-box.js.map +1 -0
- package/dist/components/chat/chat-message.d.ts +8 -0
- package/dist/components/chat/chat-message.d.ts.map +1 -0
- package/dist/components/chat/chat-message.js +6 -4
- package/dist/components/chat/chat-message.js.map +1 -0
- package/dist/components/chat/chat.d.ts +7 -0
- package/dist/components/chat/chat.d.ts.map +1 -0
- package/dist/components/chat/chat.js +1 -0
- package/dist/components/chat/chat.js.map +1 -0
- package/dist/components/chat/emoji-suggestions.d.ts +38 -0
- package/dist/components/chat/emoji-suggestions.d.ts.map +1 -0
- package/dist/components/chat/emoji-suggestions.js +1 -0
- package/dist/components/chat/emoji-suggestions.js.map +1 -0
- package/dist/components/chat/mention-suggestions.d.ts +9 -0
- package/dist/components/chat/mention-suggestions.d.ts.map +1 -0
- package/dist/components/chat/mention-suggestions.js +1 -0
- package/dist/components/chat/mention-suggestions.js.map +1 -0
- package/dist/components/chat/mod-view.d.ts +20 -0
- package/dist/components/chat/mod-view.d.ts.map +1 -0
- package/dist/components/chat/mod-view.js +5 -3
- package/dist/components/chat/mod-view.js.map +1 -0
- package/dist/components/chat/system-message.d.ts +7 -0
- package/dist/components/chat/system-message.d.ts.map +1 -0
- package/dist/components/chat/system-message.js +1 -0
- package/dist/components/chat/system-message.js.map +1 -0
- package/dist/components/content-metadata/content-metadata-form.d.ts +9 -0
- package/dist/components/content-metadata/content-metadata-form.d.ts.map +1 -0
- package/dist/components/content-metadata/content-metadata-form.js +4 -2
- package/dist/components/content-metadata/content-metadata-form.js.map +1 -0
- package/dist/components/content-metadata/content-rights.d.ts +14 -0
- package/dist/components/content-metadata/content-rights.d.ts.map +1 -0
- package/dist/components/content-metadata/content-rights.js +1 -0
- package/dist/components/content-metadata/content-rights.js.map +1 -0
- package/dist/components/content-metadata/content-warning-badge.d.ts +5 -0
- package/dist/components/content-metadata/content-warning-badge.d.ts.map +1 -0
- package/dist/components/content-metadata/content-warning-badge.js +1 -0
- package/dist/components/content-metadata/content-warning-badge.js.map +1 -0
- package/dist/components/content-metadata/content-warnings.d.ts +7 -0
- package/dist/components/content-metadata/content-warnings.d.ts.map +1 -0
- package/dist/components/content-metadata/content-warnings.js +1 -0
- package/dist/components/content-metadata/content-warnings.js.map +1 -0
- package/dist/components/content-metadata/index.d.ts +8 -0
- package/dist/components/content-metadata/index.d.ts.map +1 -0
- package/dist/components/content-metadata/index.js +1 -0
- package/dist/components/content-metadata/index.js.map +1 -0
- package/dist/components/danmu/danmu-message.d.ts +17 -0
- package/dist/components/danmu/danmu-message.d.ts.map +1 -0
- package/dist/components/danmu/danmu-message.js +112 -0
- package/dist/components/danmu/danmu-message.js.map +1 -0
- package/dist/components/danmu/danmu-overlay-obs.d.ts +11 -0
- package/dist/components/danmu/danmu-overlay-obs.d.ts.map +1 -0
- package/dist/components/danmu/danmu-overlay-obs.js +21 -0
- package/dist/components/danmu/danmu-overlay-obs.js.map +1 -0
- package/dist/components/danmu/danmu-overlay.d.ts +10 -0
- package/dist/components/danmu/danmu-overlay.d.ts.map +1 -0
- package/dist/components/danmu/danmu-overlay.js +150 -0
- package/dist/components/danmu/danmu-overlay.js.map +1 -0
- package/dist/components/danmu/math.d.ts +10 -0
- package/dist/components/danmu/math.d.ts.map +1 -0
- package/dist/components/danmu/math.js +16 -0
- package/dist/components/danmu/math.js.map +1 -0
- package/dist/components/danmu/mu.d.ts +9 -0
- package/dist/components/danmu/mu.d.ts.map +1 -0
- package/dist/components/danmu/mu.js +9 -0
- package/dist/components/danmu/mu.js.map +1 -0
- package/dist/components/danmu/use-danmu-lanes.d.ts +19 -0
- package/dist/components/danmu/use-danmu-lanes.d.ts.map +1 -0
- package/dist/components/danmu/use-danmu-lanes.js +77 -0
- package/dist/components/danmu/use-danmu-lanes.js.map +1 -0
- package/dist/components/dashboard/chat-panel.d.ts +9 -0
- package/dist/components/dashboard/chat-panel.d.ts.map +1 -0
- package/dist/components/dashboard/chat-panel.js +1 -0
- package/dist/components/dashboard/chat-panel.js.map +1 -0
- package/dist/components/dashboard/header.d.ts +14 -0
- package/dist/components/dashboard/header.d.ts.map +1 -0
- package/dist/components/dashboard/header.js +1 -0
- package/dist/components/dashboard/header.js.map +1 -0
- package/dist/components/dashboard/index.d.ts +6 -0
- package/dist/components/dashboard/index.d.ts.map +1 -0
- package/dist/components/dashboard/index.js +1 -0
- package/dist/components/dashboard/index.js.map +1 -0
- package/dist/components/dashboard/information-widget.d.ts +9 -0
- package/dist/components/dashboard/information-widget.d.ts.map +1 -0
- package/dist/components/dashboard/information-widget.js +1 -0
- package/dist/components/dashboard/information-widget.js.map +1 -0
- package/dist/components/dashboard/mod-actions.d.ts +15 -0
- package/dist/components/dashboard/mod-actions.d.ts.map +1 -0
- package/dist/components/dashboard/mod-actions.js +1 -0
- package/dist/components/dashboard/mod-actions.js.map +1 -0
- package/dist/components/dashboard/problems.d.ts +13 -0
- package/dist/components/dashboard/problems.d.ts.map +1 -0
- package/dist/components/dashboard/problems.js +1 -0
- package/dist/components/dashboard/problems.js.map +1 -0
- package/dist/components/icons/bluesky-icon.d.ts +5 -0
- package/dist/components/icons/bluesky-icon.d.ts.map +1 -0
- package/dist/components/icons/bluesky-icon.js +1 -0
- package/dist/components/icons/bluesky-icon.js.map +1 -0
- package/dist/components/keep-awake.d.ts +2 -0
- package/dist/components/keep-awake.d.ts.map +1 -0
- package/dist/components/keep-awake.js +1 -0
- package/dist/components/keep-awake.js.map +1 -0
- package/dist/components/keep-awake.native.d.ts +2 -0
- package/dist/components/keep-awake.native.d.ts.map +1 -0
- package/dist/components/keep-awake.native.js +1 -0
- package/dist/components/keep-awake.native.js.map +1 -0
- package/dist/components/mobile-player/fullscreen.d.ts +7 -0
- package/dist/components/mobile-player/fullscreen.d.ts.map +1 -0
- package/dist/components/mobile-player/fullscreen.js +7 -1
- package/dist/components/mobile-player/fullscreen.js.map +1 -0
- package/dist/components/mobile-player/fullscreen.native.d.ts +7 -0
- package/dist/components/mobile-player/fullscreen.native.d.ts.map +1 -0
- package/dist/components/mobile-player/fullscreen.native.js +8 -2
- package/dist/components/mobile-player/fullscreen.native.js.map +1 -0
- package/dist/components/mobile-player/player.d.ts +8 -0
- package/dist/components/mobile-player/player.d.ts.map +1 -0
- package/dist/components/mobile-player/player.js +1 -0
- package/dist/components/mobile-player/player.js.map +1 -0
- package/dist/components/mobile-player/props.d.ts +19 -0
- package/dist/components/mobile-player/props.d.ts.map +1 -0
- package/dist/components/mobile-player/props.js +1 -0
- package/dist/components/mobile-player/props.js.map +1 -0
- package/dist/components/mobile-player/rotation-async.native.d.ts +4 -0
- package/dist/components/mobile-player/rotation-async.native.d.ts.map +1 -0
- package/dist/components/mobile-player/rotation-async.native.js +1 -0
- package/dist/components/mobile-player/rotation-async.native.js.map +1 -0
- package/dist/components/mobile-player/rotation-lock.d.ts +18 -0
- package/dist/components/mobile-player/rotation-lock.d.ts.map +1 -0
- package/dist/components/mobile-player/rotation-lock.js +1 -0
- package/dist/components/mobile-player/rotation-lock.js.map +1 -0
- package/dist/components/mobile-player/shared.d.ts +9 -0
- package/dist/components/mobile-player/shared.d.ts.map +1 -0
- package/dist/components/mobile-player/shared.js +1 -0
- package/dist/components/mobile-player/shared.js.map +1 -0
- package/dist/components/mobile-player/ui/autoplay-button.d.ts +2 -0
- package/dist/components/mobile-player/ui/autoplay-button.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/autoplay-button.js +1 -0
- package/dist/components/mobile-player/ui/autoplay-button.js.map +1 -0
- package/dist/components/mobile-player/ui/countdown.d.ts +10 -0
- package/dist/components/mobile-player/ui/countdown.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/countdown.js +1 -0
- package/dist/components/mobile-player/ui/countdown.js.map +1 -0
- package/dist/components/mobile-player/ui/index.d.ts +11 -0
- package/dist/components/mobile-player/ui/index.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/index.js +2 -0
- package/dist/components/mobile-player/ui/index.js.map +1 -0
- package/dist/components/mobile-player/ui/input.d.ts +9 -0
- package/dist/components/mobile-player/ui/input.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/input.js +1 -0
- package/dist/components/mobile-player/ui/input.js.map +1 -0
- package/dist/components/mobile-player/ui/metrics.d.ts +6 -0
- package/dist/components/mobile-player/ui/metrics.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/metrics.js +1 -0
- package/dist/components/mobile-player/ui/metrics.js.map +1 -0
- package/dist/components/mobile-player/ui/report-modal.d.ts +13 -0
- package/dist/components/mobile-player/ui/report-modal.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/report-modal.js +1 -0
- package/dist/components/mobile-player/ui/report-modal.js.map +1 -0
- package/dist/components/mobile-player/ui/streamer-context-menu.d.ts +2 -0
- package/dist/components/mobile-player/ui/streamer-context-menu.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/streamer-context-menu.js +1 -0
- package/dist/components/mobile-player/ui/streamer-context-menu.js.map +1 -0
- package/dist/components/mobile-player/ui/streamer-loading-overlay.d.ts +11 -0
- package/dist/components/mobile-player/ui/streamer-loading-overlay.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/streamer-loading-overlay.js +1 -0
- package/dist/components/mobile-player/ui/streamer-loading-overlay.js.map +1 -0
- package/dist/components/mobile-player/ui/viewer-context-menu.d.ts +9 -0
- package/dist/components/mobile-player/ui/viewer-context-menu.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/viewer-context-menu.js +6 -5
- package/dist/components/mobile-player/ui/viewer-context-menu.js.map +1 -0
- package/dist/components/mobile-player/ui/viewer-count.d.ts +10 -0
- package/dist/components/mobile-player/ui/viewer-count.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/viewer-count.js +24 -0
- package/dist/components/mobile-player/ui/viewer-count.js.map +1 -0
- package/dist/components/mobile-player/ui/viewer-loading-overlay.d.ts +2 -0
- package/dist/components/mobile-player/ui/viewer-loading-overlay.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/viewer-loading-overlay.js +1 -0
- package/dist/components/mobile-player/ui/viewer-loading-overlay.js.map +1 -0
- package/dist/components/mobile-player/ui/viewers.d.ts +6 -0
- package/dist/components/mobile-player/ui/viewers.d.ts.map +1 -0
- package/dist/components/mobile-player/ui/viewers.js +9 -7
- package/dist/components/mobile-player/ui/viewers.js.map +1 -0
- package/dist/components/mobile-player/use-webrtc.d.ts +19 -0
- package/dist/components/mobile-player/use-webrtc.d.ts.map +1 -0
- package/dist/components/mobile-player/use-webrtc.js +1 -0
- package/dist/components/mobile-player/use-webrtc.js.map +1 -0
- package/dist/components/mobile-player/video-async.native.d.ts +16 -0
- package/dist/components/mobile-player/video-async.native.d.ts.map +1 -0
- package/dist/components/mobile-player/video-async.native.js +1 -0
- package/dist/components/mobile-player/video-async.native.js.map +1 -0
- package/dist/components/mobile-player/video-retry.d.ts +5 -0
- package/dist/components/mobile-player/video-retry.d.ts.map +1 -0
- package/dist/components/mobile-player/video-retry.js +1 -0
- package/dist/components/mobile-player/video-retry.js.map +1 -0
- package/dist/components/mobile-player/video.d.ts +25 -0
- package/dist/components/mobile-player/video.d.ts.map +1 -0
- package/dist/components/mobile-player/video.js +1 -0
- package/dist/components/mobile-player/video.js.map +1 -0
- package/dist/components/mobile-player/video.native.d.ts +3 -0
- package/dist/components/mobile-player/video.native.d.ts.map +1 -0
- package/dist/components/mobile-player/video.native.js +1 -0
- package/dist/components/mobile-player/video.native.js.map +1 -0
- package/dist/components/mobile-player/webrtc-diagnostics.d.ts +14 -0
- package/dist/components/mobile-player/webrtc-diagnostics.d.ts.map +1 -0
- package/dist/components/mobile-player/webrtc-diagnostics.js +1 -0
- package/dist/components/mobile-player/webrtc-diagnostics.js.map +1 -0
- package/dist/components/mobile-player/webrtc-primitives.d.ts +19 -0
- package/dist/components/mobile-player/webrtc-primitives.d.ts.map +1 -0
- package/dist/components/mobile-player/webrtc-primitives.js +1 -0
- package/dist/components/mobile-player/webrtc-primitives.js.map +1 -0
- package/dist/components/mobile-player/webrtc-primitives.native.d.ts +2 -0
- package/dist/components/mobile-player/webrtc-primitives.native.d.ts.map +1 -0
- package/dist/components/mobile-player/webrtc-primitives.native.js +1 -0
- package/dist/components/mobile-player/webrtc-primitives.native.js.map +1 -0
- package/dist/components/share/sharesheet.d.ts +5 -0
- package/dist/components/share/sharesheet.d.ts.map +1 -0
- package/dist/components/share/sharesheet.js +6 -14
- package/dist/components/share/sharesheet.js.map +1 -0
- package/dist/components/ui/button.d.ts +18 -0
- package/dist/components/ui/button.d.ts.map +1 -0
- package/dist/components/ui/button.js +14 -2
- package/dist/components/ui/button.js.map +1 -0
- package/dist/components/ui/checkbox.d.ts +11 -0
- package/dist/components/ui/checkbox.d.ts.map +1 -0
- package/dist/components/ui/checkbox.js +1 -0
- package/dist/components/ui/checkbox.js.map +1 -0
- package/dist/components/ui/dialog.d.ts +38 -0
- package/dist/components/ui/dialog.d.ts.map +1 -0
- package/dist/components/ui/dialog.js +1 -0
- package/dist/components/ui/dialog.js.map +1 -0
- package/dist/components/ui/dropdown.d.ts +132 -0
- package/dist/components/ui/dropdown.d.ts.map +1 -0
- package/dist/components/ui/dropdown.js +28 -238
- package/dist/components/ui/dropdown.js.map +1 -0
- package/dist/components/ui/dropdown.native.d.ts +132 -0
- package/dist/components/ui/dropdown.native.d.ts.map +1 -0
- package/dist/components/ui/dropdown.native.js +319 -0
- package/dist/components/ui/dropdown.native.js.map +1 -0
- package/dist/components/ui/icons.d.ts +12 -0
- package/dist/components/ui/icons.d.ts.map +1 -0
- package/dist/components/ui/icons.js +1 -0
- package/dist/components/ui/icons.js.map +1 -0
- package/dist/components/ui/index.d.ts +33 -0
- package/dist/components/ui/index.d.ts.map +1 -0
- package/dist/components/ui/index.js +2 -0
- package/dist/components/ui/index.js.map +1 -0
- package/dist/components/ui/info-box.d.ts +9 -0
- package/dist/components/ui/info-box.d.ts.map +1 -0
- package/dist/components/ui/info-box.js +1 -0
- package/dist/components/ui/info-box.js.map +1 -0
- package/dist/components/ui/info-row.d.ts +9 -0
- package/dist/components/ui/info-row.d.ts.map +1 -0
- package/dist/components/ui/info-row.js +1 -0
- package/dist/components/ui/info-row.js.map +1 -0
- package/dist/components/ui/input.d.ts +20 -0
- package/dist/components/ui/input.d.ts.map +1 -0
- package/dist/components/ui/input.js +1 -0
- package/dist/components/ui/input.js.map +1 -0
- package/dist/components/ui/loader.d.ts +3 -0
- package/dist/components/ui/loader.d.ts.map +1 -0
- package/dist/components/ui/loader.js +1 -0
- package/dist/components/ui/loader.js.map +1 -0
- package/dist/components/ui/menu.d.ts +34 -0
- package/dist/components/ui/menu.d.ts.map +1 -0
- package/dist/components/ui/menu.js +57 -0
- package/dist/components/ui/menu.js.map +1 -0
- package/dist/components/ui/primitives/button.d.ts +41 -0
- package/dist/components/ui/primitives/button.d.ts.map +1 -0
- package/dist/components/ui/primitives/button.js +1 -0
- package/dist/components/ui/primitives/button.js.map +1 -0
- package/dist/components/ui/primitives/input.d.ts +52 -0
- package/dist/components/ui/primitives/input.d.ts.map +1 -0
- package/dist/components/ui/primitives/input.js +1 -0
- package/dist/components/ui/primitives/input.js.map +1 -0
- package/dist/components/ui/primitives/modal.d.ts +46 -0
- package/dist/components/ui/primitives/modal.d.ts.map +1 -0
- package/dist/components/ui/primitives/modal.js +1 -0
- package/dist/components/ui/primitives/modal.js.map +1 -0
- package/dist/components/ui/primitives/text.d.ts +48 -0
- package/dist/components/ui/primitives/text.d.ts.map +1 -0
- package/dist/components/ui/primitives/text.js +1 -0
- package/dist/components/ui/primitives/text.js.map +1 -0
- package/dist/components/ui/resizeable.d.ts +14 -0
- package/dist/components/ui/resizeable.d.ts.map +1 -0
- package/dist/components/ui/resizeable.js +2 -1
- package/dist/components/ui/resizeable.js.map +1 -0
- package/dist/components/ui/select.d.ts +16 -0
- package/dist/components/ui/select.d.ts.map +1 -0
- package/dist/components/ui/select.js +1 -0
- package/dist/components/ui/select.js.map +1 -0
- package/dist/components/ui/slider.d.ts +2 -0
- package/dist/components/ui/slider.d.ts.map +1 -0
- package/dist/components/ui/slider.js +1 -0
- package/dist/components/ui/slider.js.map +1 -0
- package/dist/components/ui/text.d.ts +45 -0
- package/dist/components/ui/text.d.ts.map +1 -0
- package/dist/components/ui/text.js +1 -0
- package/dist/components/ui/text.js.map +1 -0
- package/dist/components/ui/textarea.d.ts +5 -0
- package/dist/components/ui/textarea.d.ts.map +1 -0
- package/dist/components/ui/textarea.js +1 -0
- package/dist/components/ui/textarea.js.map +1 -0
- package/dist/components/ui/toast.d.ts +92 -0
- package/dist/components/ui/toast.d.ts.map +1 -0
- package/dist/components/ui/toast.js +2 -1
- package/dist/components/ui/toast.js.map +1 -0
- package/dist/components/ui/tooltip.d.ts +8 -0
- package/dist/components/ui/tooltip.d.ts.map +1 -0
- package/dist/components/ui/tooltip.js +1 -0
- package/dist/components/ui/tooltip.js.map +1 -0
- package/dist/components/ui/view.d.ts +33 -0
- package/dist/components/ui/view.d.ts.map +1 -0
- package/dist/components/ui/view.js +1 -0
- package/dist/components/ui/view.js.map +1 -0
- package/dist/crypto-polyfill.d.ts +1 -0
- package/dist/crypto-polyfill.d.ts.map +1 -0
- package/dist/crypto-polyfill.js +1 -0
- package/dist/crypto-polyfill.js.map +1 -0
- package/dist/crypto-polyfill.native.d.ts +3 -0
- package/dist/crypto-polyfill.native.d.ts.map +1 -0
- package/dist/crypto-polyfill.native.js +1 -0
- package/dist/crypto-polyfill.native.js.map +1 -0
- package/dist/hooks/index.d.ts +11 -0
- package/dist/hooks/index.d.ts.map +1 -0
- package/dist/hooks/index.js +1 -0
- package/dist/hooks/index.js.map +1 -0
- package/dist/hooks/useAvatars.d.ts +3 -0
- package/dist/hooks/useAvatars.d.ts.map +1 -0
- package/dist/hooks/useAvatars.js +1 -0
- package/dist/hooks/useAvatars.js.map +1 -0
- package/dist/hooks/useCameraToggle.d.ts +5 -0
- package/dist/hooks/useCameraToggle.d.ts.map +1 -0
- package/dist/hooks/useCameraToggle.js +1 -0
- package/dist/hooks/useCameraToggle.js.map +1 -0
- package/dist/hooks/useKeyboard.d.ts +5 -0
- package/dist/hooks/useKeyboard.d.ts.map +1 -0
- package/dist/hooks/useKeyboard.js +1 -0
- package/dist/hooks/useKeyboard.js.map +1 -0
- package/dist/hooks/useKeyboardSlide.d.ts +5 -0
- package/dist/hooks/useKeyboardSlide.d.ts.map +1 -0
- package/dist/hooks/useKeyboardSlide.js +1 -0
- package/dist/hooks/useKeyboardSlide.js.map +1 -0
- package/dist/hooks/useLivestreamInfo.d.ts +15 -0
- package/dist/hooks/useLivestreamInfo.d.ts.map +1 -0
- package/dist/hooks/useLivestreamInfo.js +1 -0
- package/dist/hooks/useLivestreamInfo.js.map +1 -0
- package/dist/hooks/useOuterAndInnerDimensions.d.ts +10 -0
- package/dist/hooks/useOuterAndInnerDimensions.d.ts.map +1 -0
- package/dist/hooks/useOuterAndInnerDimensions.js +1 -0
- package/dist/hooks/useOuterAndInnerDimensions.js.map +1 -0
- package/dist/hooks/usePlayerDimensions.d.ts +12 -0
- package/dist/hooks/usePlayerDimensions.d.ts.map +1 -0
- package/dist/hooks/usePlayerDimensions.js +1 -0
- package/dist/hooks/usePlayerDimensions.js.map +1 -0
- package/dist/hooks/usePointerDevice.d.ts +12 -0
- package/dist/hooks/usePointerDevice.d.ts.map +1 -0
- package/dist/hooks/usePointerDevice.js +1 -0
- package/dist/hooks/usePointerDevice.js.map +1 -0
- package/dist/hooks/useSegmentDimensions.d.ts +6 -0
- package/dist/hooks/useSegmentDimensions.d.ts.map +1 -0
- package/dist/hooks/useSegmentDimensions.js +1 -0
- package/dist/hooks/useSegmentDimensions.js.map +1 -0
- package/dist/hooks/useSegmentTiming.d.ts +9 -0
- package/dist/hooks/useSegmentTiming.d.ts.map +1 -0
- package/dist/hooks/useSegmentTiming.js +1 -0
- package/dist/hooks/useSegmentTiming.js.map +1 -0
- package/dist/i18n/i18n-loader.d.ts +2 -0
- package/dist/i18n/i18n-loader.d.ts.map +1 -0
- package/dist/i18n/i18n-loader.js +17 -0
- package/dist/i18n/i18n-loader.js.map +1 -0
- package/dist/i18n/i18n-loader.native.d.ts +2 -0
- package/dist/i18n/i18n-loader.native.d.ts.map +1 -0
- package/dist/i18n/i18n-loader.native.js +51 -0
- package/dist/i18n/i18n-loader.native.js.map +1 -0
- package/dist/i18n/i18next-config.d.ts +41 -0
- package/dist/i18n/i18next-config.d.ts.map +1 -0
- package/dist/i18n/i18next-config.js +220 -0
- package/dist/i18n/i18next-config.js.map +1 -0
- package/dist/i18n/index.d.ts +11 -0
- package/dist/i18n/index.d.ts.map +1 -0
- package/dist/i18n/index.js +45 -0
- package/dist/i18n/index.js.map +1 -0
- package/dist/i18n/provider.d.ts +14 -0
- package/dist/i18n/provider.d.ts.map +1 -0
- package/dist/i18n/provider.js +58 -0
- package/dist/i18n/provider.js.map +1 -0
- package/dist/index.d.ts +34 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +10 -1
- package/dist/index.js.map +1 -0
- package/dist/lib/browser.d.ts +2 -0
- package/dist/lib/browser.d.ts.map +1 -0
- package/dist/lib/browser.js +1 -0
- package/dist/lib/browser.js.map +1 -0
- package/dist/lib/facet.d.ts +11 -0
- package/dist/lib/facet.d.ts.map +1 -0
- package/dist/lib/facet.js +1 -0
- package/dist/lib/facet.js.map +1 -0
- package/dist/lib/metadata-constants.d.ts +13 -0
- package/dist/lib/metadata-constants.d.ts.map +1 -0
- package/dist/lib/metadata-constants.js +1 -0
- package/dist/lib/metadata-constants.js.map +1 -0
- package/dist/lib/system-messages.d.ts +49 -0
- package/dist/lib/system-messages.d.ts.map +1 -0
- package/dist/lib/system-messages.js +1 -0
- package/dist/lib/system-messages.js.map +1 -0
- package/dist/lib/theme/atoms.d.ts +4226 -0
- package/dist/lib/theme/atoms.d.ts.map +1 -0
- package/dist/lib/theme/atoms.js +1 -0
- package/dist/lib/theme/atoms.js.map +1 -0
- package/dist/lib/theme/atoms.types.d.ts +228 -0
- package/dist/lib/theme/atoms.types.d.ts.map +1 -0
- package/dist/lib/theme/atoms.types.js +1 -0
- package/dist/lib/theme/atoms.types.js.map +1 -0
- package/dist/lib/theme/index.d.ts +6 -0
- package/dist/lib/theme/index.d.ts.map +1 -0
- package/dist/lib/theme/index.js +1 -0
- package/dist/lib/theme/index.js.map +1 -0
- package/dist/lib/theme/theme.d.ts +327 -0
- package/dist/lib/theme/theme.d.ts.map +1 -0
- package/dist/lib/theme/theme.js +1 -0
- package/dist/lib/theme/theme.js.map +1 -0
- package/dist/lib/theme/tokens.d.ts +766 -0
- package/dist/lib/theme/tokens.d.ts.map +1 -0
- package/dist/lib/theme/tokens.js +1 -0
- package/dist/lib/theme/tokens.js.map +1 -0
- package/dist/lib/utils.d.ts +53 -0
- package/dist/lib/utils.d.ts.map +1 -0
- package/dist/lib/utils.js +1 -0
- package/dist/lib/utils.js.map +1 -0
- package/dist/livestream-provider/index.d.ts +13 -0
- package/dist/livestream-provider/index.d.ts.map +1 -0
- package/dist/livestream-provider/index.js +1 -0
- package/dist/livestream-provider/index.js.map +1 -0
- package/dist/livestream-provider/websocket.d.ts +2 -0
- package/dist/livestream-provider/websocket.d.ts.map +1 -0
- package/dist/livestream-provider/websocket.js +1 -0
- package/dist/livestream-provider/websocket.js.map +1 -0
- package/dist/livestream-store/chat.d.ts +21 -0
- package/dist/livestream-store/chat.d.ts.map +1 -0
- package/dist/livestream-store/chat.js +1 -0
- package/dist/livestream-store/chat.js.map +1 -0
- package/dist/livestream-store/context.d.ts +7 -0
- package/dist/livestream-store/context.d.ts.map +1 -0
- package/dist/livestream-store/context.js +1 -0
- package/dist/livestream-store/context.js.map +1 -0
- package/dist/livestream-store/index.d.ts +5 -0
- package/dist/livestream-store/index.d.ts.map +1 -0
- package/dist/livestream-store/index.js +1 -0
- package/dist/livestream-store/index.js.map +1 -0
- package/dist/livestream-store/livestream-state.d.ts +29 -0
- package/dist/livestream-store/livestream-state.d.ts.map +1 -0
- package/dist/livestream-store/livestream-state.js +1 -0
- package/dist/livestream-store/livestream-state.js.map +1 -0
- package/dist/livestream-store/livestream-store.d.ts +14 -0
- package/dist/livestream-store/livestream-store.d.ts.map +1 -0
- package/dist/livestream-store/livestream-store.js +1 -0
- package/dist/livestream-store/livestream-store.js.map +1 -0
- package/dist/livestream-store/problems.d.ts +4 -0
- package/dist/livestream-store/problems.d.ts.map +1 -0
- package/dist/livestream-store/problems.js +1 -0
- package/dist/livestream-store/problems.js.map +1 -0
- package/dist/livestream-store/stream-key.d.ts +9 -0
- package/dist/livestream-store/stream-key.d.ts.map +1 -0
- package/dist/livestream-store/stream-key.js +1 -0
- package/dist/livestream-store/stream-key.js.map +1 -0
- package/dist/livestream-store/websocket-consumer.d.ts +3 -0
- package/dist/livestream-store/websocket-consumer.d.ts.map +1 -0
- package/dist/livestream-store/websocket-consumer.js +1 -0
- package/dist/livestream-store/websocket-consumer.js.map +1 -0
- package/dist/player-store/context.d.ts +10 -0
- package/dist/player-store/context.d.ts.map +1 -0
- package/dist/player-store/context.js +1 -0
- package/dist/player-store/context.js.map +1 -0
- package/dist/player-store/index.d.ts +6 -0
- package/dist/player-store/index.d.ts.map +1 -0
- package/dist/player-store/index.js +1 -0
- package/dist/player-store/index.js.map +1 -0
- package/dist/player-store/player-provider.d.ts +12 -0
- package/dist/player-store/player-provider.d.ts.map +1 -0
- package/dist/player-store/player-provider.js +1 -0
- package/dist/player-store/player-provider.js.map +1 -0
- package/dist/player-store/player-state.d.ts +138 -0
- package/dist/player-store/player-state.d.ts.map +1 -0
- package/dist/player-store/player-state.js +1 -0
- package/dist/player-store/player-state.js.map +1 -0
- package/dist/player-store/player-store.d.ts +18 -0
- package/dist/player-store/player-store.d.ts.map +1 -0
- package/dist/player-store/player-store.js +1 -0
- package/dist/player-store/player-store.js.map +1 -0
- package/dist/player-store/single-player-provider.d.ts +61 -0
- package/dist/player-store/single-player-provider.d.ts.map +1 -0
- package/dist/player-store/single-player-provider.js +1 -0
- package/dist/player-store/single-player-provider.js.map +1 -0
- package/dist/storage/index.d.ts +4 -0
- package/dist/storage/index.d.ts.map +1 -0
- package/dist/storage/index.js +1 -0
- package/dist/storage/index.js.map +1 -0
- package/dist/storage/lock.d.ts +8 -0
- package/dist/storage/lock.d.ts.map +1 -0
- package/dist/storage/lock.js +1 -0
- package/dist/storage/lock.js.map +1 -0
- package/dist/storage/storage.d.ts +7 -0
- package/dist/storage/storage.d.ts.map +1 -0
- package/dist/storage/storage.js +1 -0
- package/dist/storage/storage.js.map +1 -0
- package/dist/storage/storage.native.d.ts +7 -0
- package/dist/storage/storage.native.d.ts.map +1 -0
- package/dist/storage/storage.native.js +1 -0
- package/dist/storage/storage.native.js.map +1 -0
- package/dist/storage/storage.shared.d.ts +6 -0
- package/dist/storage/storage.shared.d.ts.map +1 -0
- package/dist/storage/storage.shared.js +1 -0
- package/dist/storage/storage.shared.js.map +1 -0
- package/dist/streamplace-provider/context.d.ts +7 -0
- package/dist/streamplace-provider/context.d.ts.map +1 -0
- package/dist/streamplace-provider/context.js +1 -0
- package/dist/streamplace-provider/context.js.map +1 -0
- package/dist/streamplace-provider/index.d.ts +11 -0
- package/dist/streamplace-provider/index.d.ts.map +1 -0
- package/dist/streamplace-provider/index.js +1 -0
- package/dist/streamplace-provider/index.js.map +1 -0
- package/dist/streamplace-provider/poller.d.ts +5 -0
- package/dist/streamplace-provider/poller.d.ts.map +1 -0
- package/dist/streamplace-provider/poller.js +3 -0
- package/dist/streamplace-provider/poller.js.map +1 -0
- package/dist/streamplace-provider/xrpc.d.ts +1 -0
- package/dist/streamplace-provider/xrpc.d.ts.map +1 -0
- package/dist/streamplace-provider/xrpc.js +1 -0
- package/dist/streamplace-provider/xrpc.js.map +1 -0
- package/dist/streamplace-store/block.d.ts +9 -0
- package/dist/streamplace-store/block.d.ts.map +1 -0
- package/dist/streamplace-store/block.js +1 -0
- package/dist/streamplace-store/block.js.map +1 -0
- package/dist/streamplace-store/content-metadata-actions.d.ts +9 -0
- package/dist/streamplace-store/content-metadata-actions.d.ts.map +1 -0
- package/dist/streamplace-store/content-metadata-actions.js +1 -0
- package/dist/streamplace-store/content-metadata-actions.js.map +1 -0
- package/dist/streamplace-store/graph.d.ts +22 -0
- package/dist/streamplace-store/graph.d.ts.map +1 -0
- package/dist/streamplace-store/graph.js +1 -0
- package/dist/streamplace-store/graph.js.map +1 -0
- package/dist/streamplace-store/index.d.ts +4 -0
- package/dist/streamplace-store/index.d.ts.map +1 -0
- package/dist/streamplace-store/index.js +1 -0
- package/dist/streamplace-store/index.js.map +1 -0
- package/dist/streamplace-store/stream.d.ts +11 -0
- package/dist/streamplace-store/stream.d.ts.map +1 -0
- package/dist/streamplace-store/stream.js +1 -0
- package/dist/streamplace-store/stream.js.map +1 -0
- package/dist/streamplace-store/streamplace-store.d.ts +92 -0
- package/dist/streamplace-store/streamplace-store.d.ts.map +1 -0
- package/dist/streamplace-store/streamplace-store.js +158 -3
- package/dist/streamplace-store/streamplace-store.js.map +1 -0
- package/dist/streamplace-store/user.d.ts +6 -0
- package/dist/streamplace-store/user.d.ts.map +1 -0
- package/dist/streamplace-store/user.js +1 -0
- package/dist/streamplace-store/user.js.map +1 -0
- package/dist/streamplace-store/xrpc.d.ts +3 -0
- package/dist/streamplace-store/xrpc.d.ts.map +1 -0
- package/dist/streamplace-store/xrpc.js +1 -0
- package/dist/streamplace-store/xrpc.js.map +1 -0
- package/dist/time-sync/index.d.ts +3 -0
- package/dist/time-sync/index.d.ts.map +1 -0
- package/dist/time-sync/index.js +15 -0
- package/dist/time-sync/index.js.map +1 -0
- package/dist/time-sync/time-sync.d.ts +13 -0
- package/dist/time-sync/time-sync.d.ts.map +1 -0
- package/dist/time-sync/time-sync.js +95 -0
- package/dist/time-sync/time-sync.js.map +1 -0
- package/dist/time-sync/useTimeSync.d.ts +2 -0
- package/dist/time-sync/useTimeSync.d.ts.map +1 -0
- package/dist/time-sync/useTimeSync.js +49 -0
- package/dist/time-sync/useTimeSync.js.map +1 -0
- package/dist/ui/index.d.ts +15 -0
- package/dist/ui/index.d.ts.map +1 -0
- package/dist/ui/index.js +1 -0
- package/dist/ui/index.js.map +1 -0
- package/dist/utils/format-handle.d.ts +11 -0
- package/dist/utils/format-handle.d.ts.map +1 -0
- package/dist/utils/format-handle.js +21 -0
- package/dist/utils/format-handle.js.map +1 -0
- package/locales/en-US/common.ftl +46 -0
- package/locales/en-US/settings.ftl +129 -0
- package/locales/es-ES/common.ftl +46 -0
- package/locales/es-ES/settings.ftl +168 -0
- package/locales/fr-FR/common.ftl +46 -0
- package/locales/fr-FR/settings.ftl +165 -0
- package/locales/manifest.json +36 -0
- package/locales/pt-BR/common.ftl +46 -0
- package/locales/pt-BR/settings.ftl +166 -0
- package/locales/zh-Hant/common.ftl +46 -0
- package/locales/zh-Hant/settings.ftl +161 -0
- package/node-compile-cache/v22.15.0-x64-efe9a9df-0/37be0eec +0 -0
- package/package.json +24 -3
- package/scripts/compile-translations.js +254 -0
- package/scripts/extract-i18n.js +336 -0
- package/src/components/chat/chat-box.tsx +28 -9
- package/src/components/chat/chat-message.tsx +3 -2
- package/src/components/chat/mod-view.tsx +4 -3
- package/src/components/content-metadata/content-metadata-form.tsx +3 -1
- package/src/components/danmu/danmu-message.tsx +182 -0
- package/src/components/danmu/danmu-overlay-obs.tsx +44 -0
- package/src/components/danmu/danmu-overlay.tsx +225 -0
- package/src/components/danmu/math.ts +27 -0
- package/src/components/danmu/mu.tsx +28 -0
- package/src/components/danmu/use-danmu-lanes.ts +114 -0
- package/src/components/mobile-player/fullscreen.native.tsx +26 -0
- package/src/components/mobile-player/fullscreen.tsx +23 -1
- package/src/components/mobile-player/ui/index.ts +1 -0
- package/src/components/mobile-player/ui/viewer-context-menu.tsx +9 -7
- package/src/components/mobile-player/ui/viewer-count.tsx +39 -0
- package/src/components/mobile-player/ui/viewers.tsx +10 -14
- package/src/components/share/sharesheet.tsx +11 -27
- package/src/components/ui/button.tsx +22 -4
- package/src/components/ui/dropdown.native.tsx +696 -0
- package/src/components/ui/dropdown.tsx +57 -427
- package/src/components/ui/index.ts +1 -0
- package/src/components/ui/menu.tsx +171 -0
- package/src/components/ui/resizeable.tsx +1 -1
- package/src/components/ui/toast.tsx +6 -1
- package/src/i18n/i18n-loader.native.ts +56 -0
- package/src/i18n/i18n-loader.ts +19 -0
- package/src/i18n/i18next-config.ts +250 -0
- package/src/i18n/index.ts +50 -0
- package/src/i18n/provider.tsx +70 -0
- package/src/index.tsx +9 -0
- package/src/streamplace-provider/poller.tsx +3 -0
- package/src/streamplace-store/streamplace-store.tsx +184 -2
- package/src/time-sync/index.ts +12 -0
- package/src/time-sync/time-sync.ts +112 -0
- package/src/time-sync/useTimeSync.tsx +58 -0
- package/src/utils/format-handle.ts +24 -0
- package/tsconfig.json +11 -2
- package/tsconfig.tsbuildinfo +0 -1
|
@@ -12,6 +12,7 @@ import { Linking } from "react-native";
|
|
|
12
12
|
import { ChatMessageViewHydrated } from "streamplace";
|
|
13
13
|
import { useDeleteChatMessage } from "../../livestream-store";
|
|
14
14
|
import { useStreamplaceStore } from "../../streamplace-store";
|
|
15
|
+
import { formatHandle, formatHandleWithAt } from "../../utils/format-handle";
|
|
15
16
|
import {
|
|
16
17
|
atoms,
|
|
17
18
|
DropdownMenu,
|
|
@@ -113,7 +114,7 @@ export const ModView = forwardRef<ModViewRef, ModViewProps>(() => {
|
|
|
113
114
|
minute: "2-digit",
|
|
114
115
|
hour12: false,
|
|
115
116
|
})}{" "}
|
|
116
|
-
|
|
117
|
+
{formatHandleWithAt(message.author)}: {message.record.text}
|
|
117
118
|
</Text>
|
|
118
119
|
</View>
|
|
119
120
|
</DropdownMenuItem>
|
|
@@ -159,7 +160,7 @@ export const ModView = forwardRef<ModViewRef, ModViewProps>(() => {
|
|
|
159
160
|
<Text color="destructive">
|
|
160
161
|
{isBlockLoading
|
|
161
162
|
? "Blocking..."
|
|
162
|
-
: `Block user
|
|
163
|
+
: `Block user ${formatHandleWithAt(message.author)} from this channel`}
|
|
163
164
|
</Text>
|
|
164
165
|
)}
|
|
165
166
|
</DropdownMenuItem>
|
|
@@ -170,7 +171,7 @@ export const ModView = forwardRef<ModViewRef, ModViewProps>(() => {
|
|
|
170
171
|
<DropdownMenuItem
|
|
171
172
|
onPress={() => {
|
|
172
173
|
Linking.openURL(
|
|
173
|
-
`https://${BSKY_FRONTEND_DOMAIN}/profile/${message.author
|
|
174
|
+
`https://${BSKY_FRONTEND_DOMAIN}/profile/${formatHandle(message.author)}`,
|
|
174
175
|
);
|
|
175
176
|
}}
|
|
176
177
|
>
|
|
@@ -684,17 +684,19 @@ export const ContentMetadataForm = forwardRef<any, ContentMetadataFormProps>(
|
|
|
684
684
|
]}
|
|
685
685
|
>
|
|
686
686
|
<Text
|
|
687
|
+
leading="tight"
|
|
687
688
|
style={[
|
|
688
689
|
text.neutral[300],
|
|
689
690
|
{
|
|
690
691
|
minWidth: 100,
|
|
692
|
+
maxWidth: 100,
|
|
691
693
|
textAlign: "left",
|
|
692
694
|
paddingBottom: 8,
|
|
693
695
|
fontSize: 14,
|
|
694
696
|
},
|
|
695
697
|
]}
|
|
696
698
|
>
|
|
697
|
-
Allowed
|
|
699
|
+
Allowed Broadcasters
|
|
698
700
|
</Text>
|
|
699
701
|
<View style={[flex.values[1]]}>
|
|
700
702
|
<Text
|
|
@@ -0,0 +1,182 @@
|
|
|
1
|
+
import { memo, useEffect, useState } from "react";
|
|
2
|
+
import { LayoutChangeEvent, StyleSheet } from "react-native";
|
|
3
|
+
import Animated, {
|
|
4
|
+
Easing,
|
|
5
|
+
runOnJS,
|
|
6
|
+
useAnimatedStyle,
|
|
7
|
+
useSharedValue,
|
|
8
|
+
withTiming,
|
|
9
|
+
} from "react-native-reanimated";
|
|
10
|
+
import { ChatMessageViewHydrated } from "streamplace";
|
|
11
|
+
import { Text } from "../ui";
|
|
12
|
+
import { baseDuration, mapRange, MAX_DURATION, MIN_DURATION } from "./math";
|
|
13
|
+
|
|
14
|
+
interface DanmuMessageProps {
|
|
15
|
+
message: ChatMessageViewHydrated;
|
|
16
|
+
lane: number;
|
|
17
|
+
laneHeight: number;
|
|
18
|
+
videoTop: number;
|
|
19
|
+
opacity: number;
|
|
20
|
+
fontSize: number;
|
|
21
|
+
speed: number;
|
|
22
|
+
containerWidth: number;
|
|
23
|
+
containerHeight: number;
|
|
24
|
+
onComplete: (messageId: string) => void;
|
|
25
|
+
onWidthMeasured?: (messageId: string, width: number) => void;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export const DanmuMessage = memo(
|
|
29
|
+
({
|
|
30
|
+
message,
|
|
31
|
+
lane,
|
|
32
|
+
laneHeight,
|
|
33
|
+
videoTop,
|
|
34
|
+
opacity,
|
|
35
|
+
fontSize,
|
|
36
|
+
speed,
|
|
37
|
+
containerWidth,
|
|
38
|
+
containerHeight,
|
|
39
|
+
onComplete,
|
|
40
|
+
onWidthMeasured,
|
|
41
|
+
}: DanmuMessageProps) => {
|
|
42
|
+
const translateX = useSharedValue(containerWidth);
|
|
43
|
+
const [messageWidth, setMessageWidth] = useState(0);
|
|
44
|
+
const [animationStartTime, setAnimationStartTime] = useState<number | null>(
|
|
45
|
+
null,
|
|
46
|
+
);
|
|
47
|
+
const [totalDuration, setTotalDuration] = useState(0);
|
|
48
|
+
|
|
49
|
+
const getRgbColor = (
|
|
50
|
+
color: {
|
|
51
|
+
red: number;
|
|
52
|
+
green: number;
|
|
53
|
+
blue: number;
|
|
54
|
+
} = { red: 123, green: 123, blue: 123 },
|
|
55
|
+
) => {
|
|
56
|
+
const red = mapRange(color.red, 0, 255, 100, 230);
|
|
57
|
+
const green = mapRange(color.green, 0, 255, 100, 230);
|
|
58
|
+
const blue = mapRange(color.blue, 0, 255, 100, 230);
|
|
59
|
+
|
|
60
|
+
return `rgb(${Math.round(red)}, ${Math.round(green)}, ${Math.round(blue)})`;
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
const handleLayout = (event: LayoutChangeEvent) => {
|
|
64
|
+
const width = event.nativeEvent.layout.width;
|
|
65
|
+
if (width > 0) {
|
|
66
|
+
setMessageWidth(width);
|
|
67
|
+
if (onWidthMeasured) {
|
|
68
|
+
onWidthMeasured(message.uri, width);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
useEffect(() => {
|
|
74
|
+
if (messageWidth === 0) return; // Wait for layout measurement
|
|
75
|
+
|
|
76
|
+
const duration = baseDuration(message, MAX_DURATION, MIN_DURATION);
|
|
77
|
+
|
|
78
|
+
// Calculate how much time has elapsed if animation already started
|
|
79
|
+
let remainingDuration = duration;
|
|
80
|
+
let startPosition = containerWidth;
|
|
81
|
+
|
|
82
|
+
if (animationStartTime !== null) {
|
|
83
|
+
const elapsed = Date.now() - animationStartTime;
|
|
84
|
+
const progress = Math.min(elapsed / totalDuration, 1);
|
|
85
|
+
remainingDuration = duration * (1 - progress);
|
|
86
|
+
startPosition =
|
|
87
|
+
containerWidth + progress * (-messageWidth - containerWidth);
|
|
88
|
+
} else {
|
|
89
|
+
setAnimationStartTime(Date.now());
|
|
90
|
+
setTotalDuration(duration);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
if (__DEV__)
|
|
94
|
+
console.log(
|
|
95
|
+
`[danmu] animation started: "${message.record.text}" (duration: ${duration.toFixed(0)}ms, remaining: ${remainingDuration.toFixed(0)}ms, speed: ${speed}x)`,
|
|
96
|
+
);
|
|
97
|
+
|
|
98
|
+
translateX.value = startPosition;
|
|
99
|
+
|
|
100
|
+
translateX.value = withTiming(
|
|
101
|
+
-messageWidth,
|
|
102
|
+
{
|
|
103
|
+
duration: remainingDuration,
|
|
104
|
+
easing: Easing.linear,
|
|
105
|
+
},
|
|
106
|
+
(finished) => {
|
|
107
|
+
if (finished) {
|
|
108
|
+
runOnJS(onComplete)(message.uri);
|
|
109
|
+
}
|
|
110
|
+
},
|
|
111
|
+
);
|
|
112
|
+
}, [
|
|
113
|
+
messageWidth,
|
|
114
|
+
containerWidth,
|
|
115
|
+
speed,
|
|
116
|
+
message.uri,
|
|
117
|
+
message.record.text.length,
|
|
118
|
+
lane,
|
|
119
|
+
onComplete,
|
|
120
|
+
]);
|
|
121
|
+
|
|
122
|
+
const animatedStyle = useAnimatedStyle(() => {
|
|
123
|
+
return {
|
|
124
|
+
transform: [{ translateX: translateX.value }],
|
|
125
|
+
};
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
return (
|
|
129
|
+
<Animated.View
|
|
130
|
+
style={[
|
|
131
|
+
styles.container,
|
|
132
|
+
{
|
|
133
|
+
top: videoTop + lane * laneHeight,
|
|
134
|
+
opacity: opacity / 100,
|
|
135
|
+
},
|
|
136
|
+
animatedStyle,
|
|
137
|
+
]}
|
|
138
|
+
onLayout={handleLayout}
|
|
139
|
+
>
|
|
140
|
+
<Text
|
|
141
|
+
style={[
|
|
142
|
+
styles.text,
|
|
143
|
+
{
|
|
144
|
+
fontSize,
|
|
145
|
+
color: getRgbColor(message.chatProfile?.color),
|
|
146
|
+
},
|
|
147
|
+
]}
|
|
148
|
+
numberOfLines={1}
|
|
149
|
+
>
|
|
150
|
+
{message.record.text}
|
|
151
|
+
</Text>
|
|
152
|
+
</Animated.View>
|
|
153
|
+
);
|
|
154
|
+
},
|
|
155
|
+
(prevProps, nextProps) => {
|
|
156
|
+
return (
|
|
157
|
+
prevProps.message.uri === nextProps.message.uri &&
|
|
158
|
+
prevProps.lane === nextProps.lane &&
|
|
159
|
+
prevProps.laneHeight === nextProps.laneHeight &&
|
|
160
|
+
prevProps.videoTop === nextProps.videoTop &&
|
|
161
|
+
prevProps.opacity === nextProps.opacity &&
|
|
162
|
+
prevProps.fontSize === nextProps.fontSize &&
|
|
163
|
+
prevProps.speed === nextProps.speed &&
|
|
164
|
+
prevProps.containerWidth === nextProps.containerWidth &&
|
|
165
|
+
prevProps.containerHeight === nextProps.containerHeight
|
|
166
|
+
);
|
|
167
|
+
},
|
|
168
|
+
);
|
|
169
|
+
|
|
170
|
+
const styles = StyleSheet.create({
|
|
171
|
+
container: {
|
|
172
|
+
position: "absolute",
|
|
173
|
+
left: 0,
|
|
174
|
+
},
|
|
175
|
+
text: {
|
|
176
|
+
color: "white",
|
|
177
|
+
textShadowColor: "rgba(0, 0, 0, 0.8)",
|
|
178
|
+
textShadowOffset: { width: 1, height: 1 },
|
|
179
|
+
textShadowRadius: 64,
|
|
180
|
+
fontWeight: "600",
|
|
181
|
+
},
|
|
182
|
+
});
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { StyleSheet } from "react-native";
|
|
2
|
+
import { View } from "../ui";
|
|
3
|
+
import { DanmuOverlay } from "./danmu-overlay";
|
|
4
|
+
|
|
5
|
+
interface DanmuOverlayOBSProps {
|
|
6
|
+
channelDid?: string;
|
|
7
|
+
enabled?: boolean;
|
|
8
|
+
opacity?: number;
|
|
9
|
+
speed?: number;
|
|
10
|
+
laneCount?: number;
|
|
11
|
+
maxMessages?: number;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export function DanmuOverlayOBS({
|
|
15
|
+
channelDid,
|
|
16
|
+
enabled = true,
|
|
17
|
+
opacity = 80,
|
|
18
|
+
speed = 1,
|
|
19
|
+
laneCount = 12,
|
|
20
|
+
maxMessages = 50,
|
|
21
|
+
}: DanmuOverlayOBSProps) {
|
|
22
|
+
return (
|
|
23
|
+
<View style={styles.container}>
|
|
24
|
+
<DanmuOverlay
|
|
25
|
+
enabled={enabled}
|
|
26
|
+
opacity={opacity}
|
|
27
|
+
speed={speed}
|
|
28
|
+
laneCount={laneCount}
|
|
29
|
+
maxMessages={maxMessages}
|
|
30
|
+
/>
|
|
31
|
+
</View>
|
|
32
|
+
);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
const styles = StyleSheet.create({
|
|
36
|
+
container: {
|
|
37
|
+
position: "absolute",
|
|
38
|
+
top: 0,
|
|
39
|
+
left: 0,
|
|
40
|
+
right: 0,
|
|
41
|
+
bottom: 0,
|
|
42
|
+
backgroundColor: "transparent",
|
|
43
|
+
},
|
|
44
|
+
});
|
|
@@ -0,0 +1,225 @@
|
|
|
1
|
+
import { useCallback, useEffect, useRef, useState } from "react";
|
|
2
|
+
import { LayoutChangeEvent, StyleSheet } from "react-native";
|
|
3
|
+
import { ChatMessageViewHydrated } from "streamplace";
|
|
4
|
+
import { useChat, useLivestreamStore } from "../../livestream-store";
|
|
5
|
+
import { View } from "../ui";
|
|
6
|
+
import { DanmuMessage } from "./danmu-message";
|
|
7
|
+
import { baseDuration, MAX_DURATION, MIN_DURATION } from "./math";
|
|
8
|
+
import { useDanmuLanes } from "./use-danmu-lanes";
|
|
9
|
+
|
|
10
|
+
interface DanmuOverlayProps {
|
|
11
|
+
enabled?: boolean;
|
|
12
|
+
opacity?: number;
|
|
13
|
+
speed?: number;
|
|
14
|
+
laneCount?: number;
|
|
15
|
+
maxMessages?: number;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
interface ActiveDanmuMessage {
|
|
19
|
+
message: ChatMessageViewHydrated;
|
|
20
|
+
lane: number;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const DEFAULT_LANE_COUNT = 12;
|
|
24
|
+
const DEFAULT_OPACITY = 80;
|
|
25
|
+
const DEFAULT_SPEED = 1;
|
|
26
|
+
const DEFAULT_MAX_MESSAGES = 50;
|
|
27
|
+
const FONT_SIZE_PERCENTAGE = 0.7;
|
|
28
|
+
const MAX_PROCESSED_MESSAGES = 10;
|
|
29
|
+
|
|
30
|
+
// px from top of video where danmu won't appear (avoid overlapping with title)
|
|
31
|
+
const TOP_GAP = 20;
|
|
32
|
+
// px from bottom of video (avoid overlapping with controls)
|
|
33
|
+
const BOTTOM_GAP = 20;
|
|
34
|
+
|
|
35
|
+
export function DanmuOverlay({
|
|
36
|
+
enabled = true,
|
|
37
|
+
opacity = DEFAULT_OPACITY,
|
|
38
|
+
speed = DEFAULT_SPEED,
|
|
39
|
+
laneCount = DEFAULT_LANE_COUNT,
|
|
40
|
+
maxMessages = DEFAULT_MAX_MESSAGES,
|
|
41
|
+
}: DanmuOverlayProps) {
|
|
42
|
+
const chat = useChat();
|
|
43
|
+
const segment = useLivestreamStore((x) => x.segment);
|
|
44
|
+
|
|
45
|
+
const [containerWidth, setContainerWidth] = useState(0);
|
|
46
|
+
const [containerHeight, setContainerHeight] = useState(0);
|
|
47
|
+
const [activeDanmu, setActiveDanmu] = useState<
|
|
48
|
+
Map<string, ActiveDanmuMessage>
|
|
49
|
+
>(new Map());
|
|
50
|
+
const processedMessages = useRef(new Set<string>());
|
|
51
|
+
const mountTime = useRef(Date.now());
|
|
52
|
+
const lastChatLength = useRef(0);
|
|
53
|
+
const { assignLane, updateDanmuWidth, releaseLane, cleanup } = useDanmuLanes(
|
|
54
|
+
laneCount,
|
|
55
|
+
containerWidth,
|
|
56
|
+
);
|
|
57
|
+
|
|
58
|
+
const handleLayout = useCallback((event: LayoutChangeEvent) => {
|
|
59
|
+
const { width, height } = event.nativeEvent.layout;
|
|
60
|
+
setContainerWidth(width);
|
|
61
|
+
setContainerHeight(height);
|
|
62
|
+
}, []);
|
|
63
|
+
|
|
64
|
+
const handleMessageComplete = useCallback(
|
|
65
|
+
(messageId: string) => {
|
|
66
|
+
releaseLane(messageId);
|
|
67
|
+
setActiveDanmu((prev) => {
|
|
68
|
+
const next = new Map(prev);
|
|
69
|
+
next.delete(messageId);
|
|
70
|
+
return next;
|
|
71
|
+
});
|
|
72
|
+
},
|
|
73
|
+
[releaseLane],
|
|
74
|
+
);
|
|
75
|
+
|
|
76
|
+
const handleWidthMeasured = useCallback(
|
|
77
|
+
(messageId: string, width: number) => {
|
|
78
|
+
updateDanmuWidth(messageId, width);
|
|
79
|
+
},
|
|
80
|
+
[updateDanmuWidth],
|
|
81
|
+
);
|
|
82
|
+
|
|
83
|
+
useEffect(() => {
|
|
84
|
+
if (!enabled || !chat || containerWidth === 0) return;
|
|
85
|
+
|
|
86
|
+
// only check new messages since last render (chat is sorted newest first)
|
|
87
|
+
const newMessageCount = chat.length - lastChatLength.current;
|
|
88
|
+
if (newMessageCount <= 0) return;
|
|
89
|
+
|
|
90
|
+
const messagesToCheck = chat.slice(0, newMessageCount);
|
|
91
|
+
lastChatLength.current = chat.length;
|
|
92
|
+
|
|
93
|
+
const newMessages = messagesToCheck.filter((msg) => {
|
|
94
|
+
const hasProcessed = processedMessages.current.has(msg.uri);
|
|
95
|
+
const isSystem = msg.author.did === "did:sys:system";
|
|
96
|
+
const msgTime = new Date(msg.record.createdAt).getTime();
|
|
97
|
+
const isAfterMount = msgTime >= mountTime.current;
|
|
98
|
+
|
|
99
|
+
return !hasProcessed && !isSystem && isAfterMount;
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
if (newMessages.length === 0) return;
|
|
103
|
+
|
|
104
|
+
const messagesToAdd: ActiveDanmuMessage[] = [];
|
|
105
|
+
|
|
106
|
+
for (const message of newMessages.slice(0, maxMessages)) {
|
|
107
|
+
// mark as processed FIRST to prevent duplicate processing if effect runs twice
|
|
108
|
+
if (processedMessages.current.has(message.uri)) {
|
|
109
|
+
continue;
|
|
110
|
+
}
|
|
111
|
+
processedMessages.current.add(message.uri);
|
|
112
|
+
|
|
113
|
+
if (processedMessages.current.size > MAX_PROCESSED_MESSAGES) {
|
|
114
|
+
const toRemove = Array.from(processedMessages.current).slice(
|
|
115
|
+
0,
|
|
116
|
+
processedMessages.current.size - MAX_PROCESSED_MESSAGES,
|
|
117
|
+
);
|
|
118
|
+
toRemove.forEach((uri) => processedMessages.current.delete(uri));
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
const duration = baseDuration(message, MAX_DURATION, MIN_DURATION);
|
|
122
|
+
if (__DEV__)
|
|
123
|
+
console.log("[danmu] message", message.record.text, {
|
|
124
|
+
duration,
|
|
125
|
+
speed,
|
|
126
|
+
});
|
|
127
|
+
const lane = assignLane(message.uri, duration);
|
|
128
|
+
|
|
129
|
+
if (lane !== null) {
|
|
130
|
+
messagesToAdd.push({ message, lane });
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
if (messagesToAdd.length > 0) {
|
|
135
|
+
setActiveDanmu((prev) => {
|
|
136
|
+
const next = new Map(prev);
|
|
137
|
+
for (const danmu of messagesToAdd) {
|
|
138
|
+
next.set(danmu.message.uri, danmu);
|
|
139
|
+
}
|
|
140
|
+
return next;
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
}, [chat, enabled, speed, maxMessages]);
|
|
144
|
+
|
|
145
|
+
useEffect(() => {
|
|
146
|
+
const interval = setInterval(() => {
|
|
147
|
+
cleanup();
|
|
148
|
+
}, 1000);
|
|
149
|
+
|
|
150
|
+
return () => clearInterval(interval);
|
|
151
|
+
}, [cleanup]);
|
|
152
|
+
|
|
153
|
+
if (!enabled || containerWidth === 0 || containerHeight === 0) {
|
|
154
|
+
return <View style={styles.container} onLayout={handleLayout} />;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
// Calculate video bounds based on actual video dimensions from segment
|
|
158
|
+
const segmentVideoWidth = segment?.video?.[0]?.width;
|
|
159
|
+
const segmentVideoHeight = segment?.video?.[0]?.height;
|
|
160
|
+
|
|
161
|
+
// Fall back to 16:9 if no segment dimensions available
|
|
162
|
+
const videoAspectRatio =
|
|
163
|
+
segmentVideoWidth && segmentVideoHeight
|
|
164
|
+
? segmentVideoWidth / segmentVideoHeight
|
|
165
|
+
: 16 / 9;
|
|
166
|
+
|
|
167
|
+
const containerAspectRatio = containerWidth / containerHeight;
|
|
168
|
+
|
|
169
|
+
let videoWidth: number;
|
|
170
|
+
let videoHeight: number;
|
|
171
|
+
let videoTop: number;
|
|
172
|
+
let videoLeft: number;
|
|
173
|
+
|
|
174
|
+
if (containerAspectRatio > videoAspectRatio) {
|
|
175
|
+
// Container is wider than video - letterbox on sides
|
|
176
|
+
videoHeight = containerHeight;
|
|
177
|
+
videoWidth = videoHeight * videoAspectRatio;
|
|
178
|
+
videoTop = 0;
|
|
179
|
+
videoLeft = (containerWidth - videoWidth) / 2;
|
|
180
|
+
// Adjust for top/bottom gaps iff we don't have top letterboxing
|
|
181
|
+
videoTop += TOP_GAP;
|
|
182
|
+
videoHeight -= TOP_GAP + BOTTOM_GAP;
|
|
183
|
+
} else {
|
|
184
|
+
// Container is taller than video - letterbox on top/bottom
|
|
185
|
+
videoWidth = containerWidth;
|
|
186
|
+
videoHeight = videoWidth / videoAspectRatio;
|
|
187
|
+
videoTop = (containerHeight - videoHeight) / 2;
|
|
188
|
+
videoLeft = 0;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
const laneHeight = videoHeight / laneCount;
|
|
192
|
+
const fontSize = laneHeight * FONT_SIZE_PERCENTAGE;
|
|
193
|
+
|
|
194
|
+
return (
|
|
195
|
+
<View style={styles.container} onLayout={handleLayout} pointerEvents="none">
|
|
196
|
+
{Array.from(activeDanmu.values()).map(({ message, lane }) => (
|
|
197
|
+
<DanmuMessage
|
|
198
|
+
key={message.uri}
|
|
199
|
+
message={message}
|
|
200
|
+
lane={lane}
|
|
201
|
+
laneHeight={laneHeight}
|
|
202
|
+
videoTop={videoTop}
|
|
203
|
+
opacity={opacity}
|
|
204
|
+
fontSize={fontSize}
|
|
205
|
+
speed={speed}
|
|
206
|
+
containerWidth={containerWidth}
|
|
207
|
+
containerHeight={containerHeight}
|
|
208
|
+
onComplete={handleMessageComplete}
|
|
209
|
+
onWidthMeasured={handleWidthMeasured}
|
|
210
|
+
/>
|
|
211
|
+
))}
|
|
212
|
+
</View>
|
|
213
|
+
);
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
const styles = StyleSheet.create({
|
|
217
|
+
container: {
|
|
218
|
+
position: "absolute",
|
|
219
|
+
top: 0,
|
|
220
|
+
left: 0,
|
|
221
|
+
right: 0,
|
|
222
|
+
bottom: 0,
|
|
223
|
+
overflow: "hidden",
|
|
224
|
+
},
|
|
225
|
+
});
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
export const mapRange = (
|
|
2
|
+
num: number,
|
|
3
|
+
inMin: number,
|
|
4
|
+
inMax: number,
|
|
5
|
+
outMin: number,
|
|
6
|
+
outMax: number,
|
|
7
|
+
) => {
|
|
8
|
+
return ((num - inMin) * (outMax - outMin)) / (inMax - inMin) + outMin;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
export const between = (num: number, min: number, max: number) => {
|
|
12
|
+
return Math.min(Math.max(num, min), max);
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
export const baseDuration = (
|
|
16
|
+
message: { record: { text: string | any[] } },
|
|
17
|
+
min: number,
|
|
18
|
+
max: number,
|
|
19
|
+
) =>
|
|
20
|
+
between(
|
|
21
|
+
mapRange(Math.log(message.record.text.length) * 8, 1, 16, min, max),
|
|
22
|
+
min,
|
|
23
|
+
max,
|
|
24
|
+
);
|
|
25
|
+
|
|
26
|
+
export const MIN_DURATION = 6000;
|
|
27
|
+
export const MAX_DURATION = 12000;
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { G, Path, Svg, SvgProps } from "react-native-svg";
|
|
2
|
+
|
|
3
|
+
export interface MuProps extends SvgProps {
|
|
4
|
+
size?: number | string;
|
|
5
|
+
color?: string;
|
|
6
|
+
strokeWidth?: number | string;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export const Mu = ({
|
|
10
|
+
size = 24,
|
|
11
|
+
color = "#000000",
|
|
12
|
+
strokeWidth = 10.9,
|
|
13
|
+
...props
|
|
14
|
+
}: MuProps) => (
|
|
15
|
+
<Svg width={size} height={size} viewBox="0 0 230 220" {...props}>
|
|
16
|
+
<G
|
|
17
|
+
strokeLinecap="round"
|
|
18
|
+
fillRule="evenodd"
|
|
19
|
+
stroke={color}
|
|
20
|
+
strokeWidth={strokeWidth}
|
|
21
|
+
fill={color}
|
|
22
|
+
>
|
|
23
|
+
<Path d="M 13.756 65.7 L 53.756 65.7 Q 55.756 65.7 57.256 64.2 Q 58.756 62.7 58.756 60.7 L 58.756 22.2 Q 58.756 20.2 57.256 18.7 Q 55.756 17.2 53.756 17.2 L 3.506 17.2 Q 2.006 17.2 1.006 16.2 Q 0.006 15.2 0.006 13.45 Q 0.006 11.95 1.006 10.95 Q 2.006 9.95 3.506 9.95 L 56.506 9.95 Q 60.756 9.95 63.506 12.7 Q 66.256 15.45 66.256 19.7 L 66.256 63.45 Q 66.256 67.45 63.506 70.325 Q 60.756 73.2 56.506 73.2 L 16.256 73.2 Q 14.006 73.2 12.506 74.7 Q 11.006 76.2 11.006 78.2 Q 10.756 89.7 10.506 98.575 Q 10.256 107.45 9.756 117.45 A 4.864 4.864 0 0 0 9.717 118.054 A 4.167 4.167 0 0 0 10.881 120.95 Q 12.256 122.45 14.506 122.45 L 60.256 122.45 Q 63.006 122.45 64.881 124.325 Q 66.756 126.2 66.756 129.7 A 1579.753 1579.753 0 0 1 65.295 155.188 Q 64.543 166.519 63.737 175.256 A 494.991 494.991 0 0 1 63.506 177.7 Q 61.756 195.7 59.506 204.7 A 39.785 39.785 0 0 1 55.486 215.384 A 14.939 14.939 0 0 1 54.006 217.45 Q 51.506 220.2 48.506 221.075 Q 45.506 221.95 40.506 222.2 Q 36.006 222.45 28.506 222.2 Q 21.006 221.95 11.756 221.45 A 5.389 5.389 0 0 1 10.79 221.197 A 5.13 5.13 0 0 1 9.131 220.2 Q 8.006 219.2 7.506 217.45 A 3.871 3.871 0 0 1 7.45 216.814 A 2.355 2.355 0 0 1 8.131 215.075 Q 9.006 214.2 10.506 214.2 Q 20.006 214.95 27.881 215.2 Q 35.756 215.45 39.006 215.45 Q 42.506 215.45 44.381 215.075 Q 46.256 214.7 47.756 212.95 A 17.691 17.691 0 0 0 51.389 205.661 A 55.6 55.6 0 0 0 52.381 201.825 A 144.371 144.371 0 0 0 54.411 189.832 A 374.249 374.249 0 0 0 55.881 176.95 A 735.769 735.769 0 0 0 57.475 157.825 A 1519.1 1519.1 0 0 0 59.006 133.45 A 5.968 5.968 0 0 0 59.061 132.663 A 2.91 2.91 0 0 0 55.506 129.7 L 11.506 129.7 Q 7.506 129.7 4.881 126.825 A 8.876 8.876 0 0 1 2.484 120.637 A 11.015 11.015 0 0 1 2.506 119.95 Q 2.756 113.95 3.006 108.7 Q 3.256 103.45 3.506 98.45 Q 3.756 93.45 3.881 87.95 Q 4.006 82.45 4.006 75.45 Q 4.006 71.2 6.881 68.45 Q 9.756 65.7 13.756 65.7 Z M 99.006 49.95 L 200.256 49.95 Q 204.506 49.95 207.256 52.7 Q 210.006 55.45 210.006 59.7 L 210.006 127.95 Q 210.006 131.95 207.256 134.825 Q 204.506 137.7 200.256 137.7 L 99.006 137.7 Q 95.006 137.7 92.131 134.825 Q 89.256 131.95 89.256 127.95 L 89.256 59.7 Q 89.256 55.45 92.131 52.7 Q 95.006 49.95 99.006 49.95 Z M 145.506 53.95 L 153.006 53.95 L 153.006 218.95 Q 153.006 220.7 152.006 221.7 Q 151.006 222.7 149.256 222.7 Q 147.506 222.7 146.506 221.7 Q 145.506 220.7 145.506 218.95 L 145.506 53.95 Z M 77.506 163.95 L 219.006 163.95 Q 220.756 163.95 221.756 164.95 Q 222.756 165.95 222.756 167.7 Q 222.756 169.45 221.756 170.45 Q 220.756 171.45 219.006 171.45 L 77.506 171.45 Q 75.756 171.45 74.756 170.45 Q 73.756 169.45 73.756 167.7 Q 73.756 165.95 74.756 164.95 Q 75.756 163.95 77.506 163.95 Z M 96.506 97.45 L 96.506 125.45 Q 96.506 127.45 98.006 128.95 Q 99.506 130.45 101.506 130.45 L 197.506 130.45 Q 199.506 130.45 201.006 128.95 Q 202.506 127.45 202.506 125.45 L 202.506 97.45 L 96.506 97.45 Z M 96.506 62.45 L 96.506 89.95 L 202.506 89.95 L 202.506 62.45 Q 202.506 60.45 201.006 58.95 Q 199.506 57.45 197.506 57.45 L 101.506 57.45 Q 99.506 57.45 98.006 58.95 Q 96.506 60.45 96.506 62.45 Z M 166.506 55.2 L 161.006 51.95 Q 168.756 41.7 176.881 27.7 A 383.321 383.321 0 0 0 189.09 5.114 A 264.812 264.812 0 0 0 190.506 2.2 Q 191.006 0.7 192.506 0.2 Q 194.006 -0.3 195.756 0.45 Q 197.256 0.95 197.756 2.45 Q 198.256 3.95 197.506 5.2 Q 190.506 17.95 182.506 31.45 Q 174.506 44.95 166.506 55.2 Z M 107.672 1.117 A 4.458 4.458 0 0 0 106.006 1.45 Q 104.756 1.95 104.381 2.95 Q 104.006 3.95 104.756 4.95 Q 111.756 13.95 117.006 22.825 Q 122.256 31.7 126.006 39.95 Q 126.506 41.2 127.756 41.7 Q 129.006 42.2 130.256 41.45 Q 132.006 40.7 132.506 39.2 Q 133.006 37.7 132.256 36.45 Q 128.506 28.45 123.256 19.95 Q 118.006 11.45 110.756 2.45 Q 109.756 1.45 108.506 1.2 A 4.24 4.24 0 0 0 107.672 1.117 Z" />
|
|
24
|
+
</G>
|
|
25
|
+
</Svg>
|
|
26
|
+
);
|
|
27
|
+
|
|
28
|
+
export default Mu;
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
import { useCallback, useRef } from "react";
|
|
2
|
+
|
|
3
|
+
export interface DanmuLane {
|
|
4
|
+
index: number;
|
|
5
|
+
occupiedUntil: number;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
export interface ActiveDanmu {
|
|
9
|
+
id: string;
|
|
10
|
+
lane: number;
|
|
11
|
+
endTime: number;
|
|
12
|
+
startTime: number;
|
|
13
|
+
duration: number;
|
|
14
|
+
width?: number;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
const LANE_GAP = 4;
|
|
18
|
+
|
|
19
|
+
export const useDanmuLanes = (laneCount: number, containerWidth: number) => {
|
|
20
|
+
const activeDanmu = useRef<Map<string, ActiveDanmu>>(new Map());
|
|
21
|
+
const lanes = useRef<DanmuLane[]>(
|
|
22
|
+
Array.from({ length: laneCount }, (_, i) => ({
|
|
23
|
+
index: i,
|
|
24
|
+
occupiedUntil: 0,
|
|
25
|
+
})),
|
|
26
|
+
);
|
|
27
|
+
|
|
28
|
+
const canFitInLane = useCallback(
|
|
29
|
+
(laneIndex: number): boolean => {
|
|
30
|
+
const now = Date.now();
|
|
31
|
+
|
|
32
|
+
// find all active danmu in this lane
|
|
33
|
+
const danmuInLane = Array.from(activeDanmu.current.values()).filter(
|
|
34
|
+
(d) => d.lane === laneIndex && d.endTime > now,
|
|
35
|
+
);
|
|
36
|
+
|
|
37
|
+
if (danmuInLane.length === 0) return true;
|
|
38
|
+
|
|
39
|
+
// check the most recent danmu in this lane
|
|
40
|
+
const mostRecent = danmuInLane.reduce((latest, current) =>
|
|
41
|
+
current.startTime > latest.startTime ? current : latest,
|
|
42
|
+
);
|
|
43
|
+
|
|
44
|
+
// calculate how far it has traveled
|
|
45
|
+
const elapsed = now - mostRecent.startTime;
|
|
46
|
+
const progress = elapsed / mostRecent.duration;
|
|
47
|
+
const traveled = containerWidth * progress;
|
|
48
|
+
|
|
49
|
+
// estimate width (assume ~8px per char, will be updated with actual width later)
|
|
50
|
+
const estimatedWidth = mostRecent.width || 200;
|
|
51
|
+
|
|
52
|
+
// check if there's enough space
|
|
53
|
+
const spaceNeeded = estimatedWidth + LANE_GAP;
|
|
54
|
+
const hasSpace = traveled >= spaceNeeded;
|
|
55
|
+
|
|
56
|
+
return hasSpace;
|
|
57
|
+
},
|
|
58
|
+
[containerWidth],
|
|
59
|
+
);
|
|
60
|
+
|
|
61
|
+
const assignLane = useCallback(
|
|
62
|
+
(messageId: string, duration: number, width?: number): number | null => {
|
|
63
|
+
const now = Date.now();
|
|
64
|
+
|
|
65
|
+
for (const lane of lanes.current) {
|
|
66
|
+
if (canFitInLane(lane.index)) {
|
|
67
|
+
const endTime = now + duration;
|
|
68
|
+
activeDanmu.current.set(messageId, {
|
|
69
|
+
id: messageId,
|
|
70
|
+
lane: lane.index,
|
|
71
|
+
endTime,
|
|
72
|
+
startTime: now,
|
|
73
|
+
duration,
|
|
74
|
+
width,
|
|
75
|
+
});
|
|
76
|
+
return lane.index;
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
return null;
|
|
81
|
+
},
|
|
82
|
+
[lanes, canFitInLane],
|
|
83
|
+
);
|
|
84
|
+
|
|
85
|
+
const updateDanmuWidth = useCallback((messageId: string, width: number) => {
|
|
86
|
+
const danmu = activeDanmu.current.get(messageId);
|
|
87
|
+
if (danmu) {
|
|
88
|
+
danmu.width = width;
|
|
89
|
+
}
|
|
90
|
+
}, []);
|
|
91
|
+
|
|
92
|
+
const releaseLane = useCallback((messageId: string) => {
|
|
93
|
+
const danmu = activeDanmu.current.get(messageId);
|
|
94
|
+
if (danmu) {
|
|
95
|
+
activeDanmu.current.delete(messageId);
|
|
96
|
+
}
|
|
97
|
+
}, []);
|
|
98
|
+
|
|
99
|
+
const cleanup = useCallback(() => {
|
|
100
|
+
const now = Date.now();
|
|
101
|
+
for (const [id, danmu] of activeDanmu.current.entries()) {
|
|
102
|
+
if (danmu.endTime <= now) {
|
|
103
|
+
activeDanmu.current.delete(id);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}, []);
|
|
107
|
+
|
|
108
|
+
return {
|
|
109
|
+
assignLane,
|
|
110
|
+
updateDanmuWidth,
|
|
111
|
+
releaseLane,
|
|
112
|
+
cleanup,
|
|
113
|
+
};
|
|
114
|
+
};
|