269 lines
7.1 KiB
Swift
269 lines
7.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) 2025 Jellyfin & Jellyfin Contributors
|
|
//
|
|
|
|
import Algorithms
|
|
import Factory
|
|
import Foundation
|
|
import JellyfinAPI
|
|
import UIKit
|
|
|
|
// TODO: clean up
|
|
|
|
extension BaseItemDto: Displayable {
|
|
|
|
var displayTitle: String {
|
|
name ?? L10n.unknown
|
|
}
|
|
}
|
|
|
|
extension BaseItemDto: LibraryIdentifiable {
|
|
|
|
var unwrappedIDHashOrZero: Int {
|
|
id?.hashValue ?? 0
|
|
}
|
|
}
|
|
|
|
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 = Container.shared.currentUserSession()!
|
|
.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
|
|
}
|
|
|
|
var parentTitle: String? {
|
|
switch type {
|
|
case .audio:
|
|
album
|
|
case .episode:
|
|
seriesName
|
|
default:
|
|
nil
|
|
}
|
|
}
|
|
}
|