1
0
mirror of https://github.com/yattee/yattee.git synced 2024-12-14 22:30:32 +05:30
yattee/Shared/Search/SearchView.swift

700 lines
22 KiB
Swift
Raw Normal View History

2021-06-28 20:32:13 +05:30
import Defaults
2021-06-28 16:13:07 +05:30
import Siesta
2021-06-11 18:06:26 +05:30
import SwiftUI
struct SearchView: View {
2021-09-25 17:47:58 +05:30
private var query: SearchQuery?
2021-06-28 16:13:07 +05:30
@State private var searchSortOrder = SearchQuery.SortOrder.relevance
@State private var searchDate = SearchQuery.Date.any
@State private var searchDuration = SearchQuery.Duration.any
2021-09-19 18:12:47 +05:30
@State private var recentsChanged = false
#if os(tvOS)
@State private var searchDebounce = Debounce()
@State private var recentsDebounce = Debounce()
private var recents = RecentsModel.shared
#endif
2021-11-09 23:13:15 +05:30
@State private var favoriteItem: FavoriteItem?
2021-09-25 17:47:58 +05:30
@Environment(\.navigationStyle) private var navigationStyle
@ObservedObject private var accounts = AccountsModel.shared
@ObservedObject private var state = SearchModel.shared
2021-12-03 02:05:25 +05:30
private var favorites = FavoritesModel.shared
2022-12-12 01:28:52 +05:30
private var navigation = NavigationModel.shared
2021-09-19 16:36:54 +05:30
2022-09-01 00:54:46 +05:30
@Default(.recentlyOpened) private var recentlyOpened
@Default(.saveRecents) private var saveRecents
2022-11-12 01:58:40 +05:30
@Default(.showHome) private var showHome
2022-12-12 05:48:29 +05:30
@Default(.searchListingStyle) private var searchListingStyle
@Default(.showSearchSuggestions) private var showSearchSuggestions
2021-09-29 17:15:00 +05:30
private var videos = [Video]()
2021-11-28 20:07:55 +05:30
init(_ query: SearchQuery? = nil, videos: [Video] = []) {
2021-09-19 16:36:54 +05:30
self.query = query
2021-09-29 17:15:00 +05:30
self.videos = videos
2021-09-19 16:36:54 +05:30
}
#if os(iOS)
var body: some View {
VStack {
2021-11-28 20:07:55 +05:30
VStack {
2022-12-09 05:45:19 +05:30
if accounts.app.supportsSearchSuggestions, state.query.query != state.queryText {
2021-11-28 20:07:55 +05:30
SearchSuggestions()
2022-08-17 21:04:25 +05:30
.opacity(state.queryText.isEmpty ? 0 : 1)
2021-11-28 20:07:55 +05:30
} else {
results
}
}
2022-08-17 21:04:25 +05:30
.backport
2023-02-07 02:21:57 +05:30
.scrollDismissesKeyboardInteractively()
}
.environment(\.listingStyle, searchListingStyle)
.toolbar {
ToolbarItem(placement: .principal) {
if #available(iOS 15, *) {
FocusableSearchTextField()
} else {
SearchTextField()
}
}
ToolbarItem(placement: .navigationBarTrailing) {
searchMenu
}
}
.navigationBarTitleDisplayMode(.inline)
.ignoresSafeArea(.keyboard, edges: .bottom)
.navigationTitle("Search")
.onAppear {
if let query {
state.queryText = query.query
state.resetQuery(query)
updateFavoriteItem()
}
if !videos.isEmpty {
state.store.replace(ContentItem.array(of: videos))
}
}
.onChange(of: accounts.current) { _ in
state.reloadQuery()
}
.onChange(of: state.queryText) { newQuery in
if newQuery.isEmpty {
favoriteItem = nil
state.resetQuery()
} else {
updateFavoriteItem()
}
state.loadSuggestions(newQuery)
}
.onChange(of: searchSortOrder) { order in
state.changeQuery { query in
query.sortBy = order
updateFavoriteItem()
}
}
.onChange(of: searchDate) { date in
state.changeQuery { query in
query.date = date
updateFavoriteItem()
}
}
.onChange(of: searchDuration) { duration in
state.changeQuery { query in
query.duration = duration
updateFavoriteItem()
}
}
}
#elseif os(tvOS)
var body: some View {
VStack {
2021-11-28 20:07:55 +05:30
ZStack {
results
}
}
.environment(\.listingStyle, searchListingStyle)
.onAppear {
if let query {
state.queryText = query.query
state.resetQuery(query)
updateFavoriteItem()
}
2021-11-28 20:07:55 +05:30
if !videos.isEmpty {
state.store.replace(ContentItem.array(of: videos))
}
}
.onChange(of: accounts.current) { _ in
state.reloadQuery()
}
.onChange(of: state.queryText) { newQuery in
if newQuery.isEmpty {
favoriteItem = nil
state.resetQuery()
} else {
updateFavoriteItem()
}
if showSearchSuggestions {
state.loadSuggestions(newQuery)
}
searchDebounce.invalidate()
recentsDebounce.invalidate()
searchDebounce.debouncing(2) {
state.changeQuery { query in
query.query = newQuery
}
}
recentsDebounce.debouncing(10) {
recents.addQuery(newQuery)
}
}
.onChange(of: searchSortOrder) { order in
state.changeQuery { query in
query.sortBy = order
updateFavoriteItem()
}
}
.onChange(of: searchDate) { date in
state.changeQuery { query in
query.date = date
updateFavoriteItem()
}
}
.onChange(of: searchDuration) { duration in
state.changeQuery { query in
query.duration = duration
updateFavoriteItem()
}
}
.searchable(text: $state.queryText) {
if !state.queryText.isEmpty {
ForEach(state.querySuggestions, id: \.self) { suggestion in
Text(suggestion)
.searchCompletion(suggestion)
}
2021-09-19 18:12:47 +05:30
}
}
2021-07-08 04:09:18 +05:30
}
#elseif os(macOS)
var body: some View {
ZStack {
results
if accounts.app.supportsSearchSuggestions, state.query.query != state.queryText, showSearchSuggestions {
HStack {
Spacer()
SearchSuggestions()
.borderLeading(width: 1, color: Color("ControlsBorderColor"))
.frame(maxWidth: 262)
.opacity(state.queryText.isEmpty ? 0 : 1)
}
}
}
.environment(\.listingStyle, searchListingStyle)
.toolbar {
ToolbarItemGroup(placement: toolbarPlacement) {
2022-12-12 05:48:29 +05:30
ListingStyleButtons(listingStyle: $searchListingStyle)
HideWatchedButtons()
2023-05-23 22:24:53 +05:30
HideShortsButtons()
2022-02-04 23:08:29 +05:30
FavoriteButton(item: favoriteItem)
.id(favoriteItem?.id)
2021-11-09 23:13:15 +05:30
2021-10-21 03:51:50 +05:30
if accounts.app.supportsSearchFilters {
Section {
2022-02-04 23:08:29 +05:30
HStack {
Text("Sort:")
.foregroundColor(.secondary)
searchSortOrderPicker
}
2021-10-21 03:51:50 +05:30
}
.transaction { t in t.animation = .none }
2021-11-09 23:13:15 +05:30
}
2021-11-09 23:13:15 +05:30
if accounts.app.supportsSearchFilters {
2021-10-21 03:51:50 +05:30
filtersMenu
}
2021-11-28 20:07:55 +05:30
if #available(macOS 12, *) {
FocusableSearchTextField()
} else {
SearchTextField()
}
2021-09-25 17:47:58 +05:30
}
}
.onAppear {
if let query {
state.queryText = query.query
state.resetQuery(query)
updateFavoriteItem()
}
if !videos.isEmpty {
state.store.replace(ContentItem.array(of: videos))
}
2021-11-09 23:13:15 +05:30
}
.onChange(of: accounts.current) { _ in
state.reloadQuery()
2021-11-09 23:13:15 +05:30
}
.onChange(of: state.queryText) { newQuery in
if newQuery.isEmpty {
favoriteItem = nil
state.resetQuery()
} else {
updateFavoriteItem()
}
state.loadSuggestions(newQuery)
2021-11-09 23:13:15 +05:30
}
.onChange(of: searchSortOrder) { order in
state.changeQuery { query in
query.sortBy = order
updateFavoriteItem()
2022-08-17 21:04:25 +05:30
}
2021-12-01 16:52:19 +05:30
}
.onChange(of: searchDate) { date in
state.changeQuery { query in
query.date = date
updateFavoriteItem()
}
}
.onChange(of: searchDuration) { duration in
state.changeQuery { query in
query.duration = duration
updateFavoriteItem()
}
2022-12-10 06:49:36 +05:30
}
.frame(minWidth: Constants.contentViewMinWidth)
.navigationTitle("Search")
2022-12-10 06:49:36 +05:30
}
#endif
2021-06-12 03:24:00 +05:30
2022-12-10 06:49:36 +05:30
#if os(iOS)
var searchMenu: some View {
Menu {
if accounts.app.supportsSearchFilters {
searchSortOrderPicker
.pickerStyle(.menu)
Picker(selection: $searchDuration, label: Text("Duration")) {
ForEach(SearchQuery.Duration.allCases) { duration in
Text(duration.name).tag(duration)
}
}
.pickerStyle(.menu)
Picker("Upload date", selection: $searchDate) {
ForEach(SearchQuery.Date.allCases) { date in
Text(date.name).tag(date)
}
}
.pickerStyle(.menu)
}
2022-12-12 01:28:52 +05:30
if !state.query.isEmpty {
Section {
FavoriteButton(item: favoriteItem)
}
}
2022-12-12 05:48:29 +05:30
ListingStyleButtons(listingStyle: $searchListingStyle)
2023-02-25 21:12:18 +05:30
Section {
HideWatchedButtons()
2023-05-23 22:24:53 +05:30
HideShortsButtons()
2023-02-25 21:12:18 +05:30
}
2022-12-10 06:49:36 +05:30
Section {
2022-12-12 03:45:56 +05:30
SettingsButtons()
2022-12-10 06:49:36 +05:30
}
} label: {
2022-12-11 18:58:16 +05:30
HStack {
Image(systemName: "chevron.down.circle.fill")
.foregroundColor(.accentColor)
.imageScale(.large)
2022-12-11 18:58:16 +05:30
}
2022-12-10 06:49:36 +05:30
}
}
#endif
2021-11-28 20:07:55 +05:30
private var results: some View {
VStack {
if showRecentQueries {
recentQueries
} else {
VerticalCells(items: state.store.collection, allowEmpty: state.query.isEmpty) {
if shouldDisplayHeader {
header
2021-11-28 20:07:55 +05:30
}
}
.environment(\.loadMoreContentHandler) { state.loadNextPage() }
if noResults {
Text("No results")
if searchFiltersActive {
Button("Reset search filters", action: resetFilters)
}
Spacer()
}
2021-11-28 20:07:55 +05:30
}
}
}
private var toolbarPlacement: ToolbarItemPlacement {
#if os(iOS)
2021-12-03 02:05:25 +05:30
accounts.app.supportsSearchFilters || favorites.isEnabled ? .bottomBar : .automatic
#else
2023-05-08 01:15:18 +05:30
.automatic
#endif
}
2021-11-28 20:07:55 +05:30
private var showRecentQueries: Bool {
navigationStyle == .tab && saveRecents && state.queryText.isEmpty
}
2021-11-28 20:07:55 +05:30
private var filtersActive: Bool {
searchDuration != .any || searchDate != .any
}
private func resetFilters() {
searchSortOrder = .relevance
searchDate = .any
searchDuration = .any
}
2021-11-28 20:07:55 +05:30
private var noResults: Bool {
2022-11-27 16:12:16 +05:30
state.store.collection.isEmpty && !state.isLoading && !state.query.isEmpty
}
2021-11-28 20:07:55 +05:30
private var recentQueries: some View {
VStack {
List {
Section(header: Text("Recents")) {
2022-09-01 00:54:46 +05:30
if recentlyOpened.isEmpty {
Text("Search history is empty")
.foregroundColor(.secondary)
2021-09-19 18:12:47 +05:30
}
2022-09-02 18:45:54 +05:30
ForEach(recentlyOpened.reversed(), id: \.tag) { item in
2022-12-12 01:28:52 +05:30
recentItemControl(item)
.lineLimit(1)
.truncationMode(.middle)
.foregroundColor(.accentColor)
}
2021-09-19 18:12:47 +05:30
}
.redrawOn(change: recentsChanged)
2022-08-21 21:09:06 +05:30
Section(footer: Color.clear.frame(minHeight: 80)) {
clearHistoryButton
}
}
2021-09-19 18:12:47 +05:30
}
#if os(iOS)
2021-11-08 21:59:35 +05:30
.listStyle(.insetGrouped)
2021-09-19 18:12:47 +05:30
#endif
}
2022-12-12 01:28:52 +05:30
@ViewBuilder private func recentItemControl(_ item: RecentItem) -> some View {
#if os(tvOS)
recentItemButton(item)
#else
if recentItemIsNavigationLink(item) {
recentItemNavigationLink(item)
} else {
recentItemButton(item)
}
#endif
}
private func recentItemNavigationLink(_ item: RecentItem) -> some View {
NavigationLink(destination: recentItemNavigationLinkDestination(item)) {
recentItemLabel(item)
}
.contextMenu { recentItemContextMenu(item) }
2022-12-12 01:28:52 +05:30
}
@ViewBuilder private func recentItemNavigationLinkDestination(_ item: RecentItem) -> some View {
switch item.type {
case .channel:
if let channel = item.channel {
ChannelVideosView(channel: channel)
}
case .playlist:
if let playlist = item.playlist {
ChannelPlaylistView(playlist: playlist)
}
default:
EmptyView()
}
}
func recentItemIsNavigationLink(_ item: RecentItem) -> Bool {
switch item.type {
case .channel:
return navigationStyle == .tab
case .playlist:
return navigationStyle == .tab
default:
return false
}
}
2022-06-14 21:42:06 +05:30
private func recentItemButton(_ item: RecentItem) -> some View {
Button {
switch item.type {
case .query:
2022-08-17 21:04:25 +05:30
state.queryText = item.title
2022-06-14 21:42:06 +05:30
state.changeQuery { query in query.query = item.title }
NavigationModel.shared.hideKeyboard()
2022-06-14 21:42:06 +05:30
updateFavoriteItem()
RecentsModel.shared.add(item)
2022-06-14 21:42:06 +05:30
case .channel:
guard let channel = item.channel else {
return
}
NavigationModel.shared.openChannel(
2022-06-14 21:42:06 +05:30
channel,
2022-06-30 13:35:32 +05:30
navigationStyle: navigationStyle
2022-06-14 21:42:06 +05:30
)
case .playlist:
guard let playlist = item.playlist else {
return
}
NavigationModel.shared.openChannelPlaylist(
2022-06-14 21:42:06 +05:30
playlist,
2022-06-30 13:35:32 +05:30
navigationStyle: navigationStyle
2022-06-14 21:42:06 +05:30
)
}
} label: {
2022-12-12 01:28:52 +05:30
recentItemLabel(item)
2022-06-14 21:42:06 +05:30
}
.contextMenu { recentItemContextMenu(item) }
}
private func recentItemContextMenu(_ item: RecentItem) -> some View {
Group {
2022-06-14 21:42:06 +05:30
removeButton(item)
2022-08-21 21:09:06 +05:30
2022-08-14 22:29:21 +05:30
#if os(tvOS)
Button("Cancel", role: .cancel) {}
#endif
2022-06-14 21:42:06 +05:30
}
}
2022-12-12 01:28:52 +05:30
private func recentItemLabel(_ item: RecentItem) -> some View {
let systemImage = item.type == .query ? "magnifyingglass" :
item.type == .channel ? RecentsModel.symbolSystemImage(item.title) :
"list.and.film"
return Label(item.title, systemImage: systemImage)
}
private func removeButton(_ item: RecentItem) -> some View {
2021-11-28 20:07:55 +05:30
Button {
RecentsModel.shared.close(item)
2021-11-28 20:07:55 +05:30
recentsChanged.toggle()
} label: {
Label("Remove", systemImage: "trash")
2021-09-19 18:12:47 +05:30
}
2021-11-28 20:07:55 +05:30
}
2021-09-19 18:12:47 +05:30
2022-08-21 21:09:06 +05:30
private var clearHistoryButton: some View {
2021-11-28 20:07:55 +05:30
Button {
NavigationModel.shared.presentAlert(
2022-08-21 21:09:06 +05:30
Alert(
title: Text("Are you sure you want to clear search history?"),
message: Text("This cannot be reverted"),
primaryButton: .destructive(Text("Clear")) {
RecentsModel.shared.clear()
2022-08-21 21:09:06 +05:30
recentsChanged.toggle()
},
secondaryButton: .cancel()
)
)
2021-11-28 20:07:55 +05:30
} label: {
2022-08-21 21:09:06 +05:30
Label("Clear Search History...", systemImage: "trash.fill")
2021-09-19 18:12:47 +05:30
}
2022-08-21 21:09:06 +05:30
.labelStyle(.titleOnly)
.foregroundColor(Color("AppRedColor"))
2021-09-19 18:12:47 +05:30
}
2021-11-28 20:07:55 +05:30
private var searchFiltersActive: Bool {
searchDate != .any || searchDuration != .any
2021-07-08 04:09:18 +05:30
}
2021-09-19 18:12:47 +05:30
2021-11-28 20:07:55 +05:30
private var searchSortOrderPicker: some View {
Picker("Sort", selection: $searchSortOrder) {
ForEach(SearchQuery.SortOrder.allCases) { sortOrder in
Text(sortOrder.name).tag(sortOrder)
}
}
}
#if os(tvOS)
2021-11-28 20:07:55 +05:30
private 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
}
}
}
}
2021-11-28 20:07:55 +05:30
private 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
}
}
}
}
2021-11-28 20:07:55 +05:30
private var searchDurationButton: some View {
Button(action: { self.searchDuration = self.searchDuration.next() }) {
2021-09-29 17:15:00 +05:30
Text(self.searchDuration.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
}
}
}
}
2021-11-28 20:07:55 +05:30
private var filtersHorizontalStack: some View {
HStack {
HStack(spacing: 30) {
Text("Sort")
.foregroundColor(.secondary)
searchSortOrderButton
}
2021-11-09 23:13:15 +05:30
.frame(maxWidth: 300, alignment: .trailing)
HStack(spacing: 30) {
Text("Duration")
.foregroundColor(.secondary)
searchDurationButton
}
2021-11-09 23:13:15 +05:30
.frame(maxWidth: 300)
HStack(spacing: 30) {
Text("Date")
.foregroundColor(.secondary)
searchDateButton
}
2021-11-09 23:13:15 +05:30
.frame(maxWidth: 300, alignment: .leading)
}
.font(.system(size: 30))
}
#else
2021-11-28 20:07:55 +05:30
private 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
2021-11-09 23:13:15 +05:30
private func updateFavoriteItem() {
favoriteItem = FavoriteItem(section: .searchQuery(
2022-08-17 21:04:25 +05:30
state.queryText,
searchDate.rawValue,
searchDuration.rawValue,
searchSortOrder.rawValue
2021-11-09 23:13:15 +05:30
))
}
2023-04-22 23:36:30 +05:30
var shouldDisplayHeader: Bool {
#if os(tvOS)
!state.query.isEmpty
#else
false
#endif
}
var header: some View {
HStack {
clearButton
#if os(tvOS)
if accounts.app.supportsSearchFilters {
filtersHorizontalStack
}
#endif
FavoriteButton(item: favoriteItem)
.id(favoriteItem?.id)
.labelStyle(.iconOnly)
.font(.system(size: 25))
Spacer()
ListingStyleButtons(listingStyle: $searchListingStyle)
HideWatchedButtons()
2023-05-23 22:24:53 +05:30
HideShortsButtons()
2023-04-22 23:36:30 +05:30
}
.labelStyle(.iconOnly)
.padding(.leading, 30)
.padding(.bottom, 15)
.padding(.trailing, 30)
}
var clearButton: some View {
Button {
state.queryText = ""
} label: {
Label("Clear", systemImage: "xmark")
.labelStyle(.iconOnly)
}
.font(.caption)
}
}
struct SearchView_Previews: PreviewProvider {
static var previews: some View {
NavigationView {
2021-09-29 17:15:00 +05:30
SearchView(SearchQuery(query: "Is Google Evil"), videos: Video.fixtures(30))
.injectFixtureEnvironmentObjects()
}
}
2021-06-11 18:06:26 +05:30
}