jellyflood/Swiftfin tvOS/Views/ItemView/SeriesItemView/Components/SeriesEpisodeSelector.swift

173 lines
5.6 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 Introspect
import JellyfinAPI
import SwiftUI
struct SeriesEpisodeSelector: View {
@ObservedObject
var viewModel: SeriesItemViewModel
@EnvironmentObject
private var parentFocusGuide: FocusGuide
var body: some View {
VStack(spacing: 0) {
SeasonsHStack(viewModel: viewModel)
.environmentObject(parentFocusGuide)
EpisodesHStack(viewModel: viewModel)
.environmentObject(parentFocusGuide)
}
}
}
extension SeriesEpisodeSelector {
// MARK: SeasonsHStack
struct SeasonsHStack: View {
@ObservedObject
var viewModel: SeriesItemViewModel
@EnvironmentObject
private var focusGuide: FocusGuide
@FocusState
private var focusedSeason: BaseItemDto?
var body: some View {
ScrollView(.horizontal, showsIndicators: false) {
HStack {
ForEach(viewModel.menuSections.keys.sorted(by: { viewModel.menuSectionSort($0, $1) }), id: \.self) { season in
Button {
Text(season.displayTitle)
.fontWeight(.semibold)
.fixedSize()
.padding(.vertical, 10)
.padding(.horizontal, 20)
.if(viewModel.menuSelection == season) { text in
text
.background(Color.white)
.foregroundColor(.black)
}
}
.buttonStyle(.plain)
.id(season)
.focused($focusedSeason, equals: season)
}
}
.focusGuide(
focusGuide,
tag: "seasons",
onContentFocus: { focusedSeason = viewModel.menuSelection },
top: "top",
bottom: "episodes"
)
.frame(height: 70)
.padding(.horizontal, 50)
.padding(.top)
.padding(.bottom, 45)
}
.onChange(of: focusedSeason) { season in
guard let season = season else { return }
viewModel.select(section: season)
}
}
}
}
extension SeriesEpisodeSelector {
// MARK: EpisodesHStack
struct EpisodesHStack: View {
@ObservedObject
var viewModel: SeriesItemViewModel
@EnvironmentObject
private var focusGuide: FocusGuide
@FocusState
private var focusedEpisodeID: String?
@State
private var lastFocusedEpisodeID: String?
@State
private var wrappedScrollView: UIScrollView?
private var items: [BaseItemDto] {
guard let selection = viewModel.menuSelection,
let items = viewModel.menuSections[selection] else { return [.noResults] }
return items
}
var body: some View {
ScrollView(.horizontal, showsIndicators: false) {
HStack(alignment: .top, spacing: 40) {
if !items.isEmpty {
ForEach(items, id: \.self) { episode in
EpisodeCard(episode: episode)
.focused($focusedEpisodeID, equals: episode.id)
}
} else if viewModel.isLoading {
ForEach(1 ..< 10) { i in
EpisodeCard(episode: .placeHolder)
.redacted(reason: .placeholder)
.focused($focusedEpisodeID, equals: "\(i)")
}
} else {
EpisodeCard(episode: .noResults)
.focused($focusedEpisodeID, equals: "no-results")
}
}
.padding(.horizontal, 50)
.padding(.bottom, 50)
.padding(.top)
}
.mask {
VStack(spacing: 0) {
Color.white
LinearGradient(
stops: [
.init(color: .white, location: 0),
.init(color: .clear, location: 1),
],
startPoint: .top,
endPoint: .bottom
)
.frame(height: 20)
}
}
.transition(.opacity)
.focusGuide(
focusGuide,
tag: "episodes",
onContentFocus: { focusedEpisodeID = lastFocusedEpisodeID },
top: "seasons"
)
.introspectScrollView { scrollView in
wrappedScrollView = scrollView
}
.onChange(of: viewModel.menuSelection) { _ in
lastFocusedEpisodeID = items.first?.id
}
.onChange(of: focusedEpisodeID) { episodeIndex in
guard let episodeIndex = episodeIndex else { return }
lastFocusedEpisodeID = episodeIndex
}
.onChange(of: viewModel.menuSections) { _ in
lastFocusedEpisodeID = items.first?.id
}
}
}
}