140 lines
4.3 KiB
Swift
140 lines
4.3 KiB
Swift
//
|
|
// Swiftfin is subject to the terms of the Mozilla Public
|
|
// License, v2.0. If a copy of the MPL was not distributed with this
|
|
// file, you can obtain one at https://mozilla.org/MPL/2.0/.
|
|
//
|
|
// Copyright (c) 2024 Jellyfin & Jellyfin Contributors
|
|
//
|
|
|
|
import Defaults
|
|
import JellyfinAPI
|
|
import SwiftUI
|
|
|
|
struct SearchView: View {
|
|
|
|
@Default(.Customization.searchPosterType)
|
|
private var searchPosterType
|
|
|
|
@EnvironmentObject
|
|
private var videoPlayerRouter: VideoPlayerWrapperCoordinator.Router
|
|
@EnvironmentObject
|
|
private var router: SearchCoordinator.Router
|
|
|
|
@StateObject
|
|
private var viewModel = SearchViewModel()
|
|
|
|
@State
|
|
private var searchQuery = ""
|
|
|
|
private var suggestionsView: some View {
|
|
VStack(spacing: 20) {
|
|
ForEach(viewModel.suggestions) { item in
|
|
Button(item.displayTitle) {
|
|
searchQuery = item.displayTitle
|
|
}
|
|
.buttonStyle(.plain)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
}
|
|
}
|
|
|
|
private var resultsView: some View {
|
|
ScrollView(showsIndicators: false) {
|
|
VStack(spacing: 20) {
|
|
if viewModel.movies.isNotEmpty {
|
|
itemsSection(title: L10n.movies, keyPath: \.movies, posterType: searchPosterType)
|
|
}
|
|
|
|
if viewModel.series.isNotEmpty {
|
|
itemsSection(title: L10n.tvShows, keyPath: \.series, posterType: searchPosterType)
|
|
}
|
|
|
|
if viewModel.collections.isNotEmpty {
|
|
itemsSection(title: L10n.collections, keyPath: \.collections, posterType: searchPosterType)
|
|
}
|
|
|
|
if viewModel.episodes.isNotEmpty {
|
|
itemsSection(title: L10n.episodes, keyPath: \.episodes, posterType: searchPosterType)
|
|
}
|
|
|
|
if viewModel.programs.isNotEmpty {
|
|
itemsSection(title: L10n.programs, keyPath: \.programs, posterType: .landscape)
|
|
}
|
|
|
|
if viewModel.channels.isNotEmpty {
|
|
itemsSection(title: L10n.channels, keyPath: \.channels, posterType: .portrait)
|
|
}
|
|
|
|
if viewModel.people.isNotEmpty {
|
|
itemsSection(title: L10n.people, keyPath: \.people, posterType: .portrait)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private func select(_ item: BaseItemDto) {
|
|
switch item.type {
|
|
case .person:
|
|
let viewModel = ItemLibraryViewModel(parent: item)
|
|
router.route(to: \.library, viewModel)
|
|
case .program:
|
|
videoPlayerRouter.route(
|
|
to: \.liveVideoPlayer,
|
|
LiveVideoPlayerManager(program: item)
|
|
)
|
|
case .tvChannel:
|
|
guard let mediaSource = item.mediaSources?.first else { return }
|
|
videoPlayerRouter.route(
|
|
to: \.liveVideoPlayer,
|
|
LiveVideoPlayerManager(item: item, mediaSource: mediaSource)
|
|
)
|
|
default:
|
|
router.route(to: \.item, item)
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
private func itemsSection(
|
|
title: String,
|
|
keyPath: KeyPath<SearchViewModel, [BaseItemDto]>,
|
|
posterType: PosterDisplayType
|
|
) -> some View {
|
|
PosterHStack(
|
|
title: title,
|
|
type: posterType,
|
|
items: viewModel[keyPath: keyPath]
|
|
)
|
|
.onSelect(select)
|
|
}
|
|
|
|
var body: some View {
|
|
WrappedView {
|
|
Group {
|
|
switch viewModel.state {
|
|
case let .error(error):
|
|
Text(error.localizedDescription)
|
|
case .initial:
|
|
suggestionsView
|
|
case .content:
|
|
if viewModel.hasNoResults {
|
|
L10n.noResults.text
|
|
} else {
|
|
resultsView
|
|
}
|
|
case .searching:
|
|
ProgressView()
|
|
}
|
|
}
|
|
.transition(.opacity.animation(.linear(duration: 0.2)))
|
|
}
|
|
.ignoresSafeArea(edges: [.bottom, .horizontal])
|
|
.onFirstAppear {
|
|
viewModel.send(.getSuggestions)
|
|
}
|
|
.onChange(of: searchQuery) { newValue in
|
|
viewModel.send(.search(query: newValue))
|
|
}
|
|
.searchable(text: $searchQuery, prompt: L10n.search)
|
|
}
|
|
}
|