jellyflood/Shared/Extensions/JellyfinAPI/BaseItemDto/BaseItemDto.swift

258 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 Algorithms
import Factory
import Foundation
import JellyfinAPI
import UIKit
// TODO: clean up
extension BaseItemDto: Displayable {
var displayTitle: String {
name ?? .emptyDash
}
}
extension BaseItemDto: LibraryParent {
var libraryType: BaseItemKind? {
type
}
}
extension BaseItemDto {
var episodeLocator: String? {
guard let episodeNo = indexNumber else { return nil }
return L10n.episodeNumber(episodeNo)
}
var itemGenres: [ItemGenre]? {
guard let genres else { return nil }
return genres.map(ItemGenre.init)
}
var runTimeSeconds: Int {
let playbackPositionTicks = runTimeTicks ?? 0
return Int(playbackPositionTicks / 10_000_000)
}
var seasonEpisodeLabel: String? {
guard let seasonNo = parentIndexNumber, let episodeNo = indexNumber else { return nil }
return L10n.seasonAndEpisode(String(seasonNo), String(episodeNo))
}
var startTimeSeconds: Int {
let playbackPositionTicks = userData?.playbackPositionTicks ?? 0
return Int(playbackPositionTicks / 10_000_000)
}
// MARK: Calculations
var runTimeLabel: String? {
let timeHMSFormatter: DateComponentsFormatter = {
let formatter = DateComponentsFormatter()
formatter.unitsStyle = .abbreviated
formatter.allowedUnits = [.hour, .minute]
return formatter
}()
guard let runTimeTicks = runTimeTicks,
let text = timeHMSFormatter.string(from: Double(runTimeTicks / 10_000_000)) else { return nil }
return text
}
var progressLabel: String? {
guard let playbackPositionTicks = userData?.playbackPositionTicks,
let totalTicks = runTimeTicks,
playbackPositionTicks != 0,
totalTicks != 0 else { return nil }
let remainingSeconds = (totalTicks - playbackPositionTicks) / 10_000_000
let formatter = DateComponentsFormatter()
formatter.allowedUnits = [.hour, .minute]
formatter.unitsStyle = .abbreviated
return formatter.string(from: .init(remainingSeconds))
}
var programDuration: TimeInterval? {
guard let startDate, let endDate else { return nil }
return endDate.timeIntervalSince(startDate)
}
var programProgress: Double? {
guard let startDate, let endDate else { return nil }
let length = endDate.timeIntervalSince(startDate)
let progress = Date.now.timeIntervalSince(startDate)
return progress / length
}
func programProgress(relativeTo other: Date) -> Double? {
guard let startDate, let endDate else { return nil }
let length = endDate.timeIntervalSince(startDate)
let progress = other.timeIntervalSince(startDate)
return progress / length
}
var subtitleStreams: [MediaStream] {
mediaStreams?.filter { $0.type == .subtitle } ?? []
}
var audioStreams: [MediaStream] {
mediaStreams?.filter { $0.type == .audio } ?? []
}
var videoStreams: [MediaStream] {
mediaStreams?.filter { $0.type == .video } ?? []
}
// MARK: Missing and Unaired
var isMissing: Bool {
locationType == .virtual
}
var isUnaired: Bool {
if let premierDate = premiereDate {
return premierDate > Date()
} else {
return false
}
}
var airDateLabel: String? {
guard let premiereDateFormatted = premiereDateLabel else { return nil }
return L10n.airWithDate(premiereDateFormatted)
}
var premiereDateLabel: String? {
guard let premiereDate = premiereDate else { return nil }
let dateFormatter = DateFormatter()
dateFormatter.dateStyle = .medium
return dateFormatter.string(from: premiereDate)
}
var premiereDateYear: String? {
guard let premiereDate = premiereDate else { return nil }
let dateFormatter = DateFormatter()
dateFormatter.dateFormat = "YYYY"
return dateFormatter.string(from: premiereDate)
}
var hasExternalLinks: Bool {
guard let externalURLs else { return false }
return externalURLs.isNotEmpty
}
var hasRatings: Bool {
[
criticRating,
communityRating,
].contains { $0 != nil }
}
// MARK: Chapter Images
var fullChapterInfo: [ChapterInfo.FullInfo] {
guard let chapters else { return [] }
let ranges: [Range<Int>] = chapters
.map(\.startTimeSeconds)
.appending(runTimeSeconds + 1)
.adjacentPairs()
.map { $0 ..< $1 }
return zip(chapters, ranges)
.enumerated()
.map { i, zip in
let parameters = Paths.GetItemImageParameters(
maxWidth: 500,
quality: 90,
imageIndex: i
)
let request = Paths.getItemImage(
itemID: id ?? "",
imageType: ImageType.chapter.rawValue,
parameters: parameters
)
let imageURL = UserSession
.current()!
.client
.fullURL(with: request)
return .init(
chapterInfo: zip.0,
imageSource: .init(url: imageURL),
secondsRange: zip.1
)
}
}
// TODO: series-season-episode hierarchy for episodes
// TODO: user hierarchy for downloads
var downloadFolder: URL? {
guard let type, let id else { return nil }
let root = URL.downloads
// .appendingPathComponent(userSession.user.id)
switch type {
case .movie, .episode:
return root
.appendingPathComponent(id)
// case .episode:
// guard let seasonID = seasonID,
// let seriesID = seriesID
// else {
// return nil
// }
// return root
// .appendingPathComponent(seriesID)
// .appendingPathComponent(seasonID)
// .appendingPathComponent(id)
default:
return nil
}
}
/// Returns `originalTitle` if it is not the same as `displayTitle`
var alternateTitle: String? {
originalTitle != displayTitle ? originalTitle : nil
}
var playButtonLabel: String {
if isUnaired {
return L10n.unaired
}
if isMissing {
return L10n.missing
}
if let progressLabel {
return progressLabel
}
return L10n.play
}
}