1
0
mirror of https://github.com/yattee/yattee.git synced 2024-12-14 06:10:32 +05:30
yattee/Shared/Player/VideoPlayerView.swift

223 lines
7.2 KiB
Swift
Raw Normal View History

2021-07-19 04:02:46 +05:30
import AVKit
import Defaults
2021-07-19 04:02:46 +05:30
import Siesta
import SwiftUI
struct VideoPlayerView: View {
2021-11-08 21:59:35 +05:30
static let defaultAspectRatio = 16 / 9.0
2021-09-19 02:06:42 +05:30
static var defaultMinimumHeightLeft: Double {
2021-08-23 00:43:33 +05:30
#if os(macOS)
300
#else
200
#endif
}
2021-11-04 04:30:17 +05:30
@State private var playerSize: CGSize = .zero
@State private var fullScreenDetails = false
2021-07-19 04:02:46 +05:30
2021-11-28 20:07:55 +05:30
@Environment(\.colorScheme) private var colorScheme
2021-08-23 00:43:33 +05:30
#if os(iOS)
2021-11-28 20:07:55 +05:30
@Environment(\.presentationMode) private var presentationMode
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
2021-08-23 00:43:33 +05:30
@Environment(\.verticalSizeClass) private var verticalSizeClass
#endif
2021-08-17 04:16:18 +05:30
@EnvironmentObject<PlayerModel> private var player
2021-09-25 13:48:22 +05:30
var body: some View {
2021-11-05 03:31:27 +05:30
#if os(macOS)
HSplitView {
content
}
.onOpenURL(perform: handleOpenedURL)
.frame(minWidth: 950, minHeight: 700)
2021-11-05 03:31:27 +05:30
#else
GeometryReader { geometry in
2021-11-04 04:30:17 +05:30
HStack(spacing: 0) {
content
}
.onAppear {
self.playerSize = geometry.size
}
.onChange(of: geometry.size) { size in
self.playerSize = size
}
2021-11-05 03:31:27 +05:30
}
2021-12-18 01:23:36 +05:30
.navigationBarHidden(true)
2021-11-05 03:31:27 +05:30
#endif
2021-07-19 04:02:46 +05:30
}
var content: some View {
Group {
2021-10-28 22:44:55 +05:30
Group {
#if os(tvOS)
2021-10-28 22:44:55 +05:30
player.playerView
#else
GeometryReader { geometry in
VStack(spacing: 0) {
#if os(iOS)
if verticalSizeClass == .regular {
PlaybackBar()
}
#elseif os(macOS)
PlaybackBar()
#endif
2021-07-19 04:02:46 +05:30
if player.currentItem.isNil {
playerPlaceholder(geometry: geometry)
} else if player.playingInPictureInPicture {
pictureInPicturePlaceholder(geometry: geometry)
} else {
player.playerView
.modifier(
VideoPlayerSizeModifier(
geometry: geometry,
aspectRatio: player.controller?.aspectRatio
)
)
2021-08-23 00:43:33 +05:30
}
}
2021-08-23 00:43:33 +05:30
#if os(iOS)
2021-11-08 21:59:35 +05:30
.onSwipeGesture(
up: {
withAnimation {
fullScreen = true
}
},
2021-11-28 20:07:55 +05:30
down: { presentationMode.wrappedValue.dismiss() }
2021-11-08 21:59:35 +05:30
)
#endif
2021-11-28 20:07:55 +05:30
.background(Color.black)
Group {
#if os(iOS)
if verticalSizeClass == .regular {
VideoDetails(sidebarQueue: sidebarQueueBinding, fullScreen: $fullScreenDetails)
}
#else
VideoDetails(sidebarQueue: sidebarQueueBinding, fullScreen: $fullScreenDetails)
#endif
}
2021-11-28 20:07:55 +05:30
.background(colorScheme == .dark ? Color.black : Color.white)
.modifier(VideoDetailsPaddingModifier(geometry: geometry, aspectRatio: player.controller?.aspectRatio, fullScreen: fullScreenDetails))
2021-08-23 00:43:33 +05:30
}
#endif
}
2021-12-03 02:05:42 +05:30
.background(colorScheme == .dark ? Color.black : Color.white)
#if os(macOS)
2021-12-03 02:05:42 +05:30
.frame(minWidth: 650)
#endif
#if os(iOS)
if sidebarQueue {
PlayerQueueView(sidebarQueue: .constant(true), fullScreen: $fullScreenDetails)
.frame(maxWidth: 350)
2021-07-19 04:02:46 +05:30
}
#elseif os(macOS)
2021-11-04 04:30:17 +05:30
if Defaults[.playerSidebar] != .never {
PlayerQueueView(sidebarQueue: sidebarQueueBinding, fullScreen: $fullScreenDetails)
2021-12-03 00:52:55 +05:30
.frame(minWidth: 300)
2021-11-04 04:30:17 +05:30
}
2021-07-19 04:02:46 +05:30
#endif
}
}
func playerPlaceholder(geometry: GeometryProxy) -> some View {
HStack {
Spacer()
VStack {
Spacer()
VStack(spacing: 10) {
#if !os(tvOS)
Image(systemName: "ticket")
.font(.system(size: 120))
#endif
}
Spacer()
}
.foregroundColor(.gray)
Spacer()
2021-07-19 04:02:46 +05:30
}
.contentShape(Rectangle())
.frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: geometry.size.width / VideoPlayerView.defaultAspectRatio)
2021-07-19 04:02:46 +05:30
}
2021-08-23 00:43:33 +05:30
func pictureInPicturePlaceholder(geometry: GeometryProxy) -> some View {
HStack {
Spacer()
VStack {
Spacer()
VStack(spacing: 10) {
#if !os(tvOS)
Image(systemName: "pip")
.font(.system(size: 120))
#endif
Text("Playing in Picture in Picture")
}
Spacer()
}
.foregroundColor(.gray)
Spacer()
}
.contextMenu {
Button {
player.closePiP()
} label: {
Label("Exit Picture in Picture", systemImage: "pip.exit")
}
}
.contentShape(Rectangle())
.frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: geometry.size.width / VideoPlayerView.defaultAspectRatio)
}
2021-11-04 04:30:17 +05:30
var sidebarQueue: Bool {
switch Defaults[.playerSidebar] {
case .never:
return false
case .always:
return true
case .whenFits:
return playerSize.width > 900
}
2021-11-04 04:30:17 +05:30
}
2021-11-04 04:30:17 +05:30
var sidebarQueueBinding: Binding<Bool> {
Binding(
get: { sidebarQueue },
set: { _ in }
)
}
#if !os(tvOS)
func handleOpenedURL(_ url: URL) {
guard !player.accounts.current.isNil else {
return
}
let parser = VideoURLParser(url: url)
guard let id = parser.id else {
return
}
player.accounts.api.video(id).load().onSuccess { response in
if let video: Video = response.typedContent() {
self.player.playNow(video, at: parser.time)
self.player.show()
}
}
}
#endif
2021-08-23 00:43:33 +05:30
}
struct VideoPlayerView_Previews: PreviewProvider {
static var previews: some View {
VideoPlayerView()
.injectFixtureEnvironmentObjects()
2021-08-23 00:43:33 +05:30
}
2021-07-19 04:02:46 +05:30
}