Search performance improvements (fix #209)

This commit is contained in:
Arkadiusz Fal 2022-08-05 00:30:09 +02:00
parent 2d5e34594a
commit af82cd3177
4 changed files with 65 additions and 49 deletions

View File

@ -6,14 +6,14 @@ final class SearchModel: ObservableObject {
@Published var store = Store<[ContentItem]>() @Published var store = Store<[ContentItem]>()
@Published var page: SearchPage? @Published var page: SearchPage?
var accounts = AccountsModel()
@Published var query = SearchQuery() @Published var query = SearchQuery()
@Published var queryText = "" @Published var queryText = ""
@Published var querySuggestions = Store<[String]>()
@Published var suggestionsText = "" @Published var suggestionsText = ""
@Published var suggestionSelection = ""
@Published var fieldIsFocused = false @Published var querySuggestions = Store<[String]>()
var accounts = AccountsModel()
private var resource: Resource! private var resource: Resource!
var isLoading: Bool { var isLoading: Bool {
@ -49,10 +49,9 @@ final class SearchModel: ObservableObject {
page = nil page = nil
store.replace([]) store.replace([])
if !query.isEmpty {
resource = newResource resource = newResource
resource.addObserver(store) resource.addObserver(store)
if !query.isEmpty {
loadResource() loadResource()
} }
} }
@ -74,7 +73,12 @@ final class SearchModel: ObservableObject {
} }
} }
private var suggestionsDebounceTimer: Timer? var suggestionsResource: Resource? { didSet {
oldValue?.removeObservers(ownedBy: querySuggestions)
oldValue?.cancelLoadIfUnobserved()
objectWillChange.send()
}}
func loadSuggestions(_ query: String) { func loadSuggestions(_ query: String) {
guard !query.isEmpty else { guard !query.isEmpty else {
@ -82,15 +86,11 @@ final class SearchModel: ObservableObject {
return return
} }
suggestionsDebounceTimer?.invalidate() DispatchQueue.main.async {
self.suggestionsResource = self.accounts.api.searchSuggestions(query: query)
self.suggestionsResource?.addObserver(self.querySuggestions)
suggestionsDebounceTimer = Timer.scheduledTimer(withTimeInterval: 0.2, repeats: false) { _ in if let request = self.suggestionsResource?.loadIfNeeded() {
let resource = self.accounts.api.searchSuggestions(query: query)
resource.addObserver(self.querySuggestions)
resource.loadIfNeeded()
if let request = resource.loadIfNeeded() {
request.onSuccess { response in request.onSuccess { response in
if let suggestions: [String] = response.typedContent() { if let suggestions: [String] = response.typedContent() {
self.querySuggestions = Store<[String]>(suggestions) self.querySuggestions = Store<[String]>(suggestions)

View File

@ -1,3 +1,4 @@
import Repeat
import SwiftUI import SwiftUI
struct SearchTextField: View { struct SearchTextField: View {
@ -7,9 +8,16 @@ struct SearchTextField: View {
@EnvironmentObject<RecentsModel> private var recents @EnvironmentObject<RecentsModel> private var recents
@EnvironmentObject<SearchModel> private var state @EnvironmentObject<SearchModel> private var state
@Binding var queryText: String
@Binding var favoriteItem: FavoriteItem? @Binding var favoriteItem: FavoriteItem?
init(favoriteItem: Binding<FavoriteItem?>? = nil) { private var queryDebouncer = Debouncer(.milliseconds(800))
init(
queryText: Binding<String>,
favoriteItem: Binding<FavoriteItem?>? = nil
) {
_queryText = queryText
_favoriteItem = favoriteItem ?? .constant(nil) _favoriteItem = favoriteItem ?? .constant(nil)
} }
@ -28,17 +36,24 @@ struct SearchTextField: View {
.padding(.horizontal, 8) .padding(.horizontal, 8)
.opacity(0.8) .opacity(0.8)
#endif #endif
TextField("Search...", text: $state.queryText) { TextField("Search...", text: $queryText) {
state.changeQuery { query in state.changeQuery { query in
query.query = state.queryText query.query = state.queryText
navigation.hideKeyboard() navigation.hideKeyboard()
} }
recents.addQuery(state.queryText, navigation: navigation) recents.addQuery(state.queryText, navigation: navigation)
} }
.onChange(of: state.queryText) { _ in .disableAutocorrection(true)
if state.query.query.compare(state.queryText, options: .caseInsensitive) == .orderedSame { .onChange(of: state.suggestionSelection) { newValue in
state.fieldIsFocused = true self.queryText = newValue
} }
.onChange(of: queryText) { newValue in
queryDebouncer.callback = {
DispatchQueue.main.async {
state.queryText = newValue
}
}
queryDebouncer.call()
} }
#if os(macOS) #if os(macOS)
.frame(maxWidth: 190) .frame(maxWidth: 190)
@ -74,16 +89,14 @@ struct SearchTextField: View {
.frame(width: 250, height: 32) .frame(width: 250, height: 32)
.overlay( .overlay(
RoundedRectangle(cornerRadius: 5, style: .continuous) RoundedRectangle(cornerRadius: 5, style: .continuous)
.stroke( .stroke(Color.gray.opacity(0.4), lineWidth: 1)
state.fieldIsFocused ? Color.blue.opacity(0.7) : Color.gray.opacity(0.4),
lineWidth: state.fieldIsFocused ? 3 : 1
)
.frame(width: 250, height: 31) .frame(width: 250, height: 31)
) )
} }
private var clearButton: some View { private var clearButton: some View {
Button(action: { Button(action: {
queryText = ""
self.state.queryText = "" self.state.queryText = ""
}) { }) {
Image(systemName: "xmark.circle.fill") Image(systemName: "xmark.circle.fill")

View File

@ -8,7 +8,7 @@ struct SearchSuggestions: View {
var body: some View { var body: some View {
List { List {
Button { Button {
runQueryAction() runQueryAction(state.queryText)
} label: { } label: {
HStack { HStack {
Image(systemName: "magnifyingglass") Image(systemName: "magnifyingglass")
@ -25,8 +25,7 @@ struct SearchSuggestions: View {
ForEach(visibleSuggestions, id: \.self) { suggestion in ForEach(visibleSuggestions, id: \.self) { suggestion in
HStack { HStack {
Button { Button {
state.queryText = suggestion runQueryAction(suggestion)
runQueryAction()
} label: { } label: {
HStack { HStack {
Image(systemName: "magnifyingglass") Image(systemName: "magnifyingglass")
@ -52,7 +51,7 @@ struct SearchSuggestions: View {
Spacer() Spacer()
Button { Button {
state.queryText = suggestion state.suggestionSelection = suggestion
} label: { } label: {
Image(systemName: "arrow.up.left.circle") Image(systemName: "arrow.up.left.circle")
.foregroundColor(.secondary) .foregroundColor(.secondary)
@ -72,14 +71,15 @@ struct SearchSuggestions: View {
#endif #endif
} }
private func runQueryAction() { private func runQueryAction(_ queryText: String) {
state.suggestionSelection = queryText
state.changeQuery { query in state.changeQuery { query in
query.query = state.queryText query.query = queryText
state.fieldIsFocused = false
navigation.hideKeyboard() navigation.hideKeyboard()
} }
recents.addQuery(state.queryText, navigation: navigation) recents.addQuery(queryText, navigation: navigation)
} }
private var visibleSuggestions: [String] { private var visibleSuggestions: [String] {

View File

@ -17,6 +17,7 @@ struct SearchView: View {
#endif #endif
@State private var favoriteItem: FavoriteItem? @State private var favoriteItem: FavoriteItem?
@State private var queryText = ""
@Environment(\.navigationStyle) private var navigationStyle @Environment(\.navigationStyle) private var navigationStyle
@ -60,9 +61,9 @@ struct SearchView: View {
}) { }) {
#if os(iOS) #if os(iOS)
VStack { VStack {
SearchTextField(favoriteItem: $favoriteItem) SearchTextField(queryText: $queryText, favoriteItem: $favoriteItem)
if state.query.query != state.queryText, !state.queryText.isEmpty, !state.querySuggestions.collection.isEmpty { if state.query.query != queryText, !queryText.isEmpty, !state.querySuggestions.collection.isEmpty {
SearchSuggestions() SearchSuggestions()
} else { } else {
results results
@ -72,8 +73,8 @@ struct SearchView: View {
ZStack { ZStack {
results results
#if !os(tvOS) #if os(macOS)
if state.query.query != state.queryText, !state.queryText.isEmpty, !state.querySuggestions.collection.isEmpty { if state.query.query != queryText, !queryText.isEmpty, !state.querySuggestions.collection.isEmpty {
HStack { HStack {
Spacer() Spacer()
SearchSuggestions() SearchSuggestions()
@ -107,14 +108,15 @@ struct SearchView: View {
filtersMenu filtersMenu
} }
SearchTextField() SearchTextField(queryText: $queryText)
} }
#endif #endif
} }
.onAppear { .onAppear {
if query != nil { if let query = query {
state.queryText = query!.query queryText = query.query
state.resetQuery(query!) state.queryText = query.query
state.resetQuery(query)
updateFavoriteItem() updateFavoriteItem()
} }
@ -122,19 +124,21 @@ struct SearchView: View {
state.store.replace(ContentItem.array(of: videos)) state.store.replace(ContentItem.array(of: videos))
} }
} }
.onChange(of: state.query.query) { newQuery in .onChange(of: state.queryText) { newQuery in
if queryText.isEmpty, queryText != newQuery {
queryText = newQuery
}
if newQuery.isEmpty { if newQuery.isEmpty {
favoriteItem = nil favoriteItem = nil
state.resetQuery()
} else { } else {
updateFavoriteItem() updateFavoriteItem()
} }
}
.onChange(of: state.queryText) { newQuery in
if newQuery.isEmpty {
state.resetQuery()
}
if state.query.query != queryText {
state.loadSuggestions(newQuery) state.loadSuggestions(newQuery)
}
#if os(tvOS) #if os(tvOS)
searchDebounce.invalidate() searchDebounce.invalidate()
@ -152,7 +156,6 @@ struct SearchView: View {
} }
#endif #endif
} }
.onChange(of: searchSortOrder) { order in .onChange(of: searchSortOrder) { order in
state.changeQuery { query in state.changeQuery { query in
query.sortBy = order query.sortBy = order
@ -308,7 +311,7 @@ struct SearchView: View {
Button { Button {
switch item.type { switch item.type {
case .query: case .query:
state.queryText = item.title queryText = item.title
state.changeQuery { query in query.query = item.title } state.changeQuery { query in query.query = item.title }
updateFavoriteItem() updateFavoriteItem()