Files
yattee/Yattee/Views/MediaBrowser/MediaBrowserViewOptionsSheet.swift
Arkadiusz Fal 181cf2f73a Fix collapsed sheets on macOS across the app
Add macOS-only minimum frame sizing to sheets that wrap a
NavigationStack/Form without intrinsic size, so they render properly
instead of collapsing to just the toolbar. Affects Customize Home,
subscription/channel view options, playlist create/edit, search
filters, media browser options, instance picker, log filters, preset
editor, and legacy data import result.
2026-04-19 11:33:16 +02:00

130 lines
3.7 KiB
Swift

//
// MediaBrowserViewOptionsSheet.swift
// Yattee
//
// Sheet for customizing media browser view options.
//
import SwiftUI
struct MediaBrowserViewOptionsSheet: View {
@Environment(\.dismiss) private var dismiss
let sourceType: MediaSourceType
@Binding var sortOrder: MediaBrowserSortOrder
@Binding var sortAscending: Bool
@Binding var showOnlyPlayable: Bool
private var availableSortOptions: [MediaBrowserSortOrder] {
MediaBrowserSortOrder.availableOptions(for: sourceType)
}
var body: some View {
NavigationStack {
#if os(tvOS)
listContent
#else
formContent
#endif
}
#if os(iOS)
.presentationDetents([.height(280)])
.presentationDragIndicator(.visible)
#endif
#if os(macOS)
.frame(minWidth: 420, minHeight: 320)
#endif
.onAppear {
// Reset sort order if current selection is not available for this source type
if !availableSortOptions.contains(sortOrder) {
sortOrder = .name
}
}
}
@ViewBuilder
private var sharedOptions: some View {
Toggle("mediaBrowser.viewOptions.showOnlyPlayable", isOn: $showOnlyPlayable)
PlatformMenuPicker(String(localized: "mediaBrowser.viewOptions.sortBy"), selection: $sortOrder) {
ForEach(availableSortOptions) { order in
Label(order.displayName, systemImage: order.systemImage)
.tag(order)
}
}
Picker("mediaBrowser.viewOptions.order", selection: $sortAscending) {
Label(String(localized: "mediaBrowser.viewOptions.ascending"), systemImage: "arrow.up")
.tag(true)
Label(String(localized: "mediaBrowser.viewOptions.descending"), systemImage: "arrow.down")
.tag(false)
}
.pickerStyle(.segmented)
.listRowBackground(Color.clear)
.listRowInsets(EdgeInsets(top: 8, leading: 0, bottom: 8, trailing: 0))
}
#if os(tvOS)
private var listContent: some View {
List {
Section {
sharedOptions
}
}
.scrollClipDisabled()
.padding(.horizontal, 40)
.padding(.vertical, 24)
}
#endif
private var formContent: some View {
Form {
Section {
sharedOptions
}
}
.navigationTitle("mediaBrowser.viewOptions.title")
#if os(iOS)
.navigationBarTitleDisplayMode(.inline)
#endif
.toolbar {
ToolbarItem(placement: .confirmationAction) {
Button(role: .cancel) {
dismiss()
} label: {
Label(String(localized: "common.close"), systemImage: "xmark")
.labelStyle(.iconOnly)
}
}
}
}
}
// MARK: - Preview
#Preview("Local Folder") {
@Previewable @State var sortOrder: MediaBrowserSortOrder = .name
@Previewable @State var sortAscending = true
@Previewable @State var showOnlyPlayable = false
MediaBrowserViewOptionsSheet(
sourceType: .localFolder,
sortOrder: $sortOrder,
sortAscending: $sortAscending,
showOnlyPlayable: $showOnlyPlayable
)
}
#Preview("WebDAV") {
@Previewable @State var sortOrder: MediaBrowserSortOrder = .name
@Previewable @State var sortAscending = true
@Previewable @State var showOnlyPlayable = false
MediaBrowserViewOptionsSheet(
sourceType: .webdav,
sortOrder: $sortOrder,
sortAscending: $sortAscending,
showOnlyPlayable: $showOnlyPlayable
)
}