mirror of
https://github.com/jellyfin/Swiftfin.git
synced 2024-12-04 04:01:47 +00:00
176 lines
7.4 KiB
Swift
176 lines
7.4 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 2021 Aiden Vigue & Jellyfin Contributors
|
|
*/
|
|
|
|
import SwiftUI
|
|
import JellyfinAPI
|
|
|
|
struct EpisodeItemView: View {
|
|
@ObservedObject var viewModel: EpisodeItemViewModel
|
|
|
|
@State var actors: [BaseItemPerson] = []
|
|
@State var studio: String?
|
|
@State var director: String?
|
|
|
|
func onAppear() {
|
|
actors = []
|
|
director = nil
|
|
studio = nil
|
|
var actor_index = 0
|
|
viewModel.item.people?.forEach { person in
|
|
if person.type == "Actor" {
|
|
if actor_index < 4 {
|
|
actors.append(person)
|
|
}
|
|
actor_index = actor_index + 1
|
|
}
|
|
if person.type == "Director" {
|
|
director = person.name ?? ""
|
|
}
|
|
}
|
|
|
|
studio = viewModel.item.studios?.first?.name ?? nil
|
|
}
|
|
|
|
var body: some View {
|
|
ZStack {
|
|
ImageView(src: viewModel.item.getBackdropImage(maxWidth: 1920), bh: viewModel.item.getBackdropImageBlurHash())
|
|
.opacity(0.4)
|
|
LazyVStack(alignment: .leading) {
|
|
Text(viewModel.item.name ?? "")
|
|
.font(.title)
|
|
.fontWeight(.bold)
|
|
.foregroundColor(.primary)
|
|
Text(viewModel.item.seriesName ?? "")
|
|
.fontWeight(.bold)
|
|
.foregroundColor(.primary)
|
|
HStack {
|
|
if viewModel.item.productionYear != nil {
|
|
Text(String(viewModel.item.productionYear!)).font(.subheadline)
|
|
.fontWeight(.medium)
|
|
.foregroundColor(.secondary)
|
|
.lineLimit(1)
|
|
}
|
|
Text(viewModel.item.getItemRuntime()).font(.subheadline)
|
|
.fontWeight(.medium)
|
|
.foregroundColor(.secondary)
|
|
.lineLimit(1)
|
|
if viewModel.item.officialRating != nil {
|
|
Text(viewModel.item.officialRating!).font(.subheadline)
|
|
.fontWeight(.semibold)
|
|
.foregroundColor(.secondary)
|
|
.lineLimit(1)
|
|
.padding(EdgeInsets(top: 1, leading: 4, bottom: 1, trailing: 4))
|
|
.overlay(RoundedRectangle(cornerRadius: 2)
|
|
.stroke(Color.secondary, lineWidth: 1))
|
|
}
|
|
Spacer()
|
|
}.padding(.top, -15)
|
|
|
|
HStack(alignment: .top) {
|
|
VStack(alignment: .trailing) {
|
|
if studio != nil {
|
|
Text("STUDIO")
|
|
.font(.body)
|
|
.fontWeight(.semibold)
|
|
.foregroundColor(.primary)
|
|
Text(studio!)
|
|
.font(.body)
|
|
.fontWeight(.semibold)
|
|
.foregroundColor(.secondary)
|
|
.padding(.bottom, 40)
|
|
}
|
|
|
|
if director != nil {
|
|
Text("DIRECTOR")
|
|
.font(.body)
|
|
.fontWeight(.semibold)
|
|
.foregroundColor(.primary)
|
|
Text(director!)
|
|
.font(.body)
|
|
.fontWeight(.semibold)
|
|
.foregroundColor(.secondary)
|
|
.padding(.bottom, 40)
|
|
}
|
|
|
|
if !actors.isEmpty {
|
|
Text("CAST")
|
|
.font(.body)
|
|
.fontWeight(.semibold)
|
|
.foregroundColor(.primary)
|
|
ForEach(actors, id: \.id) { person in
|
|
Text(person.name!)
|
|
.font(.body)
|
|
.fontWeight(.semibold)
|
|
.foregroundColor(.secondary)
|
|
}
|
|
}
|
|
Spacer()
|
|
}
|
|
VStack(alignment: .leading) {
|
|
Text(viewModel.item.overview ?? "")
|
|
.font(.body)
|
|
.fontWeight(.medium)
|
|
.foregroundColor(.primary)
|
|
|
|
HStack {
|
|
VStack {
|
|
Button {
|
|
viewModel.updateFavoriteState()
|
|
} label: {
|
|
MediaViewActionButton(icon: "heart.fill", iconColor: viewModel.isFavorited ? .red : .white)
|
|
}
|
|
Text(viewModel.isFavorited ? "Unfavorite" : "Favorite")
|
|
.font(.caption)
|
|
}
|
|
VStack {
|
|
NavigationLink(destination: VideoPlayerView(item: viewModel.item)) {
|
|
MediaViewActionButton(icon: "play.fill")
|
|
}
|
|
Text(viewModel.item.getItemProgressString() != "" ? "\(viewModel.item.getItemProgressString()) left" : "Play")
|
|
.font(.caption)
|
|
}
|
|
VStack {
|
|
Button {
|
|
viewModel.updateWatchState()
|
|
} label: {
|
|
MediaViewActionButton(icon: "eye.fill", iconColor: viewModel.isWatched ? .red : .white)
|
|
}
|
|
Text(viewModel.isWatched ? "Unwatch" : "Mark Watched")
|
|
.font(.caption)
|
|
}
|
|
Spacer()
|
|
}
|
|
.padding(.top, 15)
|
|
}
|
|
}.padding(.top, 50)
|
|
|
|
if !viewModel.similarItems.isEmpty {
|
|
Text("More Like This")
|
|
.font(.headline)
|
|
.fontWeight(.semibold)
|
|
ScrollView(.horizontal) {
|
|
LazyHStack {
|
|
Spacer().frame(width: 45)
|
|
ForEach(viewModel.similarItems, id: \.id) { similarItems in
|
|
NavigationLink(destination: ItemView(item: similarItems)) {
|
|
PortraitItemElement(item: similarItems)
|
|
}.buttonStyle(PlainNavigationLinkButtonStyle())
|
|
}
|
|
Spacer().frame(width: 45)
|
|
}
|
|
}.padding(EdgeInsets(top: -30, leading: -90, bottom: 0, trailing: -90))
|
|
.frame(height: 360)
|
|
}
|
|
Spacer()
|
|
Spacer()
|
|
}.padding(EdgeInsets(top: 90, leading: 90, bottom: 0, trailing: 90))
|
|
}.onAppear(perform: onAppear)
|
|
}
|
|
}
|