213 lines
7.1 KiB
Swift
213 lines
7.1 KiB
Swift
import SwiftUI
|
||
|
||
struct PlayerBar: View {
|
||
@Environment(AppState.self) private var app
|
||
@AppStorage("skipDurationSeconds") private var skipSeconds: Int = 30
|
||
@State private var scrubbing: Bool = false
|
||
@State private var scrubValue: Double = 0
|
||
|
||
var body: some View {
|
||
if let item = app.currentItem {
|
||
VStack(spacing: 0) {
|
||
Divider()
|
||
content(item: item)
|
||
.padding(.horizontal, 16)
|
||
.padding(.vertical, 10)
|
||
.background(.bar)
|
||
}
|
||
.transition(.move(edge: .bottom).combined(with: .opacity))
|
||
} else if app.isPreparingPlayback {
|
||
VStack(spacing: 0) {
|
||
Divider()
|
||
HStack(spacing: 12) {
|
||
ProgressView().controlSize(.small)
|
||
Text("Wiedergabe wird vorbereitet …").font(.subheadline).foregroundStyle(.secondary)
|
||
Spacer()
|
||
}
|
||
.padding(.horizontal, 16)
|
||
.padding(.vertical, 14)
|
||
.background(.bar)
|
||
}
|
||
}
|
||
}
|
||
|
||
@ViewBuilder
|
||
private func content(item: LibraryItem) -> some View {
|
||
HStack(spacing: 14) {
|
||
cover(item: item)
|
||
|
||
VStack(alignment: .leading, spacing: 2) {
|
||
Text(item.title).font(.subheadline).bold().lineLimit(1)
|
||
Text(item.author).font(.caption).foregroundStyle(.secondary).lineLimit(1)
|
||
if let err = app.player.errorMessage {
|
||
Text(err).font(.caption2).foregroundStyle(.red).lineLimit(1)
|
||
}
|
||
}
|
||
.frame(minWidth: 160, idealWidth: 200, maxWidth: 240, alignment: .leading)
|
||
|
||
transportControls
|
||
|
||
scrubber
|
||
.frame(minWidth: 200)
|
||
|
||
rateMenu
|
||
|
||
Spacer(minLength: 0)
|
||
|
||
statusIndicator
|
||
|
||
Button {
|
||
app.stopPlayback()
|
||
} label: {
|
||
Image(systemName: "xmark.circle.fill")
|
||
.foregroundStyle(.secondary)
|
||
}
|
||
.buttonStyle(.plain)
|
||
.help("Wiedergabe beenden")
|
||
}
|
||
}
|
||
|
||
private func cover(item: LibraryItem) -> some View {
|
||
Group {
|
||
if let url = app.client.coverURL(itemId: item.id) {
|
||
AsyncImage(url: url) { phase in
|
||
if let img = phase.image {
|
||
img.resizable().aspectRatio(contentMode: .fill)
|
||
} else {
|
||
Color.gray.opacity(0.3)
|
||
}
|
||
}
|
||
} else {
|
||
Color.gray.opacity(0.3)
|
||
}
|
||
}
|
||
.frame(width: 48, height: 48)
|
||
.clipShape(RoundedRectangle(cornerRadius: 6))
|
||
}
|
||
|
||
private var transportControls: some View {
|
||
HStack(spacing: 14) {
|
||
Button { app.skip(by: -Double(skipSeconds)) } label: {
|
||
Image(systemName: skipBackImage).font(.system(size: 18))
|
||
}
|
||
.buttonStyle(.plain)
|
||
.disabled(!app.player.isReady)
|
||
|
||
Button { app.togglePlay() } label: {
|
||
Image(systemName: app.player.isPlaying ? "pause.circle.fill" : "play.circle.fill")
|
||
.font(.system(size: 34))
|
||
}
|
||
.buttonStyle(.plain)
|
||
.disabled(!app.player.isReady)
|
||
.keyboardShortcut(.space, modifiers: [])
|
||
|
||
Button { app.skip(by: Double(skipSeconds)) } label: {
|
||
Image(systemName: skipForwardImage).font(.system(size: 18))
|
||
}
|
||
.buttonStyle(.plain)
|
||
.disabled(!app.player.isReady)
|
||
}
|
||
}
|
||
|
||
private var skipForwardImage: String {
|
||
switch skipSeconds {
|
||
case ...10: return "goforward.10"
|
||
case 11...15: return "goforward.15"
|
||
case 16...30: return "goforward.30"
|
||
case 31...45: return "goforward.45"
|
||
case 46...60: return "goforward.60"
|
||
default: return "goforward.90"
|
||
}
|
||
}
|
||
|
||
private var skipBackImage: String {
|
||
switch skipSeconds {
|
||
case ...10: return "gobackward.10"
|
||
case 11...15: return "gobackward.15"
|
||
case 16...30: return "gobackward.30"
|
||
case 31...45: return "gobackward.45"
|
||
case 46...60: return "gobackward.60"
|
||
default: return "gobackward.90"
|
||
}
|
||
}
|
||
|
||
private var scrubber: some View {
|
||
VStack(spacing: 2) {
|
||
Slider(
|
||
value: Binding(
|
||
get: { scrubbing ? scrubValue : app.player.absoluteCurrentTime },
|
||
set: { scrubValue = $0; scrubbing = true }
|
||
),
|
||
in: 0...max(app.player.totalDuration, 1),
|
||
onEditingChanged: { editing in
|
||
if !editing {
|
||
app.seekAbsolute(scrubValue)
|
||
scrubbing = false
|
||
}
|
||
}
|
||
)
|
||
.disabled(!app.player.isReady)
|
||
HStack {
|
||
Text(formatTime(scrubbing ? scrubValue : app.player.absoluteCurrentTime))
|
||
.font(.caption2.monospacedDigit())
|
||
.foregroundStyle(.secondary)
|
||
Spacer()
|
||
Text(formatTime(app.player.totalDuration))
|
||
.font(.caption2.monospacedDigit())
|
||
.foregroundStyle(.secondary)
|
||
}
|
||
}
|
||
}
|
||
|
||
private var rateMenu: some View {
|
||
Menu {
|
||
ForEach([0.75, 1.0, 1.25, 1.5, 1.75, 2.0], id: \.self) { r in
|
||
Button {
|
||
app.setRate(Float(r))
|
||
} label: {
|
||
HStack {
|
||
Text(String(format: "%.2g×", r))
|
||
if abs(Double(app.player.rate) - r) < 0.01 {
|
||
Image(systemName: "checkmark")
|
||
}
|
||
}
|
||
}
|
||
}
|
||
} label: {
|
||
Text(String(format: "%.2g×", Double(app.player.rate)))
|
||
.font(.caption.monospacedDigit())
|
||
.padding(.horizontal, 8).padding(.vertical, 4)
|
||
.overlay(Capsule().stroke(Color.secondary.opacity(0.4)))
|
||
}
|
||
.menuStyle(.borderlessButton)
|
||
.fixedSize()
|
||
.help("Geschwindigkeit")
|
||
}
|
||
|
||
private var statusIndicator: some View {
|
||
HStack(spacing: 4) {
|
||
Circle()
|
||
.fill(app.network.isOnline ? .green : .orange)
|
||
.frame(width: 6, height: 6)
|
||
if app.sync.queuedCount > 0 {
|
||
Text("\(app.sync.queuedCount)")
|
||
.font(.caption2)
|
||
.foregroundStyle(.secondary)
|
||
}
|
||
}
|
||
.help(app.network.isOnline ? "Online – Fortschritt wird synchronisiert" : "Offline – \(app.sync.queuedCount) Eintrag/Einträge wartend")
|
||
}
|
||
|
||
private func formatTime(_ seconds: Double) -> String {
|
||
guard seconds.isFinite, seconds >= 0 else { return "0:00" }
|
||
let total = Int(seconds)
|
||
let h = total / 3600
|
||
let m = (total % 3600) / 60
|
||
let s = total % 60
|
||
if h > 0 {
|
||
return String(format: "%d:%02d:%02d", h, m, s)
|
||
}
|
||
return String(format: "%d:%02d", m, s)
|
||
}
|
||
}
|