116 lines
5.0 KiB
Swift
116 lines
5.0 KiB
Swift
/* JellyfinPlayer/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 2021 Aiden Vigue & Jellyfin Contributors
|
|
*/
|
|
|
|
import SwiftUI
|
|
import Introspect
|
|
import JellyfinAPI
|
|
|
|
class VideoPlayerItem: ObservableObject {
|
|
@Published var shouldShowPlayer: Bool = false
|
|
@Published var itemToPlay: BaseItemDto = BaseItemDto()
|
|
}
|
|
|
|
struct ItemView: View {
|
|
private var item: BaseItemDto
|
|
|
|
@StateObject private var videoPlayerItem: VideoPlayerItem = VideoPlayerItem()
|
|
@State private var videoIsLoading: Bool = false; // This variable is only changed by the underlying VLC view.
|
|
@State private var isLoading: Bool = false
|
|
@State private var viewDidLoad: Bool = false
|
|
|
|
init(item: BaseItemDto) {
|
|
self.item = item
|
|
}
|
|
|
|
var portraitHeaderView: some View {
|
|
ImageView(src: item.getBackdropImage(maxWidth: UIDevice.current.userInterfaceIdiom == .pad ? 622 : Int(UIScreen.main.bounds.width)),
|
|
bh: item.getBackdropImageBlurHash())
|
|
.opacity(0.4)
|
|
.blur(radius: 2.0)
|
|
}
|
|
|
|
var portraitHeaderOverlayView: some View {
|
|
PortraitHeaderOverlayView(item: item)
|
|
}
|
|
|
|
var body: some View {
|
|
VStack {
|
|
NavigationLink(destination: LoadingViewNoBlur(isShowing: $videoIsLoading) {
|
|
VLCPlayerWithControls(item: videoPlayerItem.itemToPlay,
|
|
loadBinding: $videoIsLoading,
|
|
pBinding: _videoPlayerItem.projectedValue.shouldShowPlayer)
|
|
.navigationBarHidden(true)
|
|
.navigationBarBackButtonHidden(true)
|
|
.statusBar(hidden: true)
|
|
.edgesIgnoringSafeArea(.all)
|
|
.prefersHomeIndicatorAutoHidden(true)
|
|
}, isActive: $videoPlayerItem.shouldShowPlayer) {
|
|
EmptyView()
|
|
}
|
|
|
|
ParallaxHeaderScrollView(header: portraitHeaderView,
|
|
staticOverlayView: portraitHeaderOverlayView,
|
|
overlayAlignment: .bottomLeading,
|
|
headerHeight: UIDevice.current.userInterfaceIdiom == .pad ? 350 : UIScreen.main.bounds.width * 0.5625) {
|
|
VStack {
|
|
Spacer()
|
|
.frame(height: UIDevice.current.userInterfaceIdiom == .pad ? 135 : 40)
|
|
.padding(.bottom, UIDevice.current.userInterfaceIdiom == .pad ? 54 : 24)
|
|
|
|
// MARK: Overview
|
|
Text(item.overview ?? "")
|
|
.font(.footnote)
|
|
.padding(.top, 3)
|
|
.fixedSize(horizontal: false, vertical: true)
|
|
.padding(.bottom, 3)
|
|
.padding(.leading, 16)
|
|
.padding(.trailing, UIDevice.current.userInterfaceIdiom == .pad ? 16 : 55)
|
|
|
|
// MARK: Genres
|
|
PillHStackView(title: "Genres", items: item.genreItems ?? []) { genre in
|
|
LibraryView(viewModel: .init(genre: genre), title: genre.title)
|
|
}
|
|
|
|
// MARK: Studios
|
|
if !(item.studios ?? []).isEmpty {
|
|
ScrollView(.horizontal, showsIndicators: false) {
|
|
HStack {
|
|
Text("Studios:").font(.callout).fontWeight(.semibold)
|
|
ForEach(item.studios!, id: \.id) { studio in
|
|
NavigationLink(destination: LazyView {
|
|
LibraryView(viewModel: .init(studio: studio), title: studio.name ?? "")
|
|
}) {
|
|
Text(studio.name ?? "").font(.footnote)
|
|
}
|
|
}
|
|
}
|
|
.padding(.leading, 16)
|
|
.padding(.trailing, UIDevice.current.userInterfaceIdiom == .pad ? 16 : 55)
|
|
}
|
|
}
|
|
|
|
// MARK: Cast
|
|
PortraitImageHStackView(title: "Cast",
|
|
items: item.people!,
|
|
maxWidth: 150) { person in
|
|
LibraryView(viewModel: .init(person: person), title: person.title)
|
|
}
|
|
|
|
// MARK: More Like This
|
|
|
|
}
|
|
.introspectTabBarController { (UITabBarController) in
|
|
UITabBarController.tabBar.isHidden = false
|
|
}
|
|
.navigationBarBackButtonHidden(false)
|
|
.environmentObject(videoPlayerItem)
|
|
}
|
|
.navigationBarHidden(false)
|
|
}
|
|
}
|
|
}
|