134 lines
4.1 KiB
Swift
134 lines
4.1 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 CollectionView
|
|
import Defaults
|
|
import JellyfinAPI
|
|
import Stinsen
|
|
import SwiftUI
|
|
|
|
struct MediaView: View {
|
|
|
|
@EnvironmentObject
|
|
private var router: MediaCoordinator.Router
|
|
|
|
@ObservedObject
|
|
var viewModel: MediaViewModel
|
|
|
|
private var gridLayout: NSCollectionLayoutSection.GridLayoutMode {
|
|
if UIDevice.isPhone {
|
|
return .fixedNumberOfColumns(2)
|
|
} else {
|
|
return .adaptive(withMinItemSize: PosterType.landscape.width)
|
|
}
|
|
}
|
|
|
|
var body: some View {
|
|
CollectionView(items: viewModel.libraryItems) { _, viewModel, _ in
|
|
LibraryCard(viewModel: viewModel)
|
|
.onSelect {
|
|
switch viewModel.item.collectionType {
|
|
case "downloads":
|
|
router.route(to: \.downloads)
|
|
case "favorites":
|
|
router.route(to: \.library, .init(parent: viewModel.item, type: .library, filters: .favorites))
|
|
case "folders":
|
|
router.route(to: \.library, .init(parent: viewModel.item, type: .folders, filters: .init()))
|
|
case "liveTV":
|
|
router.route(to: \.liveTV)
|
|
default:
|
|
router.route(to: \.library, .init(parent: viewModel.item, type: .library, filters: .init()))
|
|
}
|
|
}
|
|
}
|
|
.layout { _, layoutEnvironment in
|
|
.grid(
|
|
layoutEnvironment: layoutEnvironment,
|
|
layoutMode: gridLayout,
|
|
sectionInsets: .init(top: 0, leading: 10, bottom: 0, trailing: 10)
|
|
)
|
|
}
|
|
.configure { configuration in
|
|
configuration.showsVerticalScrollIndicator = false
|
|
}
|
|
.ignoresSafeArea()
|
|
.navigationTitle(L10n.allMedia)
|
|
.toolbar {
|
|
ToolbarItemGroup(placement: .navigationBarTrailing) {
|
|
if viewModel.isLoading {
|
|
ProgressView()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extension MediaView {
|
|
|
|
struct LibraryCard: View {
|
|
|
|
@ObservedObject
|
|
var viewModel: MediaItemViewModel
|
|
|
|
private var onSelect: () -> Void
|
|
|
|
private var itemWidth: CGFloat {
|
|
PosterType.landscape.width * (UIDevice.isPhone ? 0.85 : 1)
|
|
}
|
|
|
|
var body: some View {
|
|
Button {
|
|
onSelect()
|
|
} label: {
|
|
Group {
|
|
if let imageSources = viewModel.imageSources {
|
|
ImageView(imageSources)
|
|
} else {
|
|
ImageView(nil)
|
|
}
|
|
}
|
|
.overlay {
|
|
if Defaults[.Customization.Library.randomImage] ||
|
|
viewModel.item.collectionType == "favorites" ||
|
|
viewModel.item.collectionType == "downloads"
|
|
{
|
|
ZStack {
|
|
Color.black
|
|
.opacity(0.5)
|
|
|
|
Text(viewModel.item.displayTitle)
|
|
.foregroundColor(.white)
|
|
.font(.title2)
|
|
.fontWeight(.semibold)
|
|
.lineLimit(2)
|
|
.multilineTextAlignment(.center)
|
|
.frame(alignment: .center)
|
|
}
|
|
}
|
|
}
|
|
.posterStyle(.landscape)
|
|
.frame(width: itemWidth)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extension MediaView.LibraryCard {
|
|
|
|
init(viewModel: MediaItemViewModel) {
|
|
self.init(
|
|
viewModel: viewModel,
|
|
onSelect: {}
|
|
)
|
|
}
|
|
|
|
func onSelect(_ action: @escaping () -> Void) -> Self {
|
|
copy(modifying: \.onSelect, with: action)
|
|
}
|
|
}
|