1
0
mirror of https://github.com/yattee/yattee.git synced 2025-01-07 18:10:33 +05:30
yattee/Model/Stream.swift

413 lines
12 KiB
Swift
Raw Normal View History

2021-06-14 23:35:02 +05:30
import AVFoundation
2021-09-25 13:48:22 +05:30
import Defaults
2021-06-14 23:35:02 +05:30
import Foundation
// swiftlint:disable:next final_class
2021-10-17 04:18:58 +05:30
class Stream: Equatable, Hashable, Identifiable {
enum Resolution: Comparable, Codable, Defaults.Serializable {
case predefined(PredefinedResolution)
case custom(height: Int, refreshRate: Int)
enum PredefinedResolution: String, CaseIterable, Codable {
// 8K UHD (16:9) Resolutions
case hd4320p60, hd4320p30
// 4K UHD (16:9) Resolutions
case hd2160p60, hd2160p30
// 1440p (16:9) Resolutions
case hd1440p60, hd1440p30
// 1080p (Full HD, 16:9) Resolutions
case hd1080p60, hd1080p30
// 720p (HD, 16:9) Resolutions
case hd720p60, hd720p30
// Standard Definition (SD) Resolutions
case sd480p30
case sd360p30
case sd240p30
case sd144p30
}
2021-10-17 04:18:58 +05:30
var name: String {
switch self {
case let .predefined(predefined):
return predefined.rawValue
case let .custom(height, refreshRate):
return "\(height)p\(refreshRate != 30 ? ", \(refreshRate) fps" : "")"
}
2021-10-17 04:18:58 +05:30
}
2021-07-22 18:13:13 +05:30
var height: Int {
switch self {
case let .predefined(predefined):
return predefined.height
case let .custom(height, _):
return height
2021-10-17 04:18:58 +05:30
}
}
var refreshRate: Int {
switch self {
case let .predefined(predefined):
return predefined.refreshRate
case let .custom(_, refreshRate):
return refreshRate
2022-06-17 16:22:10 +05:30
}
2021-07-22 18:13:13 +05:30
}
var bitrate: Int {
switch self {
case let .predefined(predefined):
return predefined.bitrate
case let .custom(height, refreshRate):
// Find the closest predefined resolution based on height and refresh rate
let closestPredefined = Stream.Resolution.PredefinedResolution.allCases.min {
abs($0.height - height) + abs($0.refreshRate - refreshRate) <
abs($1.height - height) + abs($1.refreshRate - refreshRate)
}
// Return the bitrate of the closest predefined resolution or a default bitrate if no close match is found
return closestPredefined?.bitrate ?? 5_000_000
}
}
static func from(resolution: String, fps: Int? = nil) -> Self {
if let predefined = PredefinedResolution(rawValue: resolution) {
return .predefined(predefined)
}
// Attempt to parse height and refresh rate
if let height = Int(resolution.components(separatedBy: "p").first ?? ""), height > 0 {
let refreshRate = fps ?? 30
return .custom(height: height, refreshRate: refreshRate)
}
// Default behavior if parsing fails
return .custom(height: 720, refreshRate: 30)
}
static func < (lhs: Self, rhs: Self) -> Bool {
lhs.height == rhs.height ? (lhs.refreshRate < rhs.refreshRate) : (lhs.height < rhs.height)
}
enum CodingKeys: String, CodingKey {
case predefined
case custom
case height
case refreshRate
}
init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
if let predefinedValue = try? container.decode(PredefinedResolution.self, forKey: .predefined) {
self = .predefined(predefinedValue)
} else if let height = try? container.decode(Int.self, forKey: .height),
let refreshRate = try? container.decode(Int.self, forKey: .refreshRate)
{
self = .custom(height: height, refreshRate: refreshRate)
} else {
// Set default resolution to 720p 30 if decoding fails
self = .custom(height: 720, refreshRate: 30)
}
2021-07-22 18:13:13 +05:30
}
func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: CodingKeys.self)
switch self {
case let .predefined(predefinedValue):
try container.encode(predefinedValue, forKey: .predefined)
case let .custom(height, refreshRate):
try container.encode(height, forKey: .height)
try container.encode(refreshRate, forKey: .refreshRate)
}
2021-07-22 18:13:13 +05:30
}
}
enum Kind: String, Comparable {
case hls, adaptive, stream
2021-07-22 18:13:13 +05:30
private var sortOrder: Int {
switch self {
2021-10-17 04:18:58 +05:30
case .hls:
2021-07-22 18:13:13 +05:30
return 0
2021-10-17 04:18:58 +05:30
case .stream:
2021-07-22 18:13:13 +05:30
return 1
2021-10-17 04:18:58 +05:30
case .adaptive:
return 2
2021-07-22 18:13:13 +05:30
}
}
2023-06-17 17:39:51 +05:30
static func < (lhs: Self, rhs: Self) -> Bool {
2021-07-22 18:13:13 +05:30
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 17:39:51 +05:30
if lowercased.contains("avc1") {
return .avc1
2023-06-17 17:39:51 +05:30
}
if lowercased.contains("mpeg_4") || lowercased.contains("mp4") {
return .mp4
}
2023-06-17 17:39:51 +05:30
if lowercased.contains("av01") {
return .av1
2023-06-17 17:39:51 +05:30
}
if lowercased.contains("webm") {
return .webm
}
if lowercased.contains("stream") {
return .stream
}
if lowercased.contains("hls") {
return .hls
}
2023-06-17 17:39:51 +05:30
return .unknown
}
}
2021-10-17 04:18:58 +05:30
let id = UUID()
var instance: Instance!
var audioAsset: AVURLAsset!
var videoAsset: AVURLAsset!
var hlsURL: URL!
2022-11-10 22:41:28 +05:30
var localURL: URL!
2021-06-14 23:35:02 +05:30
2021-10-17 04:18:58 +05:30
var resolution: Resolution!
var kind: Kind!
var format: Format!
2021-06-14 23:35:02 +05:30
2022-08-21 02:35:40 +05:30
var encoding: String?
var videoFormat: String?
var bitrate: Int?
var requestRange: String?
2021-06-14 23:35:02 +05:30
2021-10-17 04:18:58 +05:30
init(
instance: Instance? = nil,
audioAsset: AVURLAsset? = nil,
videoAsset: AVURLAsset? = nil,
hlsURL: URL? = nil,
2022-11-10 22:41:28 +05:30
localURL: URL? = nil,
2021-10-17 04:18:58 +05:30
resolution: Resolution? = nil,
kind: Kind = .hls,
2022-02-17 01:53:11 +05:30
encoding: String? = nil,
videoFormat: String? = nil,
bitrate: Int? = nil,
requestRange: String? = nil
2021-10-17 04:18:58 +05:30
) {
self.instance = instance
2021-06-14 23:35:02 +05:30
self.audioAsset = audioAsset
self.videoAsset = videoAsset
2021-10-17 04:18:58 +05:30
self.hlsURL = hlsURL
2022-11-10 22:41:28 +05:30
self.localURL = localURL
2021-06-14 23:35:02 +05:30
self.resolution = resolution
2021-07-22 18:13:13 +05:30
self.kind = kind
2021-06-14 23:35:02 +05:30
self.encoding = encoding
format = .from(videoFormat ?? "")
self.bitrate = bitrate
self.requestRange = requestRange
2021-06-14 23:35:02 +05:30
}
2022-11-10 22:41:28 +05:30
var isLocal: Bool {
localURL != nil
}
2023-05-21 02:19:10 +05:30
var isHLS: Bool {
hlsURL != nil
}
2021-10-17 04:18:58 +05:30
var quality: String {
2022-11-10 22:41:28 +05:30
guard localURL.isNil else { return "Opened File" }
2024-05-10 00:45:14 +05:30
if kind == .hls {
return "adaptive (HLS)"
}
return resolution.name
2022-02-17 01:53:11 +05:30
}
2022-06-15 04:11:49 +05:30
var shortQuality: String {
2022-11-10 22:41:28 +05:30
guard localURL.isNil else { return "File" }
if kind == .hls {
2024-05-10 00:45:14 +05:30
return "adaptive (HLS)"
}
if kind == .stream {
return resolution.name
2022-06-15 04:11:49 +05:30
}
return resolutionAndFormat
2022-06-15 04:11:49 +05:30
}
2021-06-14 23:35:02 +05:30
var description: String {
2022-11-10 22:41:28 +05:30
guard localURL.isNil else { return resolutionAndFormat }
let instanceString = instance.isNil ? "" : " - (\(instance!.description))"
2024-05-10 00:45:14 +05:30
return format != .hls ? "\(resolutionAndFormat)\(instanceString)" : "adaptive (HLS)\(instanceString)"
}
var resolutionAndFormat: String {
let formatString = format == .unknown ? "" : " (\(format.description))"
return "\(quality)\(formatString)"
2021-06-14 23:35:02 +05:30
}
2021-06-15 22:05:21 +05:30
var assets: [AVURLAsset] {
[audioAsset, videoAsset]
}
2021-10-17 04:18:58 +05:30
var videoAssetContainsAudio: Bool {
assets.dropFirst().allSatisfy { $0.url == assets.first!.url }
2021-07-22 18:13:13 +05:30
}
2021-10-17 04:18:58 +05:30
var singleAssetURL: URL? {
2022-11-10 22:41:28 +05:30
guard localURL.isNil else {
return URLBookmarkModel.shared.loadBookmark(localURL) ?? localURL
}
2021-10-17 04:18:58 +05:30
if kind == .hls {
return hlsURL
2023-06-17 17:39:51 +05:30
}
if videoAssetContainsAudio {
2021-10-17 04:18:58 +05:30
return videoAsset.url
}
return nil
}
2021-06-14 23:35:02 +05:30
static func == (lhs: Stream, rhs: Stream) -> Bool {
2021-10-17 04:18:58 +05:30
lhs.id == rhs.id
2021-07-22 18:13:13 +05:30
}
func hash(into hasher: inout Hasher) {
2023-05-21 02:19:10 +05:30
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-17 04:18:58 +05:30
}
2021-06-14 23:35:02 +05:30
}
extension Stream.Resolution.PredefinedResolution {
var height: Int {
switch self {
// 8K UHD (16:9) Resolutions
case .hd4320p60, .hd4320p30:
return 4320
// 4K UHD (16:9) Resolutions
case .hd2160p60, .hd2160p30:
return 2160
// 1440p (16:9) Resolutions
case .hd1440p60, .hd1440p30:
return 1440
// 1080p (Full HD, 16:9) Resolutions
case .hd1080p60, .hd1080p30:
return 1080
// 720p (HD, 16:9) Resolutions
case .hd720p60, .hd720p30:
return 720
// Standard Definition (SD) Resolutions
case .sd480p30:
return 480
case .sd360p30:
return 360
case .sd240p30:
return 240
case .sd144p30:
return 144
}
}
var refreshRate: Int {
switch self {
// 60 fps Resolutions
case .hd4320p60, .hd2160p60, .hd1440p60, .hd1080p60, .hd720p60:
return 60
// 30 fps Resolutions
case .hd4320p30, .hd2160p30, .hd1440p30, .hd1080p30, .hd720p30,
.sd480p30, .sd360p30, .sd240p30, .sd144p30:
return 30
}
}
// These values are an approximation.
// https://support.google.com/youtube/answer/1722171?hl=en#zippy=%2Cbitrate
var bitrate: Int {
switch self {
// 8K UHD (16:9) Resolutions
case .hd4320p60:
return 180_000_000 // Midpoint between 120 Mbps and 240 Mbps
case .hd4320p30:
return 120_000_000 // Midpoint between 80 Mbps and 160 Mbps
// 4K UHD (16:9) Resolutions
case .hd2160p60:
return 60_500_000 // Midpoint between 53 Mbps and 68 Mbps
case .hd2160p30:
return 40_000_000 // Midpoint between 35 Mbps and 45 Mbps
// 1440p (2K) Resolutions
case .hd1440p60:
return 24_000_000 // 24 Mbps
case .hd1440p30:
return 16_000_000 // 16 Mbps
// 1080p (Full HD, 16:9) Resolutions
case .hd1080p60:
return 12_000_000 // 12 Mbps
case .hd1080p30:
return 8_000_000 // 8 Mbps
// 720p (HD, 16:9) Resolutions
case .hd720p60:
return 7_500_000 // 7.5 Mbps
case .hd720p30:
return 5_000_000 // 5 Mbps
// Standard Definition (SD) Resolutions
case .sd480p30:
return 2_500_000 // 2.5 Mbps
case .sd360p30:
return 1_000_000 // 1 Mbps
case .sd240p30:
return 1_000_000 // 1 Mbps
case .sd144p30:
return 600_000 // 0.6 Mbps
}
}
}