yattee/Shared/Player/VideoPlayerView.swift

203 lines
6.7 KiB
Swift
Raw Normal View History

2021-07-18 22:32:46 +00:00
import AVKit
import Defaults
2021-07-18 22:32:46 +00:00
import Siesta
import SwiftUI
struct VideoPlayerView: View {
2021-11-08 16:29:35 +00:00
static let defaultAspectRatio = 16 / 9.0
2021-09-18 20:36:42 +00:00
static var defaultMinimumHeightLeft: Double {
2021-08-22 19:13:33 +00:00
#if os(macOS)
300
#else
200
#endif
}
2021-11-03 23:00:17 +00:00
@State private var playerSize: CGSize = .zero
@State private var fullScreenDetails = false
2021-07-18 22:32:46 +00:00
2021-11-28 14:37:55 +00:00
@Environment(\.colorScheme) private var colorScheme
2021-08-22 19:13:33 +00:00
#if os(iOS)
2021-11-28 14:37:55 +00:00
@Environment(\.presentationMode) private var presentationMode
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
2021-08-22 19:13:33 +00:00
@Environment(\.verticalSizeClass) private var verticalSizeClass
#endif
2021-08-16 22:46:18 +00:00
2021-12-24 19:20:05 +00:00
@EnvironmentObject<AccountsModel> private var accounts
@EnvironmentObject<PlayerModel> private var player
2021-09-25 08:18:22 +00:00
var body: some View {
2021-11-04 22:01:27 +00:00
#if os(macOS)
HSplitView {
content
}
2021-12-24 19:20:05 +00:00
.onOpenURL { OpenURLHandler(accounts: accounts, player: player).handle($0) }
.frame(minWidth: 950, minHeight: 700)
2021-11-04 22:01:27 +00:00
#else
GeometryReader { geometry in
2021-11-03 23:00:17 +00:00
HStack(spacing: 0) {
content
}
.onAppear {
self.playerSize = geometry.size
}
.onChange(of: geometry.size) { size in
self.playerSize = size
}
2021-11-04 22:01:27 +00:00
}
2021-12-17 19:53:36 +00:00
.navigationBarHidden(true)
2021-11-04 22:01:27 +00:00
#endif
2021-07-18 22:32:46 +00:00
}
var content: some View {
Group {
2021-10-28 17:14:55 +00:00
Group {
#if os(tvOS)
2021-10-28 17:14:55 +00:00
player.playerView
#else
GeometryReader { geometry in
VStack(spacing: 0) {
#if os(iOS)
if verticalSizeClass == .regular {
PlaybackBar()
}
#elseif os(macOS)
PlaybackBar()
#endif
2021-07-18 22:32:46 +00:00
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-22 19:13:33 +00:00
}
}
2021-08-22 19:13:33 +00:00
#if os(iOS)
2021-11-08 16:29:35 +00:00
.onSwipeGesture(
up: {
withAnimation {
2021-12-24 19:20:05 +00:00
fullScreenDetails = true
2021-11-08 16:29:35 +00:00
}
},
2021-12-24 19:20:05 +00:00
down: { player.hide() }
2021-11-08 16:29:35 +00:00
)
#endif
2021-11-28 14:37:55 +00:00
.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 14:37:55 +00:00
.background(colorScheme == .dark ? Color.black : Color.white)
.modifier(VideoDetailsPaddingModifier(geometry: geometry, aspectRatio: player.controller?.aspectRatio, fullScreen: fullScreenDetails))
2021-08-22 19:13:33 +00:00
}
#endif
}
2021-12-02 20:35:42 +00:00
.background(colorScheme == .dark ? Color.black : Color.white)
#if os(macOS)
2021-12-02 20:35:42 +00:00
.frame(minWidth: 650)
#endif
#if os(iOS)
if sidebarQueue {
PlayerQueueView(sidebarQueue: .constant(true), fullScreen: $fullScreenDetails)
.frame(maxWidth: 350)
2021-07-18 22:32:46 +00:00
}
#elseif os(macOS)
2021-11-03 23:00:17 +00:00
if Defaults[.playerSidebar] != .never {
PlayerQueueView(sidebarQueue: sidebarQueueBinding, fullScreen: $fullScreenDetails)
2021-12-02 19:22:55 +00:00
.frame(minWidth: 300)
2021-11-03 23:00:17 +00:00
}
2021-07-18 22:32:46 +00:00
#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-18 22:32:46 +00:00
}
.contentShape(Rectangle())
.frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: geometry.size.width / VideoPlayerView.defaultAspectRatio)
2021-07-18 22:32:46 +00:00
}
2021-08-22 19:13:33 +00:00
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-03 23:00:17 +00:00
var sidebarQueue: Bool {
switch Defaults[.playerSidebar] {
case .never:
return false
case .always:
return true
case .whenFits:
return playerSize.width > 900
}
2021-11-03 23:00:17 +00:00
}
2021-11-03 23:00:17 +00:00
var sidebarQueueBinding: Binding<Bool> {
Binding(
get: { sidebarQueue },
set: { _ in }
)
}
2021-08-22 19:13:33 +00:00
}
struct VideoPlayerView_Previews: PreviewProvider {
static var previews: some View {
VideoPlayerView()
.injectFixtureEnvironmentObjects()
2021-08-22 19:13:33 +00:00
}
2021-07-18 22:32:46 +00:00
}