jellyflood/Shared/Extensions/JellyfinAPI/BaseItemDto.swift

252 lines
7.2 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 Algorithms
import Factory
import Foundation
import JellyfinAPI
import UIKit
extension BaseItemDto: Displayable {
var displayTitle: String {
name ?? .emptyDash
}
}
extension BaseItemDto: LibraryParent {}
extension BaseItemDto {
var episodeLocator: String? {
guard let episodeNo = indexNumber else { return nil }
return L10n.episodeNumber(episodeNo)
}
var runTimeSeconds: Int {
let playbackPositionTicks = runTimeTicks ?? 0
return Int(playbackPositionTicks / 10_000_000)
}
var seasonEpisodeLocator: String? {
if let seasonNo = parentIndexNumber, let episodeNo = indexNumber {
return L10n.seasonAndEpisode(String(seasonNo), String(episodeNo))
}
return nil
}
var startTimeSeconds: Int {
let playbackPositionTicks = userData?.playbackPositionTicks ?? 0
return Int(playbackPositionTicks / 10_000_000)
}
// MARK: Calculations
// TODO: make computed var or function that takes allowed units
func getItemRuntime() -> 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))
}
func getLiveStartTimeString(formatter: DateFormatter) -> String {
if let startDate = self.startDate {
return formatter.string(from: startDate)
}
return " "
}
func getLiveEndTimeString(formatter: DateFormatter) -> String {
if let endDate = self.endDate {
return formatter.string(from: endDate)
}
return " "
}
func getLiveProgressPercentage() -> Double {
if let startDate = self.startDate,
let endDate = self.endDate
{
let start = startDate.timeIntervalSinceReferenceDate
let end = endDate.timeIntervalSinceReferenceDate
let now = Date().timeIntervalSinceReferenceDate
let length = end - start
let progress = now - start
return progress / length
}
return 0
}
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 = premiereDateFormatted else { return nil }
return L10n.airWithDate(premiereDateFormatted)
}
var premiereDateFormatted: 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.isEmpty
}
var hasRatings: Bool {
[criticRating, communityRating].oneSatisfies { $0 != nil }
}
// MARK: Chapter Images
var fullChapterInfo: [ChapterInfo.FullInfo] {
guard let chapters else { return [] }
let ranges: [Range<Int>] = []
.appending(chapters.map(\.startTimeSeconds))
.appending(runTimeSeconds + 1)
.adjacentPairs()
.map { $0 ..< $1 }
return chapters
.enumerated()
.map { index, chapterInfo in
let client = Container.userSession.callAsFunction().client
let parameters = Paths.GetItemImageParameters(
maxWidth: 500,
quality: 90,
imageIndex: index
)
let request = Paths.getItemImage(
itemID: id ?? "",
imageType: ImageType.chapter.rawValue,
parameters: parameters
)
let imageURL = client.fullURL(with: request)
let range = ranges.first(where: { $0.first == chapterInfo.startTimeSeconds }) ?? startTimeSeconds ..< startTimeSeconds + 1
return ChapterInfo.FullInfo(
chapterInfo: chapterInfo,
imageSource: .init(url: imageURL),
secondsRange: range
)
}
}
// 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
}
}
// TODO: Don't use spoof objects as a placeholder or no results
static var placeHolder: BaseItemDto {
.init(
id: "1",
name: "Placeholder",
overview: String(repeating: "a", count: 100)
// indexNumber: 20
)
}
static func randomItem() -> BaseItemDto {
.init(
id: UUID().uuidString,
name: "Lorem Ipsum",
overview: "Lorem ipsum dolor sit amet"
)
}
static var noResults: BaseItemDto {
.init(name: L10n.noResults)
}
}