import Foundation import SwiftUI struct PlayerControlsView: View { let content: Content @Environment(\.navigationStyle) private var navigationStyle @EnvironmentObject private var model @EnvironmentObject private var navigation init(@ViewBuilder content: @escaping () -> Content) { self.content = content() } var body: some View { ZStack(alignment: .bottomLeading) { content #if !os(tvOS) .frame(minHeight: 0, maxHeight: .infinity) .padding(.bottom, 50) #endif #if !os(tvOS) controls #endif } } private var controls: some View { HStack { Button(action: { model.presentingPlayer.toggle() }) { HStack { VStack(alignment: .leading, spacing: 3) { Text(model.currentItem?.video.title ?? "Not playing") .font(.system(size: 14).bold()) .foregroundColor(model.currentItem.isNil ? .secondary : .accentColor) .lineLimit(1) Text(model.currentItem?.video.author ?? "Yattee v0.1") .fontWeight(model.currentItem.isNil ? .light : .bold) .font(.system(size: 10)) .foregroundColor(.secondary) .lineLimit(1) } Spacer() } .contentShape(Rectangle()) } .padding(.vertical, 20) #if !os(tvOS) .keyboardShortcut("o") #endif ZStack(alignment: .bottom) { HStack { Group { if model.isPlaying { Button(action: { model.pause() }) { Label("Pause", systemImage: "pause.fill") } } else { Button(action: { model.play() }) { Label("Play", systemImage: "play.fill") } .disabled(model.player.currentItem.isNil) } } .font(.system(size: 30)) .frame(minWidth: 30) #if !os(tvOS) .keyboardShortcut("p") #endif Button(action: { model.advanceToNextItem() }) { Label("Next", systemImage: "forward.fill") } .disabled(model.queue.isEmpty) } ProgressView(value: progressViewValue, total: progressViewTotal) .progressViewStyle(.linear) #if os(iOS) .offset(x: 0, y: 8) .frame(maxWidth: 60) #else .offset(x: 0, y: 15) .frame(maxWidth: 70) #endif } } .buttonStyle(.plain) .labelStyle(.iconOnly) .padding(.horizontal) .frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: 55) .padding(.vertical, 0) .background(.ultraThinMaterial) .borderTop(height: 0.4, color: Color("PlayerControlsBorderColor")) .borderBottom(height: navigationStyle == .sidebar ? 0 : 0.4, color: Color("PlayerControlsBorderColor")) #if !os(tvOS) .onSwipeGesture(up: { model.presentingPlayer = true }) #endif } private var progressViewValue: Double { [model.time?.seconds, model.videoDuration].compactMap { $0 }.min() ?? 0 } private var progressViewTotal: Double { model.playerItemDuration?.seconds ?? model.currentVideo?.length ?? progressViewValue } } struct PlayerControlsView_Previews: PreviewProvider { static var previews: some View { PlayerControlsView { VStack { Spacer() Text("Hello") Spacer() } } .injectFixtureEnvironmentObjects() } }