yattee/Model/Stream.swift

279 lines
7.2 KiB
Swift
Raw Normal View History

2021-06-14 18:05:02 +00:00
import AVFoundation
2021-09-25 08:18:22 +00:00
import Defaults
2021-06-14 18:05:02 +00:00
import Foundation
// swiftlint:disable:next final_class
2021-10-16 22:48:58 +00:00
class Stream: Equatable, Hashable, Identifiable {
2021-09-25 08:18:22 +00:00
enum Resolution: String, CaseIterable, Comparable, Defaults.Serializable {
2022-04-09 21:19:00 +00:00
case hd2160p60
case hd2160p50
case hd2160p48
2022-06-17 10:52:10 +00:00
case hd2160p30
2022-03-27 18:59:22 +00:00
case hd1440p60
case hd1440p50
case hd1440p48
2022-06-17 10:52:10 +00:00
case hd1440p30
2022-03-27 18:59:22 +00:00
case hd1080p60
case hd1080p50
case hd1080p48
2022-06-17 10:52:10 +00:00
case hd1080p30
2022-03-27 18:59:22 +00:00
case hd720p60
case hd720p50
case hd720p48
2022-06-17 10:52:10 +00:00
case hd720p30
case sd480p30
case sd360p30
case sd240p30
case sd144p30
2022-03-27 18:59:22 +00:00
case unknown
2021-10-16 22:48:58 +00:00
var name: String {
2022-06-17 10:52:10 +00:00
"\(height)p\(refreshRate != -1 && refreshRate != 30 ? ", \(refreshRate) fps" : "")"
2021-10-16 22:48:58 +00:00
}
2021-07-22 12:43:13 +00:00
var height: Int {
2021-10-16 22:48:58 +00:00
if self == .unknown {
return -1
}
let resolutionPart = rawValue.components(separatedBy: "p").first!
return Int(resolutionPart.components(separatedBy: CharacterSet.decimalDigits.inverted).joined())!
}
var refreshRate: Int {
if self == .unknown {
return -1
}
let refreshRatePart = rawValue.components(separatedBy: "p")[1]
2022-06-17 10:52:10 +00:00
if refreshRatePart.isEmpty {
return 30
}
2021-10-16 22:48:58 +00:00
return Int(refreshRatePart.components(separatedBy: CharacterSet.decimalDigits.inverted).joined()) ?? -1
2021-07-22 12:43:13 +00:00
}
// These values are an approximation.
// https://support.google.com/youtube/answer/1722171?hl=en#zippy=%2Cbitrate
var bitrate: Int {
switch self {
case .hd2160p60, .hd2160p50, .hd2160p48, .hd2160p30:
return 56000000 // 56 Mbit/s
case .hd1440p60, .hd1440p50, .hd1440p48, .hd1440p30:
return 24000000 // 24 Mbit/s
case .hd1080p60, .hd1080p50, .hd1080p48, .hd1080p30:
return 12000000 // 12 Mbit/s
case .hd720p60, .hd720p50, .hd720p48, .hd720p30:
return 9500000 // 9.5 Mbit/s
case .sd480p30:
return 4000000 // 4 Mbit/s
case .sd360p30:
return 1500000 // 1.5 Mbit/s
case .sd240p30:
return 1000000 // 1 Mbit/s
case .sd144p30:
return 600000 // 0.6 Mbit/s
case .unknown:
return 0
}
}
2023-06-17 12:09:51 +00:00
static func from(resolution: String, fps: Int? = nil) -> Self {
2022-06-17 10:52:10 +00:00
allCases.first { $0.rawValue.contains(resolution) && $0.refreshRate == (fps ?? 30) } ?? .unknown
2021-07-22 12:43:13 +00:00
}
2023-06-17 12:09:51 +00:00
static func < (lhs: Self, rhs: Self) -> Bool {
lhs.height == rhs.height ? (lhs.refreshRate < rhs.refreshRate) : (lhs.height < rhs.height)
2021-07-22 12:43:13 +00:00
}
}
enum Kind: String, Comparable {
case hls, adaptive, stream
2021-07-22 12:43:13 +00:00
private var sortOrder: Int {
switch self {
2021-10-16 22:48:58 +00:00
case .hls:
2021-07-22 12:43:13 +00:00
return 0
2021-10-16 22:48:58 +00:00
case .stream:
2021-07-22 12:43:13 +00:00
return 1
2021-10-16 22:48:58 +00:00
case .adaptive:
return 2
2021-07-22 12:43:13 +00:00
}
}
2023-06-17 12:09:51 +00:00
static func < (lhs: Self, rhs: Self) -> Bool {
2021-07-22 12:43:13 +00:00
lhs.sortOrder < rhs.sortOrder
}
}
enum Format: String {
case avc1
case mp4
case av1
case webm
case hls
case stream
case unknown
var description: String {
switch self {
case .webm:
return "WebM"
case .hls:
return "adaptive (HLS)"
case .stream:
return "Stream"
default:
return rawValue.uppercased()
}
}
static func from(_ string: String) -> Self {
let lowercased = string.lowercased()
2023-06-17 12:09:51 +00:00
if lowercased.contains("avc1") {
return .avc1
2023-06-17 12:09:51 +00:00
}
if lowercased.contains("mpeg_4") || lowercased.contains("mp4") {
return .mp4
}
2023-06-17 12:09:51 +00:00
if lowercased.contains("av01") {
return .av1
2023-06-17 12:09:51 +00:00
}
if lowercased.contains("webm") {
return .webm
}
if lowercased.contains("stream") {
return .stream
}
if lowercased.contains("hls") {
return .hls
}
2023-06-17 12:09:51 +00:00
return .unknown
}
}
2021-10-16 22:48:58 +00:00
let id = UUID()
var instance: Instance!
var audioAsset: AVURLAsset!
var videoAsset: AVURLAsset!
var hlsURL: URL!
2022-11-10 17:11:28 +00:00
var localURL: URL!
2021-06-14 18:05:02 +00:00
2021-10-16 22:48:58 +00:00
var resolution: Resolution!
var kind: Kind!
var format: Format!
2021-06-14 18:05:02 +00:00
2022-08-20 21:05:40 +00:00
var encoding: String?
var videoFormat: String?
var bitrate: Int?
2021-06-14 18:05:02 +00:00
2021-10-16 22:48:58 +00:00
init(
instance: Instance? = nil,
audioAsset: AVURLAsset? = nil,
videoAsset: AVURLAsset? = nil,
hlsURL: URL? = nil,
2022-11-10 17:11:28 +00:00
localURL: URL? = nil,
2021-10-16 22:48:58 +00:00
resolution: Resolution? = nil,
kind: Kind = .hls,
2022-02-16 20:23:11 +00:00
encoding: String? = nil,
videoFormat: String? = nil,
bitrate: Int? = nil
2021-10-16 22:48:58 +00:00
) {
self.instance = instance
2021-06-14 18:05:02 +00:00
self.audioAsset = audioAsset
self.videoAsset = videoAsset
2021-10-16 22:48:58 +00:00
self.hlsURL = hlsURL
2022-11-10 17:11:28 +00:00
self.localURL = localURL
2021-06-14 18:05:02 +00:00
self.resolution = resolution
2021-07-22 12:43:13 +00:00
self.kind = kind
2021-06-14 18:05:02 +00:00
self.encoding = encoding
format = .from(videoFormat ?? "")
self.bitrate = bitrate
2021-06-14 18:05:02 +00:00
}
2022-11-10 17:11:28 +00:00
var isLocal: Bool {
localURL != nil
}
2023-05-20 20:49:10 +00:00
var isHLS: Bool {
hlsURL != nil
}
2021-10-16 22:48:58 +00:00
var quality: String {
2022-11-10 17:11:28 +00:00
guard localURL.isNil else { return "Opened File" }
2024-05-09 19:15:14 +00:00
if kind == .hls {
return "adaptive (HLS)"
}
return resolution.name
2022-02-16 20:23:11 +00:00
}
2022-06-14 22:41:49 +00:00
var shortQuality: String {
2022-11-10 17:11:28 +00:00
guard localURL.isNil else { return "File" }
if kind == .hls {
2024-05-09 19:15:14 +00:00
return "adaptive (HLS)"
}
if kind == .stream {
return resolution.name
2022-06-14 22:41:49 +00:00
}
return resolutionAndFormat
2022-06-14 22:41:49 +00:00
}
2021-06-14 18:05:02 +00:00
var description: String {
2022-11-10 17:11:28 +00:00
guard localURL.isNil else { return resolutionAndFormat }
let instanceString = instance.isNil ? "" : " - (\(instance!.description))"
2024-05-09 19:15:14 +00:00
return format != .hls ? "\(resolutionAndFormat)\(instanceString)" : "adaptive (HLS)\(instanceString)"
}
var resolutionAndFormat: String {
let formatString = format == .unknown ? "" : " (\(format.description))"
return "\(quality)\(formatString)"
2021-06-14 18:05:02 +00:00
}
2021-06-15 16:35:21 +00:00
var assets: [AVURLAsset] {
[audioAsset, videoAsset]
}
2021-10-16 22:48:58 +00:00
var videoAssetContainsAudio: Bool {
assets.dropFirst().allSatisfy { $0.url == assets.first!.url }
2021-07-22 12:43:13 +00:00
}
2021-10-16 22:48:58 +00:00
var singleAssetURL: URL? {
2022-11-10 17:11:28 +00:00
guard localURL.isNil else {
return URLBookmarkModel.shared.loadBookmark(localURL) ?? localURL
}
2021-10-16 22:48:58 +00:00
if kind == .hls {
return hlsURL
2023-06-17 12:09:51 +00:00
}
if videoAssetContainsAudio {
2021-10-16 22:48:58 +00:00
return videoAsset.url
}
return nil
}
2021-06-14 18:05:02 +00:00
static func == (lhs: Stream, rhs: Stream) -> Bool {
2021-10-16 22:48:58 +00:00
lhs.id == rhs.id
2021-07-22 12:43:13 +00:00
}
func hash(into hasher: inout Hasher) {
2023-05-20 20:49:10 +00:00
if let url = videoAsset?.url {
hasher.combine(url)
}
if let url = audioAsset?.url {
hasher.combine(url)
}
if let url = hlsURL {
hasher.combine(url)
}
2021-10-16 22:48:58 +00:00
}
2021-06-14 18:05:02 +00:00
}