* Add pin prompt to sign-in screen * Bring over security views from iOS * silence tvOS 17 warnings * Add user profile and security views to routing * Changes * revert and remove commented code * cleanup * CodeFactor fixes * Joe's Suggestions: - Move UserProfileSettings to their own Coordinator - Make Views Modal to better reflect existing items - Fix CustomizeSettingsCoordinator (This is on me!) - Change PINs to use SecureField - Move all Settings View to use SplitFormWindowView to mirror existing Settings - Use user profile image for SplitFormWindowView Icon - Change Profile Security to use LearnMoreModal - Use suggestion from https://forums.developer.apple.com/forums/thread/739545 - Tag Alert > TextFields with TODO so we can check this on tvOS 18 * Fix PIN for https://forums.developer.apple.com/forums/thread/739545 on SelectUserView * Fix Build Issue. * use user --------- Co-authored-by: chickdan <=> Co-authored-by: Joe <jpkribs@outlook.com> Co-authored-by: Ethan Pippin <ethanpippin2343@gmail.com>
134 lines
4.2 KiB
Swift
134 lines
4.2 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) 2025 Jellyfin & Jellyfin Contributors
|
|
//
|
|
|
|
import CollectionHStack
|
|
import JellyfinAPI
|
|
import SwiftUI
|
|
|
|
struct SeriesEpisodeSelector: View {
|
|
|
|
@ObservedObject
|
|
var viewModel: SeriesItemViewModel
|
|
|
|
@EnvironmentObject
|
|
private var parentFocusGuide: FocusGuide
|
|
|
|
@State
|
|
private var didSelectPlayButtonSeason = false
|
|
@State
|
|
private var selection: SeasonItemViewModel.ID?
|
|
|
|
private var selectionViewModel: SeasonItemViewModel? {
|
|
viewModel.seasons.first(where: { $0.id == selection })
|
|
}
|
|
|
|
var body: some View {
|
|
VStack(spacing: 0) {
|
|
SeasonsHStack(viewModel: viewModel, selection: $selection)
|
|
.environmentObject(parentFocusGuide)
|
|
|
|
if let selectionViewModel {
|
|
EpisodeHStack(viewModel: selectionViewModel, playButtonItem: viewModel.playButtonItem)
|
|
.environmentObject(parentFocusGuide)
|
|
} else {
|
|
LoadingHStack()
|
|
}
|
|
}
|
|
.onReceive(viewModel.playButtonItem.publisher) { newValue in
|
|
|
|
guard !didSelectPlayButtonSeason else { return }
|
|
didSelectPlayButtonSeason = true
|
|
|
|
if let playButtonSeason = viewModel.seasons.first(where: { $0.id == newValue.seasonID }) {
|
|
selection = playButtonSeason.id
|
|
} else {
|
|
selection = viewModel.seasons.first?.id
|
|
}
|
|
}
|
|
.onChange(of: selection) { _, _ in
|
|
guard let selectionViewModel else { return }
|
|
|
|
if selectionViewModel.state == .initial {
|
|
selectionViewModel.send(.refresh)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extension SeriesEpisodeSelector {
|
|
|
|
// MARK: SeasonsHStack
|
|
|
|
struct SeasonsHStack: View {
|
|
|
|
@EnvironmentObject
|
|
private var focusGuide: FocusGuide
|
|
|
|
@FocusState
|
|
private var focusedSeason: SeasonItemViewModel.ID?
|
|
|
|
@ObservedObject
|
|
var viewModel: SeriesItemViewModel
|
|
|
|
var selection: Binding<SeasonItemViewModel.ID?>
|
|
|
|
var body: some View {
|
|
ScrollView(.horizontal, showsIndicators: false) {
|
|
HStack {
|
|
ForEach(viewModel.seasons) { seasonViewModel in
|
|
Button {
|
|
Text(seasonViewModel.season.displayTitle)
|
|
.font(.headline)
|
|
.fontWeight(.semibold)
|
|
.padding(.vertical, 10)
|
|
.padding(.horizontal, 20)
|
|
.if(selection.wrappedValue == seasonViewModel.id) { text in
|
|
text
|
|
.background(Color.white)
|
|
.foregroundColor(.black)
|
|
}
|
|
}
|
|
.buttonStyle(.card)
|
|
.focused($focusedSeason, equals: seasonViewModel.id)
|
|
}
|
|
}
|
|
.focusGuide(
|
|
focusGuide,
|
|
tag: "seasons",
|
|
onContentFocus: { focusedSeason = selection.wrappedValue },
|
|
top: "top",
|
|
bottom: "episodes"
|
|
)
|
|
.frame(height: 70)
|
|
.padding(.horizontal, 50)
|
|
.padding(.top)
|
|
.padding(.bottom, 45)
|
|
}
|
|
.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)
|
|
}
|
|
}
|
|
.onChange(of: focusedSeason) { _, newValue in
|
|
guard let newValue else { return }
|
|
selection.wrappedValue = newValue
|
|
}
|
|
}
|
|
}
|
|
}
|