174 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
			
		
		
	
	
			174 lines
		
	
	
		
			5.7 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) 2023 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.compactMap(\._item)
 | |
|         }
 | |
| 
 | |
|         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
 | |
|                 wrappedScrollView?.scrollToTop(animated: false)
 | |
|             }
 | |
|             .onChange(of: focusedEpisodeID) { episodeIndex in
 | |
|                 guard let episodeIndex = episodeIndex else { return }
 | |
|                 lastFocusedEpisodeID = episodeIndex
 | |
|             }
 | |
|             .onChange(of: viewModel.menuSections) { _ in
 | |
|                 lastFocusedEpisodeID = items.first?.id
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |