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

498 lines
19 KiB
Swift
Raw Normal View History

2021-07-19 04:02:46 +05:30
import AVKit
#if os(iOS)
import CoreMotion
#endif
import Defaults
2021-07-19 04:02:46 +05:30
import Siesta
import SwiftUI
struct VideoPlayerView: View {
2022-05-29 17:59:43 +05:30
#if os(iOS)
2022-05-30 01:43:21 +05:30
static let hiddenOffset = max(UIScreen.main.bounds.height, UIScreen.main.bounds.width) + 100
2022-05-29 17:59:43 +05:30
#endif
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
2022-02-28 02:01:17 +05:30
@State private var hoveringPlayer = false
@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)
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
2021-08-23 00:43:33 +05:30
@Environment(\.verticalSizeClass) private var verticalSizeClass
@Default(.enterFullscreenInLandscape) private var enterFullscreenInLandscape
@Default(.honorSystemOrientationLock) private var honorSystemOrientationLock
2022-05-29 19:04:40 +05:30
@Default(.lockOrientationInFullScreen) private var lockOrientationInFullScreen
@State private var motionManager: CMMotionManager!
@State private var orientation = UIInterfaceOrientation.portrait
@State private var lastOrientation: UIInterfaceOrientation?
2022-02-28 02:01:17 +05:30
#elseif os(macOS)
var mouseLocation: CGPoint { NSEvent.mouseLocation }
2021-08-23 00:43:33 +05:30
#endif
2021-08-17 04:16:18 +05:30
2022-05-30 02:00:00 +05:30
#if os(iOS)
2022-05-30 01:43:21 +05:30
@State private var viewVerticalOffset = Self.hiddenOffset
#endif
2021-12-25 00:50:05 +05:30
@EnvironmentObject<AccountsModel> private var accounts
2022-02-17 01:53:11 +05:30
@EnvironmentObject<PlayerControlsModel> private var playerControls
@EnvironmentObject<PlayerModel> private var player
2022-06-08 02:57:48 +05:30
@EnvironmentObject<ThumbnailsModel> private var thumbnails
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
}
2021-12-25 00:50:05 +05:30
.onOpenURL { OpenURLHandler(accounts: accounts, player: player).handle($0) }
.frame(minWidth: 950, minHeight: 700)
2021-11-05 03:31:27 +05:30
#else
2022-04-04 04:03:09 +05:30
GeometryReader { geometry in
HStack(spacing: 0) {
content
.onAppear {
playerSize = geometry.size
}
}
.onChange(of: geometry.size) { size in
self.playerSize = size
}
2022-04-04 04:03:09 +05:30
.onChange(of: fullScreenDetails) { value in
player.backend.setNeedsDrawing(!value)
}
#if os(iOS)
.onReceive(NotificationCenter.default.publisher(for: UIDevice.orientationDidChangeNotification)) { _ in
handleOrientationDidChangeNotification()
}
2022-05-29 17:59:43 +05:30
.onChange(of: player.presentingPlayer) { newValue in
if newValue {
2022-05-30 01:43:21 +05:30
viewVerticalOffset = 0
2022-05-30 02:00:00 +05:30
configureOrientationUpdatesBasedOnAccelerometer()
2022-05-29 17:59:43 +05:30
} else {
2022-05-30 02:00:00 +05:30
if Defaults[.lockPortraitWhenBrowsing] {
Orientation.lockOrientation(.portrait, andRotateTo: .portrait)
} else {
Orientation.lockOrientation(.allButUpsideDown)
}
2022-05-29 17:59:43 +05:30
2022-05-30 02:00:00 +05:30
motionManager?.stopAccelerometerUpdates()
motionManager = nil
viewVerticalOffset = Self.hiddenOffset
2022-05-29 17:59:43 +05:30
}
}
2022-05-30 02:00:00 +05:30
#endif
2021-11-05 03:31:27 +05:30
}
2022-05-30 02:00:00 +05:30
#if os(iOS)
2022-05-30 01:43:21 +05:30
.offset(y: viewVerticalOffset)
.animation(.easeIn(duration: 0.2), value: viewVerticalOffset)
2022-05-30 02:00:00 +05:30
#endif
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)
2022-03-28 00:52:13 +05:30
playerView
2022-02-17 01:53:11 +05:30
.ignoresSafeArea(.all, edges: .all)
2022-03-28 00:52:13 +05:30
.onMoveCommand { direction in
if direction == .left {
playerControls.resetTimer()
player.backend.seek(relative: .secondsInDefaultTimescale(-10))
}
if direction == .right {
playerControls.resetTimer()
player.backend.seek(relative: .secondsInDefaultTimescale(10))
}
if direction == .up {
playerControls.show()
playerControls.resetTimer()
}
if direction == .down {
playerControls.show()
playerControls.resetTimer()
}
}
#else
GeometryReader { geometry in
VStack(spacing: 0) {
2022-06-08 02:57:48 +05:30
if player.playingInPictureInPicture {
pictureInPicturePlaceholder(geometry: geometry)
} else {
2022-03-28 00:52:13 +05:30
playerView
#if !os(tvOS)
2022-02-17 01:53:11 +05:30
.modifier(
VideoPlayerSizeModifier(
geometry: geometry,
aspectRatio: player.avPlayerBackend.controller?.aspectRatio,
fullScreen: playerControls.playingFullscreen
)
2022-02-17 01:53:11 +05:30
)
2022-06-08 02:57:48 +05:30
.overlay(playerPlaceholder(geometry: geometry))
2022-03-28 00:52:13 +05:30
#endif
2021-08-23 00:43:33 +05:30
}
}
2022-02-17 01:53:11 +05:30
.frame(maxWidth: fullScreenLayout ? .infinity : nil, maxHeight: fullScreenLayout ? .infinity : nil)
2022-02-28 02:01:17 +05:30
.onHover { hovering in
hoveringPlayer = hovering
hovering ? playerControls.show() : playerControls.hide()
}
2022-05-29 17:59:43 +05:30
#if !os(macOS)
.gesture(
2022-05-29 19:05:02 +05:30
DragGesture(coordinateSpace: .global)
.onChanged { value in
guard player.presentingPlayer else {
return // swiftlint:disable:this implicit_return
}
let drag = value.translation.height
guard drag > 0 else {
2022-05-29 19:05:02 +05:30
return // swiftlint:disable:this implicit_return
}
guard drag < 100 else {
player.hide()
return
}
2022-05-29 19:05:02 +05:30
withAnimation(.easeInOut(duration: 0.2)) {
2022-05-30 01:43:21 +05:30
viewVerticalOffset = drag
}
}
.onEnded { _ in
2022-05-30 01:43:21 +05:30
if viewVerticalOffset > 100 {
2022-05-29 19:05:02 +05:30
player.backend.setNeedsDrawing(false)
player.hide()
} else {
2022-05-30 01:43:21 +05:30
viewVerticalOffset = 0
2022-05-29 19:05:02 +05:30
player.backend.setNeedsDrawing(true)
player.show()
}
}
)
2022-05-29 17:59:43 +05:30
#else
2022-05-29 19:05:02 +05:30
.onAppear(perform: {
NSEvent.addLocalMonitorForEvents(matching: [.mouseMoved]) {
if hoveringPlayer {
playerControls.resetTimer()
}
2022-02-28 02:01:17 +05:30
2022-05-29 19:05:02 +05:30
return $0
}
})
#endif
2022-05-29 19:05:02 +05:30
.background(Color.black)
2022-03-28 00:52:13 +05:30
#if !os(tvOS)
if !playerControls.playingFullscreen {
Group {
#if os(iOS)
if verticalSizeClass == .regular {
VideoDetails(sidebarQueue: sidebarQueueBinding, fullScreen: $fullScreenDetails)
}
2022-03-28 00:52:13 +05:30
#else
VideoDetails(sidebarQueue: sidebarQueueBinding, fullScreen: $fullScreenDetails)
#endif
}
.background(colorScheme == .dark ? Color.black : Color.white)
.modifier(VideoDetailsPaddingModifier(
geometry: geometry,
aspectRatio: player.avPlayerBackend.controller?.aspectRatio,
fullScreen: fullScreenDetails
))
2022-02-17 01:53:11 +05:30
}
2022-03-28 00:52:13 +05:30
#endif
2021-08-23 00:43:33 +05:30
}
#endif
}
2022-02-17 01:53:11 +05:30
.background(((colorScheme == .dark || fullScreenLayout) ? Color.black : Color.white).edgesIgnoringSafeArea(.all))
#if os(macOS)
2021-12-03 02:05:42 +05:30
.frame(minWidth: 650)
#endif
2022-02-17 01:53:11 +05:30
if !playerControls.playingFullscreen {
#if os(iOS)
if sidebarQueue {
PlayerQueueView(sidebarQueue: .constant(true), fullScreen: $fullScreenDetails)
.frame(maxWidth: 350)
}
#elseif os(macOS)
if Defaults[.playerSidebar] != .never {
PlayerQueueView(sidebarQueue: sidebarQueueBinding, fullScreen: $fullScreenDetails)
.frame(minWidth: 300)
}
#endif
}
2021-07-19 04:02:46 +05:30
}
2022-02-17 01:53:11 +05:30
.ignoresSafeArea(.all, edges: fullScreenLayout ? .vertical : Edge.Set())
2022-03-28 00:52:13 +05:30
#if os(iOS)
2022-02-28 02:01:17 +05:30
.statusBar(hidden: playerControls.playingFullscreen)
.navigationBarHidden(true)
#endif
2022-02-17 01:53:11 +05:30
}
2022-03-28 00:52:13 +05:30
var playerView: some View {
ZStack(alignment: .top) {
switch player.activeBackend {
case .mpv:
player.mpvPlayerView
.overlay(GeometryReader { proxy in
Color.clear
.onAppear {
player.playerSize = proxy.size
}
.onChange(of: proxy.size) { _ in
player.playerSize = proxy.size
}
})
case .appleAVPlayer:
player.avPlayerView
2022-05-21 02:53:14 +05:30
#if os(iOS)
.onAppear {
player.pipController = .init(playerLayer: player.playerLayerView.playerLayer)
let pipDelegate = PiPDelegate()
pipDelegate.player = player
player.pipDelegate = pipDelegate
player.pipController!.delegate = pipDelegate
player.playerLayerView.playerLayer.player = player.avPlayerBackend.avPlayer
}
#endif
2022-03-28 00:52:13 +05:30
}
#if !os(tvOS)
PlayerGestures()
#endif
2022-06-08 02:57:48 +05:30
PlayerControls(player: player, thumbnails: thumbnails)
2022-03-28 00:52:13 +05:30
}
}
2022-02-17 01:53:11 +05:30
var fullScreenLayout: Bool {
2022-03-28 00:52:13 +05:30
#if os(iOS)
2022-02-28 02:01:17 +05:30
playerControls.playingFullscreen || verticalSizeClass == .compact
#else
playerControls.playingFullscreen
#endif
}
2022-06-08 02:57:48 +05:30
@ViewBuilder func playerPlaceholder(geometry: GeometryProxy) -> some View {
if player.currentItem.isNil {
ZStack(alignment: .topLeading) {
HStack {
Spacer()
2022-06-08 02:57:48 +05:30
VStack {
Spacer()
VStack(spacing: 10) {
#if !os(tvOS)
Image(systemName: "ticket")
.font(.system(size: 120))
#endif
}
Spacer()
}
2022-06-08 02:57:48 +05:30
.foregroundColor(.gray)
Spacer()
}
2022-06-08 02:57:48 +05:30
#if os(iOS)
Button {
player.hide()
} label: {
Image(systemName: "xmark")
.font(.system(size: 40))
}
.buttonStyle(.plain)
.padding(10)
.foregroundColor(.gray)
#endif
}
.background(Color.black)
.contentShape(Rectangle())
.frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: geometry.size.width / Self.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())
2022-05-21 01:23:17 +05:30
.frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: geometry.size.width / Self.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(iOS)
private func configureOrientationUpdatesBasedOnAccelerometer() {
if UIDevice.current.orientation.isLandscape,
enterFullscreenInLandscape,
2022-02-17 01:53:11 +05:30
!playerControls.playingFullscreen,
!player.playingInPictureInPicture
{
DispatchQueue.main.async {
player.enterFullScreen()
}
}
guard !honorSystemOrientationLock, motionManager.isNil else {
return
}
motionManager = CMMotionManager()
motionManager.accelerometerUpdateInterval = 0.2
motionManager.startAccelerometerUpdates(to: OperationQueue()) { data, _ in
guard player.presentingPlayer, !player.playingInPictureInPicture, !data.isNil else {
return
}
guard let acceleration = data?.acceleration else {
return
}
var orientation = UIInterfaceOrientation.unknown
if acceleration.x >= 0.65 {
orientation = .landscapeLeft
} else if acceleration.x <= -0.65 {
orientation = .landscapeRight
} else if acceleration.y <= -0.65 {
orientation = .portrait
} else if acceleration.y >= 0.65 {
orientation = .portraitUpsideDown
}
guard lastOrientation != orientation else {
return
}
lastOrientation = orientation
if orientation.isLandscape {
DispatchQueue.main.asyncAfter(deadline: .now() + 0.8) {
guard enterFullscreenInLandscape else {
return
}
player.enterFullScreen()
let orientationLockMask = orientation == .landscapeLeft ?
UIInterfaceOrientationMask.landscapeLeft : .landscapeRight
Orientation.lockOrientation(orientationLockMask, andRotateTo: orientation)
2022-05-29 19:04:40 +05:30
guard lockOrientationInFullScreen else {
return
}
player.lockedOrientation = orientation
}
} else {
guard abs(acceleration.z) <= 0.74,
player.lockedOrientation.isNil,
enterFullscreenInLandscape,
2022-05-29 19:04:40 +05:30
!lockOrientationInFullScreen
else {
return
}
DispatchQueue.main.asyncAfter(deadline: .now() + 0.4) {
player.exitFullScreen()
}
Orientation.lockOrientation(.portrait)
}
}
}
private func handleOrientationDidChangeNotification() {
2022-05-30 01:43:21 +05:30
viewVerticalOffset = viewVerticalOffset == 0 ? 0 : Self.hiddenOffset
let newOrientation = UIApplication.shared.windows.first?.windowScene?.interfaceOrientation
if newOrientation?.isLandscape ?? false,
player.presentingPlayer,
2022-05-29 19:04:40 +05:30
lockOrientationInFullScreen,
!player.lockedOrientation.isNil
{
Orientation.lockOrientation(.landscape, andRotateTo: newOrientation)
return
}
guard player.presentingPlayer, enterFullscreenInLandscape, honorSystemOrientationLock else {
return
}
if UIDevice.current.orientation.isLandscape {
DispatchQueue.main.async {
player.lockedOrientation = newOrientation
player.enterFullScreen()
}
} else {
DispatchQueue.main.async {
player.exitFullScreen()
}
DispatchQueue.main.asyncAfter(deadline: .now() + 0.8) {
player.exitFullScreen()
}
}
}
#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
}