|
@@ -11,109 +11,155 @@
|
|
|
MediaFullscreenButton
|
|
|
MediaPipButton
|
|
|
MediaPlayButton
|
|
|
- MediaPlaybackRateButton
|
|
|
- MediaMuteButton)]))
|
|
|
+ MediaMuteButton
|
|
|
+ MediaLoadingIndicator)]
|
|
|
+ ["media-chrome/dist/react/menu/index.js"
|
|
|
+ :refer
|
|
|
+ (MediaCaptionsMenu
|
|
|
+ MediaPlaybackRateMenu
|
|
|
+ MediaSettingsMenu
|
|
|
+ MediaSettingsMenuButton
|
|
|
+ MediaSettingsMenuItem
|
|
|
+ MediaRenditionMenu)]))
|
|
|
+
|
|
|
+(defn button
|
|
|
+ [& {:keys [icon on-click disabled? show-on-mobile? extra-classes]}]
|
|
|
+ [:button.outline-none.focus:ring-transparent
|
|
|
+ {:class (into (into (when disabled? [:opacity-50 :cursor-auto])
|
|
|
+ (when-not show-on-mobile? [:hidden :lg:block]))
|
|
|
+ extra-classes)
|
|
|
+ :on-click on-click}
|
|
|
+ icon])
|
|
|
+
|
|
|
+(defn loop-button
|
|
|
+ [loop-playback color show-on-mobile?]
|
|
|
+ [button
|
|
|
+ :icon
|
|
|
+ [:div.relative.flex.items-center
|
|
|
+ [:i.fa-solid.fa-repeat
|
|
|
+ {:style {:color (when loop-playback color)}}]
|
|
|
+ (when (= loop-playback :stream)
|
|
|
+ [:div.absolute.w-full.h-full.flex.justify-center.items-center.font-bold
|
|
|
+ {:class "text-[6px]"
|
|
|
+ :style {:color (when loop-playback color)}}
|
|
|
+ "1"])]
|
|
|
+ :on-click #(rf/dispatch [:player/loop])
|
|
|
+ :extra-classes [:text-sm]
|
|
|
+ :show-on-mobile? show-on-mobile?])
|
|
|
+
|
|
|
+(defn shuffle-button
|
|
|
+ [shuffle? color show-on-mobile?]
|
|
|
+ [button
|
|
|
+ :icon
|
|
|
+ [:i.fa-solid.fa-shuffle {:style {:color (when shuffle? color)}}]
|
|
|
+ :on-click #(rf/dispatch [:queue/shuffle (not shuffle?)])
|
|
|
+ :extra-classes [:text-sm]
|
|
|
+ :show-on-mobile? show-on-mobile?])
|
|
|
|
|
|
(defn video-player
|
|
|
- []
|
|
|
- (let [!elapsed-time @(rf/subscribe [:elapsed-time])
|
|
|
- !main-player-first? (r/atom true)]
|
|
|
+ [_ _ _ on-mount]
|
|
|
+ (let [service-color @(rf/subscribe [:service-color])]
|
|
|
(r/create-class
|
|
|
- {:component-will-unmount #(rf/dispatch [:main-player/ready false])
|
|
|
+ {:component-did-mount on-mount
|
|
|
:reagent-render
|
|
|
- (fn [{:keys [video-streams audio-streams thumbnails]}
|
|
|
- !player]
|
|
|
- (let [show-main-player? @(rf/subscribe [:main-player/show])
|
|
|
- service-color @(rf/subscribe [:service-color])]
|
|
|
- [:div
|
|
|
- {:class "w-full h-80 md:h-[450px] lg:h-[600px]"}
|
|
|
- [:> MediaController
|
|
|
- {:style {"--media-secondary-color" "transparent"
|
|
|
- "--media-primary-color" "white"
|
|
|
- "aspectRatio" "16/9"
|
|
|
- "height" "100%"
|
|
|
- "width" "100%"}}
|
|
|
- [:video
|
|
|
- {:style {"maxHeight" "100%"
|
|
|
- "minHeight" "100%"
|
|
|
- "minWidth" "100%"
|
|
|
- "maxWidth" "100%"}
|
|
|
- :ref #(reset! !player %)
|
|
|
- :poster (-> thumbnails
|
|
|
- last
|
|
|
- :url)
|
|
|
- :loop (when show-main-player?
|
|
|
- (= @(rf/subscribe [:player/loop]) :stream))
|
|
|
- :on-can-play #(rf/dispatch [:main-player/ready true])
|
|
|
- :on-ended #(when show-main-player?
|
|
|
- (rf/dispatch [:queue/change-pos
|
|
|
- (inc @(rf/subscribe
|
|
|
- [:queue/position]))])
|
|
|
- (reset! !elapsed-time 0))
|
|
|
- :on-play #(rf/dispatch [:main-player/play])
|
|
|
- :on-loaded-data (fn []
|
|
|
- (when show-main-player?
|
|
|
- (rf/dispatch [:main-player/start]))
|
|
|
- (when (and @!main-player-first?
|
|
|
- show-main-player?)
|
|
|
- (reset! !main-player-first? false)))
|
|
|
- :on-time-update (when show-main-player?
|
|
|
- #(reset! !elapsed-time (.-currentTime
|
|
|
- @!player)))
|
|
|
- :on-seeked (when show-main-player?
|
|
|
- #(reset! !elapsed-time (.-currentTime
|
|
|
- @!player)))
|
|
|
- :slot "media"
|
|
|
-
|
|
|
- :src (-> (into video-streams audio-streams)
|
|
|
- first
|
|
|
- :content)
|
|
|
- :preload "auto"
|
|
|
- :muted @(rf/subscribe [:player/muted])}]
|
|
|
- [:div.ytp-gradient-bottom.absolute.w-full.bottom-0.pointer-events-none.bg-bottom.bg-repeat-x
|
|
|
- {:style
|
|
|
- {"paddingTop" "37px"
|
|
|
- "height" "170px"
|
|
|
- "backgroundImage"
|
|
|
- "url('')"}}]
|
|
|
- [:> MediaTimeRange
|
|
|
- {:class "w-full h-[5px]"
|
|
|
- :style
|
|
|
- {"--media-control-hover-background" "transparent"
|
|
|
- "--media-range-track-transition" "height 0.1s linear"
|
|
|
- "--media-range-track-background" "rgba(255,255,255,.2)"
|
|
|
- "--media-range-track-pointer-background" "rgba(255,255,255,.5)"
|
|
|
- "--media-time-range-buffered-color" "rgba(255,255,255,.4)"
|
|
|
- "--media-range-bar-color" service-color
|
|
|
- "--media-range-thumb-border-radius" "13px"
|
|
|
- "--media-range-thumb-background" service-color
|
|
|
- "--media-range-thumb-transition" "transform 0.1s linear"
|
|
|
- "--media-range-thumb-transform" "scale(0) translate(0%, 0%)"}}]
|
|
|
- [:> MediaControlBar
|
|
|
- {:class "relative pl-[10px] pr-[5px]"
|
|
|
- :style
|
|
|
- {"--media-control-hover-background" "transparent"
|
|
|
- "--media-range-track-height" "3px"
|
|
|
- "--media-range-thumb-height" "13px"
|
|
|
- "--media-range-thumb-width" "13px"
|
|
|
- "--media-range-thumb-border-radius" "13px"}}
|
|
|
- [:> MediaPlayButton
|
|
|
- {:class "py-[6px] px-[10px]"
|
|
|
- :style
|
|
|
- {"--media-button-icon-width" "30px"}}]
|
|
|
- [:> MediaMuteButton
|
|
|
- {:class "peer/mute"}]
|
|
|
- [:> MediaVolumeRange
|
|
|
- {:class
|
|
|
- ["w-0" "overflow-hidden" "transition-[width]"
|
|
|
- "transition-200" "ease-in" "peer-hover/mute:w-[70px]"
|
|
|
- "peer-focus/mute:w-[70px]" "hover:w-[70px]" "focus:w-[70px]"]
|
|
|
- :style
|
|
|
- {"--media-range-track-background" "rgba(255,255,255,.2)"
|
|
|
- "--media-range-bar-color" service-color
|
|
|
- "--media-range-thumb-background" service-color}}]
|
|
|
- [:> MediaTimeDisplay {:showDuration true}]
|
|
|
- [:span.control-spacer.grow]
|
|
|
- [:> MediaPlaybackRateButton]
|
|
|
- [:> MediaPipButton]
|
|
|
- [:> MediaFullscreenButton]]]]))})))
|
|
|
+ (fn [{:keys [thumbnails subtitles]} !player video-args]
|
|
|
+ [:div
|
|
|
+ {:class "w-full h-80 md:h-[450px] lg:h-[600px]"}
|
|
|
+ [:> MediaController
|
|
|
+ {:style {"--media-secondary-color" "transparent"
|
|
|
+ "--media-primary-color" "white"
|
|
|
+ "aspectRatio" "16/9"
|
|
|
+ "height" "100%"
|
|
|
+ "width" "100%"}}
|
|
|
+ [:video
|
|
|
+ (into
|
|
|
+ {:style {"maxHeight" "100%"
|
|
|
+ "minHeight" "100%"
|
|
|
+ "minWidth" "100%"
|
|
|
+ "maxWidth" "100%"}
|
|
|
+ :poster (-> thumbnails
|
|
|
+ last
|
|
|
+ :url)
|
|
|
+ :ref #(reset! !player %)
|
|
|
+ :slot "media"
|
|
|
+ :preload "metadata"}
|
|
|
+ video-args)
|
|
|
+ [:track
|
|
|
+ {:label (:displayLanguageName (first subtitles))
|
|
|
+ :kind "captions"
|
|
|
+ :srcLang (:languageTag (first subtitles))
|
|
|
+ :src (:content (first subtitles))}]]
|
|
|
+ [:div.ytp-gradient-bottom.absolute.w-full.bottom-0.pointer-events-none.bg-bottom.bg-repeat-x
|
|
|
+ {:style
|
|
|
+ {"paddingTop" "37px"
|
|
|
+ "height" "170px"
|
|
|
+ "backgroundImage"
|
|
|
+ "url('')"}}]
|
|
|
+ [:> MediaTimeRange
|
|
|
+ {:class ["w-full" "h-[5px]"]
|
|
|
+ :style
|
|
|
+ {"--media-control-hover-background" "transparent"
|
|
|
+ "--media-range-track-transition" "height 0.1s linear"
|
|
|
+ "--media-range-track-background" "rgba(255,255,255,.2)"
|
|
|
+ "--media-range-track-pointer-background" "rgba(255,255,255,.5)"
|
|
|
+ "--media-time-range-buffered-color" "rgba(255,255,255,.4)"
|
|
|
+ "--media-range-bar-color" service-color
|
|
|
+ "--media-range-thumb-border-radius" "13px"
|
|
|
+ "--media-range-thumb-background" service-color
|
|
|
+ "--media-range-thumb-transition" "transform 0.1s linear"}}]
|
|
|
+ [:> MediaLoadingIndicator
|
|
|
+ {:slot "centered-chrome"
|
|
|
+ :noautohide true
|
|
|
+ :style {"--media-loading-indicator-icon-height" "200px"}}]
|
|
|
+ [:> MediaSettingsMenu
|
|
|
+ {:anchor "auto"
|
|
|
+ :hidden true
|
|
|
+ :style {"--media-secondary-color" "rgba(23, 23, 23, .9)"
|
|
|
+ "--media-menu-border" "1px solid white"
|
|
|
+ "--media-menu-border-radius" "35px"
|
|
|
+ "--media-font-family" "Nunito Sans"
|
|
|
+ "--media-control-hover-background" "red"}}
|
|
|
+ [:> MediaSettingsMenuItem
|
|
|
+ "Speed"
|
|
|
+ [:> MediaPlaybackRateMenu
|
|
|
+ {:slot "submenu" :hidden true}
|
|
|
+ [:div {:slot "title"} "Speed"]]]
|
|
|
+ [:> MediaSettingsMenuItem
|
|
|
+ "Quality"
|
|
|
+ [:> MediaRenditionMenu
|
|
|
+ {:slot "submenu" :hidden true}
|
|
|
+ [:div {:slot "title"} "Quality"]]]
|
|
|
+ [:> MediaSettingsMenuItem
|
|
|
+ "Captions"
|
|
|
+ [:> MediaCaptionsMenu
|
|
|
+ {:slot "submenu" :hidden true}
|
|
|
+ [:div {:slot "title"} "Captions"]]]]
|
|
|
+ [:> MediaControlBar
|
|
|
+ {:class "relative pl-[10px] pr-[5px]"
|
|
|
+ :style
|
|
|
+ {"--media-control-hover-background" "transparent"
|
|
|
+ "--media-range-track-height" "3px"
|
|
|
+ "--media-range-thumb-height" "13px"
|
|
|
+ "--media-range-thumb-width" "13px"
|
|
|
+ "--media-range-thumb-border-radius" "13px"
|
|
|
+ "--media-tooltip-display" "none"}}
|
|
|
+ [:> MediaPlayButton
|
|
|
+ {:class "py-[6px] px-[10px]"
|
|
|
+ :style
|
|
|
+ {"--media-button-icon-width" "30px"}}]
|
|
|
+ [:> MediaMuteButton
|
|
|
+ {:class "peer/mute"}]
|
|
|
+ [:> MediaVolumeRange
|
|
|
+ {:class
|
|
|
+ ["w-0" "overflow-hidden" "transition-[width]"
|
|
|
+ "transition-200" "ease-in" "peer-hover/mute:w-[70px]"
|
|
|
+ "peer-focus/mute:w-[70px]" "hover:w-[70px]" "focus:w-[70px]"]
|
|
|
+ :style
|
|
|
+ {"--media-range-track-background" "rgba(255,255,255,.2)"
|
|
|
+ "--media-range-bar-color" service-color
|
|
|
+ "--media-range-thumb-background" service-color}}]
|
|
|
+ [:> MediaTimeDisplay {:showDuration true}]
|
|
|
+ [:span.control-spacer.grow]
|
|
|
+ [:> MediaSettingsMenuButton]
|
|
|
+ [:> MediaPipButton]
|
|
|
+ [:> MediaFullscreenButton]]]])})))
|