Restructure project folders
This commit is contained in:
212
ABS Client Mac/Audiobookshelf swift/Views/PlayerBar.swift
Normal file
212
ABS Client Mac/Audiobookshelf swift/Views/PlayerBar.swift
Normal file
@@ -0,0 +1,212 @@
|
||||
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)
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user