More search UI improvements across all the platforms

This commit is contained in:
Arkadiusz Fal 2021-09-26 19:40:25 +02:00
parent 4e0d7b60f7
commit f9396985c9
16 changed files with 241 additions and 197 deletions

View File

@ -2,7 +2,7 @@ extension Array where Element: Equatable {
func next(after element: Element) -> Element? { func next(after element: Element) -> Element? {
let idx = firstIndex(of: element) let idx = firstIndex(of: element)
if idx == nil { if idx.isNil {
return first return first
} }

View File

@ -225,12 +225,12 @@ final class InvidiousAPI: Service, ObservableObject {
.withParam("q", searchQuery(query.query)) .withParam("q", searchQuery(query.query))
.withParam("sort_by", query.sortBy.parameter) .withParam("sort_by", query.sortBy.parameter)
if let date = query.date?.rawValue { if let date = query.date, date != .any {
resource = resource.withParam("date", date) resource = resource.withParam("date", date.rawValue)
} }
if let duration = query.duration?.rawValue { if let duration = query.duration, duration != .any {
resource = resource.withParam("duration", duration) resource = resource.withParam("duration", duration.rawValue)
} }
return resource return resource

View File

@ -116,7 +116,7 @@ final class PlayerModel: ObservableObject {
self.saveTime() self.saveTime()
self.player?.replaceCurrentItem(with: self.playerItemWithMetadata(for: stream)) self.player?.replaceCurrentItem(with: self.playerItemWithMetadata(for: stream))
self.playback.stream = stream self.playback.stream = stream
if self.timeObserver == nil { if self.timeObserver.isNil {
self.addTimeObserver() self.addTimeObserver()
} }
self.player?.play() self.player?.play()
@ -201,7 +201,7 @@ final class PlayerModel: ObservableObject {
} }
fileprivate func composition(for stream: Stream) -> AVMutableComposition { fileprivate func composition(for stream: Stream) -> AVMutableComposition {
if compositions[stream] == nil { if compositions[stream].isNil {
compositions[stream] = AVMutableComposition() compositions[stream] = AVMutableComposition()
} }

View File

@ -29,8 +29,10 @@ final class RecentsModel: ObservableObject {
} }
func addQuery(_ query: String) { func addQuery(_ query: String) {
if !query.isEmpty {
open(.init(from: query)) open(.init(from: query))
} }
}
var presentedChannel: Channel? { var presentedChannel: Channel? {
if let recent = items.last(where: { $0.type == .channel }) { if let recent = items.last(where: { $0.type == .channel }) {

View File

@ -5,7 +5,7 @@ import SwiftUI
final class SearchModel: ObservableObject { final class SearchModel: ObservableObject {
@Published var store = Store<[Video]>() @Published var store = Store<[Video]>()
@Published var api: InvidiousAPI! @Published var api = InvidiousAPI()
@Published var query = SearchQuery() @Published var query = SearchQuery()
@Published var queryText = "" @Published var queryText = ""
@Published var querySuggestions = Store<[String]>() @Published var querySuggestions = Store<[String]>()
@ -30,10 +30,13 @@ final class SearchModel: ObservableObject {
resource = newResource resource = newResource
resource.addObserver(store) resource.addObserver(store)
if !query.isEmpty {
loadResourceIfNeededAndReplaceStore() loadResourceIfNeededAndReplaceStore()
} }
}
func resetQuery(_ query: SearchQuery) { func resetQuery(_ query: SearchQuery = SearchQuery()) {
self.query = query self.query = query
let newResource = api.search(query) let newResource = api.search(query)
@ -48,8 +51,11 @@ final class SearchModel: ObservableObject {
resource = newResource resource = newResource
resource.addObserver(store) resource.addObserver(store)
if !query.isEmpty {
loadResourceIfNeededAndReplaceStore() loadResourceIfNeededAndReplaceStore()
} }
}
func loadResourceIfNeededAndReplaceStore() { func loadResourceIfNeededAndReplaceStore() {
let currentResource = resource! let currentResource = resource!

View File

@ -3,7 +3,7 @@ import Foundation
final class SearchQuery: ObservableObject { final class SearchQuery: ObservableObject {
enum Date: String, CaseIterable, Identifiable, DefaultsSerializable { enum Date: String, CaseIterable, Identifiable, DefaultsSerializable {
case hour, today, week, month, year case any, hour, today, week, month, year
var id: SearchQuery.Date.RawValue { var id: SearchQuery.Date.RawValue {
rawValue rawValue
@ -15,7 +15,7 @@ final class SearchQuery: ObservableObject {
} }
enum Duration: String, CaseIterable, Identifiable, DefaultsSerializable { enum Duration: String, CaseIterable, Identifiable, DefaultsSerializable {
case short, long case any, short, long
var id: SearchQuery.Duration.RawValue { var id: SearchQuery.Duration.RawValue {
rawValue rawValue

View File

@ -36,7 +36,6 @@
373CFAC226966159003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; }; 373CFAC226966159003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; };
373CFAC32696616C003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; }; 373CFAC32696616C003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; };
373CFAC42696616C003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; }; 373CFAC42696616C003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; };
373CFAC926966188003CB2C6 /* SearchOptionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */; };
373CFACB26966264003CB2C6 /* SearchQuery.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFACA26966264003CB2C6 /* SearchQuery.swift */; }; 373CFACB26966264003CB2C6 /* SearchQuery.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFACA26966264003CB2C6 /* SearchQuery.swift */; };
373CFACC26966264003CB2C6 /* SearchQuery.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFACA26966264003CB2C6 /* SearchQuery.swift */; }; 373CFACC26966264003CB2C6 /* SearchQuery.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFACA26966264003CB2C6 /* SearchQuery.swift */; };
373CFACD26966264003CB2C6 /* SearchQuery.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFACA26966264003CB2C6 /* SearchQuery.swift */; }; 373CFACD26966264003CB2C6 /* SearchQuery.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFACA26966264003CB2C6 /* SearchQuery.swift */; };
@ -312,7 +311,6 @@
372915E52687E3B900F5A35B /* Defaults.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Defaults.swift; sourceTree = "<group>"; }; 372915E52687E3B900F5A35B /* Defaults.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Defaults.swift; sourceTree = "<group>"; };
373CFABD26966115003CB2C6 /* CoverSectionView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CoverSectionView.swift; sourceTree = "<group>"; }; 373CFABD26966115003CB2C6 /* CoverSectionView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CoverSectionView.swift; sourceTree = "<group>"; };
373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CoverSectionRowView.swift; sourceTree = "<group>"; }; 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CoverSectionRowView.swift; sourceTree = "<group>"; };
373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SearchOptionsView.swift; sourceTree = "<group>"; };
373CFACA26966264003CB2C6 /* SearchQuery.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SearchQuery.swift; sourceTree = "<group>"; }; 373CFACA26966264003CB2C6 /* SearchQuery.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SearchQuery.swift; sourceTree = "<group>"; };
373CFADA269663F1003CB2C6 /* Thumbnail.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Thumbnail.swift; sourceTree = "<group>"; }; 373CFADA269663F1003CB2C6 /* Thumbnail.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Thumbnail.swift; sourceTree = "<group>"; };
373CFAEA26975CBF003CB2C6 /* PlaylistFormView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlaylistFormView.swift; sourceTree = "<group>"; }; 373CFAEA26975CBF003CB2C6 /* PlaylistFormView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlaylistFormView.swift; sourceTree = "<group>"; };
@ -552,15 +550,6 @@
path = Views; path = Views;
sourceTree = "<group>"; sourceTree = "<group>";
}; };
371AAE2926CF143200901972 /* Options */ = {
isa = PBXGroup;
children = (
37B76E95268747C900CE5671 /* OptionsView.swift */,
373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */,
);
path = Options;
sourceTree = "<group>";
};
3748186426A762300084E870 /* Fixtures */ = { 3748186426A762300084E870 /* Fixtures */ = {
isa = PBXGroup; isa = PBXGroup;
children = ( children = (
@ -729,8 +718,8 @@
37D4B159267164AE00C925CA /* tvOS */ = { 37D4B159267164AE00C925CA /* tvOS */ = {
isa = PBXGroup; isa = PBXGroup;
children = ( children = (
371AAE2926CF143200901972 /* Options */,
373CFAEE2697A78B003CB2C6 /* AddToPlaylistView.swift */, 373CFAEE2697A78B003CB2C6 /* AddToPlaylistView.swift */,
37B76E95268747C900CE5671 /* OptionsView.swift */,
37BAB54B269B39FD00E75ED1 /* TVNavigationView.swift */, 37BAB54B269B39FD00E75ED1 /* TVNavigationView.swift */,
37D4B15E267164AF00C925CA /* Assets.xcassets */, 37D4B15E267164AF00C925CA /* Assets.xcassets */,
37D4B1AE26729DEB00C925CA /* Info.plist */, 37D4B1AE26729DEB00C925CA /* Info.plist */,
@ -1269,7 +1258,6 @@
376578872685429C00D4EA09 /* CaseIterable+Next.swift in Sources */, 376578872685429C00D4EA09 /* CaseIterable+Next.swift in Sources */,
37D4B1802671650A00C925CA /* PearvidiousApp.swift in Sources */, 37D4B1802671650A00C925CA /* PearvidiousApp.swift in Sources */,
3748187026A769D60084E870 /* DetailBadge.swift in Sources */, 3748187026A769D60084E870 /* DetailBadge.swift in Sources */,
373CFAC926966188003CB2C6 /* SearchOptionsView.swift in Sources */,
37A9965C26D6F8CA006E3224 /* VideosCellsHorizontal.swift in Sources */, 37A9965C26D6F8CA006E3224 /* VideosCellsHorizontal.swift in Sources */,
37BD07C92698FBDB003EBB87 /* ContentView.swift in Sources */, 37BD07C92698FBDB003EBB87 /* ContentView.swift in Sources */,
376B2E0926F920D600B1D64D /* SignInRequiredView.swift in Sources */, 376B2E0926F920D600B1D64D /* SignInRequiredView.swift in Sources */,

View File

@ -40,7 +40,7 @@ struct PearvidiousApp: App {
search.api = api search.api = api
subscriptions.api = api subscriptions.api = api
guard api.account == nil, instances.defaultAccount != nil else { guard api.account.isNil, instances.defaultAccount != nil else {
return return
} }

View File

@ -159,7 +159,7 @@ struct PlaylistsView: View {
} }
func selectEditedPlaylist() { func selectEditedPlaylist() {
if editedPlaylist == nil { if editedPlaylist.isNil {
selectPlaylist(nil) selectPlaylist(nil)
} }

View File

@ -61,6 +61,9 @@ struct TrendingView: View {
.foregroundColor(.secondary) .foregroundColor(.secondary)
categoryButton categoryButton
// only way to disable Menu animation is to
// force redraw of the view when it changes
.id(UUID())
} }
HStack { HStack {
@ -70,7 +73,6 @@ struct TrendingView: View {
countryButton countryButton
} }
} }
.transaction { t in t.animation = .none }
} }
} }
#endif #endif
@ -117,15 +119,9 @@ struct TrendingView: View {
} }
#else #else
Menu(category.name) { Picker("Category", selection: $category) {
ForEach(TrendingCategory.allCases) { category in ForEach(TrendingCategory.allCases) { category in
Button(action: { self.category = category }) { Text(category.name).tag(category)
if category == self.category {
Label(category.name, systemImage: "checkmark")
} else {
Text(category.name)
}
}
} }
} }
#endif #endif

View File

@ -5,9 +5,9 @@ import SwiftUI
struct SearchView: View { struct SearchView: View {
private var query: SearchQuery? private var query: SearchQuery?
@State private var searchSortOrder: SearchQuery.SortOrder = .relevance @State private var searchSortOrder = SearchQuery.SortOrder.relevance
@State private var searchDate: SearchQuery.Date? @State private var searchDate = SearchQuery.Date.any
@State private var searchDuration: SearchQuery.Duration? @State private var searchDuration = SearchQuery.Duration.any
@State private var presentingClearConfirmation = false @State private var presentingClearConfirmation = false
@State private var recentsChanged = false @State private var recentsChanged = false
@ -17,20 +17,28 @@ struct SearchView: View {
@EnvironmentObject<RecentsModel> private var recents @EnvironmentObject<RecentsModel> private var recents
@EnvironmentObject<SearchModel> private var state @EnvironmentObject<SearchModel> private var state
@State private var searchDebounceTimer: Timer?
@State private var recentSearchDebounceTimer: Timer?
init(_ query: SearchQuery? = nil) { init(_ query: SearchQuery? = nil) {
self.query = query self.query = query
} }
var body: some View { var body: some View {
Group {
if navigationStyle == .tab && state.queryText.isEmpty {
VStack { VStack {
if !recentItems.isEmpty { if navigationStyle == .tab && state.queryText.isEmpty {
recentQueries recentQueries
}
}
} else { } else {
#if os(tvOS)
ScrollView(.vertical, showsIndicators: false) {
filtersHorizontalStack
VideosCellsHorizontal(videos: state.store.collection)
}
.edgesIgnoringSafeArea(.horizontal)
#else
VideosView(videos: state.store.collection) VideosView(videos: state.store.collection)
#endif
if state.store.collection.isEmpty && !state.isLoading && !state.query.isEmpty { if state.store.collection.isEmpty && !state.isLoading && !state.query.isEmpty {
Text("No results") Text("No results")
@ -38,8 +46,8 @@ struct SearchView: View {
if searchFiltersActive { if searchFiltersActive {
Button("Reset search filters") { Button("Reset search filters") {
self.searchSortOrder = .relevance self.searchSortOrder = .relevance
self.searchDate = nil self.searchDate = .any
self.searchDuration = nil self.searchDuration = .any
} }
} }
@ -48,53 +56,29 @@ struct SearchView: View {
} }
} }
.toolbar { .toolbar {
#if os(iOS) #if !os(tvOS)
ToolbarItemGroup(placement: .bottomBar) { ToolbarItemGroup(placement: toolbarPlacement) {
Section { Section {
if !state.queryText.isEmpty { #if os(macOS)
HStack {
Text("Sort:") Text("Sort:")
.foregroundColor(.secondary) .foregroundColor(.secondary)
Menu(searchSortOrder.name) { searchSortOrderPicker
ForEach(SearchQuery.SortOrder.allCases) { sortOrder in
Button(sortOrder.name) {
searchSortOrder = sortOrder
} }
#else
Menu("Sort: \(searchSortOrder.name)") {
searchSortOrderPicker
} }
#endif
} }
.transaction { t in t.animation = .none }
Spacer() Spacer()
Text("Filter:") filtersMenu
.foregroundColor(.secondary) }
Menu(searchDuration?.name ?? "Duration") {
Button("All") {
searchDuration = nil
}
ForEach(SearchQuery.Duration.allCases) { duration in
Button(duration.name) {
searchDuration = duration
}
}
}
.foregroundColor(searchDuration.isNil ? .secondary : .accentColor)
Menu(searchDate?.name ?? "Date") {
Button("All") {
searchDate = nil
}
ForEach(SearchQuery.Date.allCases) { date in
Button(date.name) {
searchDate = date
}
}
}
.foregroundColor(searchDate.isNil ? .secondary : .accentColor)
}
}
.transaction { t in t.animation = .none }
}
#endif #endif
} }
.onAppear { .onAppear {
@ -109,8 +93,25 @@ struct SearchView: View {
.searchCompletion(suggestion) .searchCompletion(suggestion)
} }
} }
.onChange(of: state.queryText) { query in .onChange(of: state.queryText) { newQuery in
state.loadSuggestions(query) if newQuery.isEmpty {
state.resetQuery()
}
state.loadSuggestions(newQuery)
#if os(tvOS)
searchDebounceTimer?.invalidate()
recentSearchDebounceTimer?.invalidate()
searchDebounceTimer = Timer.scheduledTimer(withTimeInterval: 2, repeats: false) { _ in
state.changeQuery { query in query.query = newQuery }
}
recentSearchDebounceTimer = Timer.scheduledTimer(withTimeInterval: 10, repeats: false) { _ in
recents.addQuery(newQuery)
}
#endif
} }
.onSubmit(of: .search) { .onSubmit(of: .search) {
state.changeQuery { query in query.query = state.queryText } state.changeQuery { query in query.query = state.queryText }
@ -138,9 +139,26 @@ struct SearchView: View {
#endif #endif
} }
var toolbarPlacement: ToolbarItemPlacement {
#if os(iOS)
.bottomBar
#else
.automatic
#endif
}
var filtersActive: Bool {
searchDuration != .any || searchDate != .any
}
var recentQueries: some View { var recentQueries: some View {
VStack {
List { List {
Section(header: Text("Recents")) { Section(header: Text("Recents")) {
if recentItems.isEmpty {
Text("Search history is empty")
.foregroundColor(.secondary)
}
ForEach(recentItems) { item in ForEach(recentItems) { item in
Button(item.title) { Button(item.title) {
state.queryText = item.title state.queryText = item.title
@ -150,13 +168,20 @@ struct SearchView: View {
.swipeActions(edge: .trailing) { .swipeActions(edge: .trailing) {
clearButton(item) clearButton(item)
} }
#elseif os(tvOS)
.contextMenu {
clearButton(item)
}
#endif #endif
} }
} }
.redrawOn(change: recentsChanged) .redrawOn(change: recentsChanged)
if !recentItems.isEmpty {
clearAllButton clearAllButton
} }
}
}
#if os(iOS) #if os(iOS)
.listStyle(.insetGrouped) .listStyle(.insetGrouped)
#endif #endif
@ -183,10 +208,125 @@ struct SearchView: View {
} }
var searchFiltersActive: Bool { var searchFiltersActive: Bool {
searchDate != nil || searchDuration != nil searchDate != .any || searchDuration != .any
} }
var recentItems: [RecentItem] { var recentItems: [RecentItem] {
Defaults[.recentlyOpened].filter { $0.type == .query }.reversed() Defaults[.recentlyOpened].filter { $0.type == .query }.reversed()
} }
var searchSortOrderPicker: some View {
Picker("Sort", selection: $searchSortOrder) {
ForEach(SearchQuery.SortOrder.allCases) { sortOrder in
Text(sortOrder.name).tag(sortOrder)
}
}
}
#if os(tvOS)
var searchSortOrderButton: some View {
Button(action: { self.searchSortOrder = self.searchSortOrder.next() }) { Text(self.searchSortOrder.name)
.font(.system(size: 30))
.padding(.horizontal)
.padding(.vertical, 2)
}
.buttonStyle(.card)
.contextMenu {
ForEach(SearchQuery.SortOrder.allCases) { sortOrder in
Button(sortOrder.name) {
self.searchSortOrder = sortOrder
}
}
}
}
var searchDateButton: some View {
Button(action: { self.searchDate = self.searchDate.next() }) {
Text(self.searchDate.name)
.font(.system(size: 30))
.padding(.horizontal)
.padding(.vertical, 2)
}
.buttonStyle(.card)
.contextMenu {
ForEach(SearchQuery.Date.allCases) { searchDate in
Button(searchDate.name) {
self.searchDate = searchDate
}
}
}
}
var searchDurationButton: some View {
Button(action: { self.searchDuration = self.searchDuration.next() }) {
Text(self.searchDate.name)
.font(.system(size: 30))
.padding(.horizontal)
.padding(.vertical, 2)
}
.buttonStyle(.card)
.contextMenu {
ForEach(SearchQuery.Duration.allCases) { searchDuration in
Button(searchDuration.name) {
self.searchDuration = searchDuration
}
}
}
}
var filtersHorizontalStack: some View {
HStack {
HStack(spacing: 30) {
Text("Sort")
.foregroundColor(.secondary)
searchSortOrderButton
}
.frame(maxWidth: .infinity, alignment: .trailing)
HStack(spacing: 30) {
Text("Duration")
.foregroundColor(.secondary)
searchDurationButton
}
.frame(maxWidth: .infinity)
HStack(spacing: 30) {
Text("Date")
.foregroundColor(.secondary)
searchDateButton
}
.frame(maxWidth: .infinity, alignment: .leading)
}
.font(.system(size: 20))
}
#else
var filtersMenu: some View {
Menu(filtersActive ? "Filter: active" : "Filter") {
Picker(selection: $searchDuration, label: Text("Duration")) {
ForEach(SearchQuery.Duration.allCases) { duration in
Text(duration.name).tag(duration)
}
}
Picker("Upload date", selection: $searchDate) {
ForEach(SearchQuery.Date.allCases) { date in
Text(date.name).tag(date)
}
}
}
.foregroundColor(filtersActive ? .accentColor : .secondary)
.transaction { t in t.animation = .none }
}
#endif
}
struct SearchView_Previews: PreviewProvider {
static var previews: some View {
NavigationView {
SearchView(SearchQuery(query: "Is Google Evil"))
.environmentObject(NavigationModel())
.environmentObject(SearchModel())
.environmentObject(SubscriptionsModel())
}
}
} }

View File

@ -24,7 +24,7 @@ final class PlayerViewController: NSViewController {
override func loadView() { override func loadView() {
playerModel = PlayerModel(playback: playback, api: api, resolution: resolution) playerModel = PlayerModel(playback: playback, api: api, resolution: resolution)
guard playerModel.player == nil else { guard playerModel.player.isNil else {
return return
} }

View File

@ -43,7 +43,7 @@ struct AddToPlaylistView: View {
CoverSectionRowView { CoverSectionRowView {
Button("Add", action: addToPlaylist) Button("Add", action: addToPlaylist)
.disabled(currentPlaylist == nil) .disabled(currentPlaylist.isNil)
} }
} }

View File

@ -1,64 +0,0 @@
import Defaults
import SwiftUI
struct SearchOptionsView: View {
@Default(.searchSortOrder) private var searchSortOrder
@Default(.searchDate) private var searchDate
@Default(.searchDuration) private var searchDuration
var body: some View {
CoverSectionView("Search Options") {
CoverSectionRowView("Sort By") { searchSortOrderButton }
CoverSectionRowView("Upload date") { searchDateButton }
CoverSectionRowView("Duration") { searchDurationButton }
}
}
var searchSortOrderButton: some View {
Button(self.searchSortOrder.name) {
self.searchSortOrder = self.searchSortOrder.next()
}
.contextMenu {
ForEach(SearchQuery.SortOrder.allCases) { sortOrder in
Button(sortOrder.name) {
self.searchSortOrder = sortOrder
}
}
}
}
var searchDateButton: some View {
Button(self.searchDate?.name ?? "All") {
self.searchDate = self.searchDate == nil ? SearchQuery.Date.allCases.first : self.searchDate!.next(nilAtEnd: true)
}
.contextMenu {
ForEach(SearchQuery.Date.allCases) { searchDate in
Button(searchDate.name) {
self.searchDate = searchDate
}
}
Button("Reset") {
self.searchDate = nil
}
}
}
var searchDurationButton: some View {
Button(self.searchDuration?.name ?? "All") {
self.searchDuration = self.searchDuration == nil ? SearchQuery.Duration.allCases.first : self.searchDuration!.next(nilAtEnd: true)
}
.contextMenu {
ForEach(SearchQuery.Duration.allCases) { searchDuration in
Button(searchDuration.name) {
self.searchDuration = searchDuration
}
}
Button("Reset") {
self.searchDuration = nil
}
}
}
}

View File

@ -17,8 +17,6 @@ struct OptionsView: View {
VStack(alignment: .leading) { VStack(alignment: .leading) {
Spacer() Spacer()
tabSelectionOptions
CoverSectionView("View Options") { CoverSectionView("View Options") {
CoverSectionRowView("Show videos as") { nextLayoutButton } CoverSectionRowView("Show videos as") { nextLayoutButton }
} }
@ -42,18 +40,6 @@ struct OptionsView: View {
.background(.thinMaterial) .background(.thinMaterial)
} }
var tabSelectionOptions: some View {
VStack {
switch navigation.tabSelection {
case .search:
SearchOptionsView()
default:
EmptyView()
}
}
}
var nextLayoutButton: some View { var nextLayoutButton: some View {
Button(layout.name) { Button(layout.name) {
self.layout = layout.next() self.layout = layout.next()

View File

@ -4,7 +4,7 @@ import SwiftUI
struct TVNavigationView: View { struct TVNavigationView: View {
@EnvironmentObject<NavigationModel> private var navigation @EnvironmentObject<NavigationModel> private var navigation
@EnvironmentObject<PlaybackModel> private var playback @EnvironmentObject<PlaybackModel> private var playback
@EnvironmentObject<Recents> private var recents @EnvironmentObject<RecentsModel> private var recents
@EnvironmentObject<SearchModel> private var search @EnvironmentObject<SearchModel> private var search
@State private var showingOptions = false @State private var showingOptions = false
@ -34,16 +34,6 @@ struct TVNavigationView: View {
.tag(TabSelection.playlists) .tag(TabSelection.playlists)
SearchView() SearchView()
.searchable(text: $search.queryText) {
ForEach(search.querySuggestions.collection, id: \.self) { suggestion in
Text(suggestion)
.searchCompletion(suggestion)
}
}
.onChange(of: search.queryText) { newQuery in
search.loadSuggestions(newQuery)
search.changeQuery { query in query.query = newQuery }
}
.tabItem { Image(systemName: "magnifyingglass") } .tabItem { Image(systemName: "magnifyingglass") }
.tag(TabSelection.search) .tag(TabSelection.search)
} }