229 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			Swift
		
	
	
	
			
		
		
	
	
			229 lines
		
	
	
		
			6.9 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 CollectionVGrid
 | |
| import Defaults
 | |
| import JellyfinAPI
 | |
| import SwiftUI
 | |
| 
 | |
| // TODO: Figure out proper tab bar handling with the collection offset
 | |
| // TODO: list columns
 | |
| // TODO: list row view (LibraryRow)
 | |
| // TODO: fix paging for next item focusing the tab
 | |
| 
 | |
| struct PagingLibraryView<Element: Poster>: View {
 | |
| 
 | |
|     @Default(.Customization.Library.cinematicBackground)
 | |
|     private var cinematicBackground
 | |
|     @Default(.Customization.Library.posterType)
 | |
|     private var posterType
 | |
|     @Default(.Customization.Library.displayType)
 | |
|     private var viewType
 | |
|     @Default(.Customization.showPosterLabels)
 | |
|     private var showPosterLabels
 | |
| 
 | |
|     @EnvironmentObject
 | |
|     private var router: LibraryCoordinator<Element>.Router
 | |
| 
 | |
|     @State
 | |
|     private var focusedItem: Element?
 | |
| 
 | |
|     @State
 | |
|     private var presentBackground = false
 | |
|     @State
 | |
|     private var layout: CollectionVGridLayout
 | |
| 
 | |
|     @StateObject
 | |
|     private var viewModel: PagingLibraryViewModel<Element>
 | |
| 
 | |
|     @StateObject
 | |
|     private var cinematicBackgroundViewModel: CinematicBackgroundView<Element>.ViewModel = .init()
 | |
| 
 | |
|     init(viewModel: PagingLibraryViewModel<Element>) {
 | |
|         self._viewModel = StateObject(wrappedValue: viewModel)
 | |
| 
 | |
|         let initialPosterType = Defaults[.Customization.Library.posterType]
 | |
|         let initialViewType = Defaults[.Customization.Library.displayType]
 | |
|         let listColumnCount = Defaults[.Customization.Library.listColumnCount]
 | |
| 
 | |
|         self._layout = State(
 | |
|             initialValue: Self.makeLayout(
 | |
|                 posterType: initialPosterType,
 | |
|                 displayType: initialViewType,
 | |
|                 listColumnCount: listColumnCount
 | |
|             )
 | |
|         )
 | |
|     }
 | |
| 
 | |
|     // MARK: onSelect
 | |
| 
 | |
|     private func onSelect(_ element: Element) {
 | |
|         switch element {
 | |
|         case let element as BaseItemDto:
 | |
|             select(item: element)
 | |
|         case let element as BaseItemPerson:
 | |
|             select(person: element)
 | |
|         default:
 | |
|             assertionFailure("Used an unexpected type within a `PagingLibaryView`?")
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func select(item: BaseItemDto) {
 | |
|         switch item.type {
 | |
|         case .collectionFolder, .folder:
 | |
|             let viewModel = ItemLibraryViewModel(parent: item)
 | |
|             router.route(to: \.library, viewModel)
 | |
|         default:
 | |
|             router.route(to: \.item, item)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func select(person: BaseItemPerson) {
 | |
|         let viewModel = ItemLibraryViewModel(parent: person)
 | |
|         router.route(to: \.library, viewModel)
 | |
|     }
 | |
| 
 | |
|     // MARK: layout
 | |
| 
 | |
|     private static func makeLayout(
 | |
|         posterType: PosterDisplayType,
 | |
|         displayType: LibraryDisplayType,
 | |
|         listColumnCount: Int
 | |
|     ) -> CollectionVGridLayout {
 | |
|         switch (posterType, displayType) {
 | |
|         case (.landscape, .grid):
 | |
|             return .columns(5, insets: .init(50), itemSpacing: 50, lineSpacing: 50)
 | |
|         case (.portrait, .grid):
 | |
|             return .columns(7, insets: .init(50), itemSpacing: 50, lineSpacing: 50)
 | |
|         case (_, .list):
 | |
|             return .columns(listColumnCount, insets: .init(50), itemSpacing: 50, lineSpacing: 50)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func landscapeGridItemView(item: Element) -> some View {
 | |
|         PosterButton(item: item, type: .landscape)
 | |
|             .content {
 | |
|                 if item.showTitle {
 | |
|                     PosterButton.TitleContentView(item: item)
 | |
|                         .backport
 | |
|                         .lineLimit(1, reservesSpace: true)
 | |
|                 }
 | |
|             }
 | |
|             .onFocusChanged { newValue in
 | |
|                 if newValue {
 | |
|                     focusedItem = item
 | |
|                 }
 | |
|             }
 | |
|             .onSelect {
 | |
|                 onSelect(item)
 | |
|             }
 | |
|     }
 | |
| 
 | |
|     private func portraitGridItemView(item: Element) -> some View {
 | |
|         PosterButton(item: item, type: .portrait)
 | |
|             .content {
 | |
|                 if item.showTitle {
 | |
|                     PosterButton.TitleContentView(item: item)
 | |
|                         .backport
 | |
|                         .lineLimit(1, reservesSpace: true)
 | |
|                 }
 | |
|             }
 | |
|             .onFocusChanged { newValue in
 | |
|                 if newValue {
 | |
|                     focusedItem = item
 | |
|                 }
 | |
|             }
 | |
|             .onSelect {
 | |
|                 onSelect(item)
 | |
|             }
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private func listItemView(item: Element, posterType: PosterDisplayType) -> some View {
 | |
|         LibraryRow(item: item, posterType: posterType)
 | |
|             .onFocusChanged { newValue in
 | |
|                 if newValue {
 | |
|                     focusedItem = item
 | |
|                 }
 | |
|             }
 | |
|             .onSelect {
 | |
|                 onSelect(item)
 | |
|             }
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var contentView: some View {
 | |
|         CollectionVGrid(
 | |
|             $viewModel.elements,
 | |
|             layout: layout
 | |
|         ) { item in
 | |
|             switch (posterType, viewType) {
 | |
|             case (.landscape, .grid):
 | |
|                 landscapeGridItemView(item: item)
 | |
|             case (.portrait, .grid):
 | |
|                 portraitGridItemView(item: item)
 | |
|             case (_, .list):
 | |
|                 listItemView(item: item, posterType: posterType)
 | |
|             }
 | |
|         }
 | |
|         .onReachedBottomEdge(offset: .rows(3)) {
 | |
|             viewModel.send(.getNextPage)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     var body: some View {
 | |
|         ZStack {
 | |
|             if cinematicBackground {
 | |
|                 CinematicBackgroundView(viewModel: cinematicBackgroundViewModel)
 | |
|                     .visible(presentBackground)
 | |
|                     .blurred()
 | |
|             }
 | |
| 
 | |
|             WrappedView {
 | |
|                 Group {
 | |
|                     switch viewModel.state {
 | |
|                     case let .error(error):
 | |
|                         Text(error.localizedDescription)
 | |
|                     case .initial, .refreshing:
 | |
|                         ProgressView()
 | |
|                     case .content:
 | |
|                         if viewModel.elements.isEmpty {
 | |
|                             L10n.noResults.text
 | |
|                         } else {
 | |
|                             contentView
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         .ignoresSafeArea()
 | |
|         .navigationTitle(viewModel.parent?.displayTitle ?? "")
 | |
|         .onFirstAppear {
 | |
|             if viewModel.state == .initial {
 | |
|                 viewModel.send(.refresh)
 | |
|             }
 | |
|         }
 | |
|         .onChange(of: focusedItem) { _, newValue in
 | |
|             guard let newValue else {
 | |
|                 withAnimation {
 | |
|                     presentBackground = false
 | |
|                 }
 | |
|                 return
 | |
|             }
 | |
| 
 | |
|             cinematicBackgroundViewModel.select(item: newValue)
 | |
| 
 | |
|             if !presentBackground {
 | |
|                 withAnimation {
 | |
|                     presentBackground = true
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |