mirror of
https://github.com/yattee/yattee.git
synced 2024-12-22 21:43:41 +00:00
Add demo instance, remove public manifest
This commit is contained in:
parent
8ab6a0fa89
commit
1fe01808a4
@ -55,6 +55,9 @@ final class AccountValidator: Service {
|
|||||||
|
|
||||||
case .piped:
|
case .piped:
|
||||||
return resource("/streams/dQw4w9WgXcQ")
|
return resource("/streams/dQw4w9WgXcQ")
|
||||||
|
|
||||||
|
case .demoApp:
|
||||||
|
return resource("/")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7,6 +7,7 @@ final class AccountsModel: ObservableObject {
|
|||||||
|
|
||||||
@Published private var invidious = InvidiousAPI()
|
@Published private var invidious = InvidiousAPI()
|
||||||
@Published private var piped = PipedAPI()
|
@Published private var piped = PipedAPI()
|
||||||
|
@Published private var demo = DemoAppAPI()
|
||||||
|
|
||||||
@Published var publicAccount: Account?
|
@Published var publicAccount: Account?
|
||||||
|
|
||||||
@ -33,7 +34,14 @@ final class AccountsModel: ObservableObject {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var api: VideosAPI {
|
var api: VideosAPI {
|
||||||
app == .piped ? piped : invidious
|
switch app {
|
||||||
|
case .piped:
|
||||||
|
return piped
|
||||||
|
case .invidious:
|
||||||
|
return invidious
|
||||||
|
case .demoApp:
|
||||||
|
return demo
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var isEmpty: Bool {
|
var isEmpty: Bool {
|
||||||
@ -44,6 +52,10 @@ final class AccountsModel: ObservableObject {
|
|||||||
!isEmpty && !current.anonymous && api.signedIn
|
!isEmpty && !current.anonymous && api.signedIn
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var isDemo: Bool {
|
||||||
|
current?.app == .demoApp
|
||||||
|
}
|
||||||
|
|
||||||
init() {
|
init() {
|
||||||
cancellables.append(
|
cancellables.append(
|
||||||
invidious.objectWillChange.sink { [weak self] _ in self?.objectWillChange.send() }
|
invidious.objectWillChange.sink { [weak self] _ in self?.objectWillChange.send() }
|
||||||
@ -79,6 +91,8 @@ final class AccountsModel: ObservableObject {
|
|||||||
invidious.setAccount(account)
|
invidious.setAccount(account)
|
||||||
case .piped:
|
case .piped:
|
||||||
piped.setAccount(account)
|
piped.setAccount(account)
|
||||||
|
case .demoApp:
|
||||||
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
Defaults[.lastAccountIsPublic] = account.isPublic
|
Defaults[.lastAccountIsPublic] = account.isPublic
|
||||||
|
@ -26,6 +26,8 @@ struct Instance: Defaults.Serializable, Hashable, Identifiable {
|
|||||||
return InvidiousAPI(account: anonymousAccount)
|
return InvidiousAPI(account: anonymousAccount)
|
||||||
case .piped:
|
case .piped:
|
||||||
return PipedAPI(account: anonymousAccount)
|
return PipedAPI(account: anonymousAccount)
|
||||||
|
case .demoApp:
|
||||||
|
return DemoAppAPI()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -34,7 +36,9 @@ struct Instance: Defaults.Serializable, Hashable, Identifiable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var longDescription: String {
|
var longDescription: String {
|
||||||
name.isEmpty ? "\(app.name) - \(apiURL)" : "\(app.name) - \(name) (\(apiURL))"
|
guard app != .demoApp else { return "Demo" }
|
||||||
|
|
||||||
|
return name.isEmpty ? "\(app.name) - \(apiURL)" : "\(app.name) - \(name) (\(apiURL))"
|
||||||
}
|
}
|
||||||
|
|
||||||
var shortDescription: String {
|
var shortDescription: String {
|
||||||
|
395
Model/Applications/DemoAppAPI.swift
Normal file
395
Model/Applications/DemoAppAPI.swift
Normal file
@ -0,0 +1,395 @@
|
|||||||
|
import AVFoundation
|
||||||
|
import Foundation
|
||||||
|
import Siesta
|
||||||
|
import SwiftyJSON
|
||||||
|
|
||||||
|
final class DemoAppAPI: Service, ObservableObject, VideosAPI {
|
||||||
|
static var url = "https://r.yattee.stream/demo"
|
||||||
|
|
||||||
|
var account: Account! {
|
||||||
|
.init(
|
||||||
|
id: UUID().uuidString,
|
||||||
|
app: .demoApp,
|
||||||
|
name: "Demo",
|
||||||
|
url: Self.url,
|
||||||
|
anonymous: true
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
var signedIn: Bool {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
init() {
|
||||||
|
super.init()
|
||||||
|
|
||||||
|
configure()
|
||||||
|
}
|
||||||
|
|
||||||
|
func configure() {
|
||||||
|
configure {
|
||||||
|
$0.pipeline[.parsing].add(SwiftyJSONTransformer, contentTypes: ["*/json"])
|
||||||
|
}
|
||||||
|
|
||||||
|
configureTransformer(pathPattern("channels/*"), requestMethods: [.get]) { (content: Entity<Any>) -> Channel? in
|
||||||
|
self.extractChannel(from: content.json)
|
||||||
|
}
|
||||||
|
|
||||||
|
configureTransformer(pathPattern("search*"), requestMethods: [.get]) { (content: Entity<Any>) -> SearchPage in
|
||||||
|
let nextPage = content.json.dictionaryValue["nextpage"]?.string
|
||||||
|
return SearchPage(
|
||||||
|
results: self.extractContentItems(from: content.json.dictionaryValue["items"]!),
|
||||||
|
nextPage: nextPage,
|
||||||
|
last: nextPage == "null"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
configureTransformer(pathPattern("suggestions*")) { (content: Entity<JSON>) -> [String] in
|
||||||
|
content.json.arrayValue.map(String.init)
|
||||||
|
}
|
||||||
|
|
||||||
|
configureTransformer(pathPattern("videos/*")) { (content: Entity<JSON>) -> Video? in
|
||||||
|
self.extractVideo(from: content.json)
|
||||||
|
}
|
||||||
|
|
||||||
|
configureTransformer(pathPattern("trending*")) { (content: Entity<JSON>) -> [Video] in
|
||||||
|
self.extractVideos(from: content.json)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func channel(_ channel: String) -> Resource {
|
||||||
|
resource(baseURL: Self.url, path: "/channels/\(channel).json")
|
||||||
|
}
|
||||||
|
|
||||||
|
func channelByName(_: String) -> Resource? {
|
||||||
|
resource(baseURL: Self.url, path: "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func channelByUsername(_: String) -> Resource? {
|
||||||
|
resource(baseURL: Self.url, path: "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func channelVideos(_ id: String) -> Resource {
|
||||||
|
resource(baseURL: Self.url, path: "/channels/\(id).json")
|
||||||
|
}
|
||||||
|
|
||||||
|
func trending(country _: Country, category _: TrendingCategory?) -> Resource {
|
||||||
|
resource(baseURL: Self.url, path: "/trending.json")
|
||||||
|
}
|
||||||
|
|
||||||
|
func search(_ query: SearchQuery, page: String?) -> Resource {
|
||||||
|
resource(baseURL: Self.url, path: "/search.json")
|
||||||
|
.withParam("q", query.query)
|
||||||
|
.withParam("p", page)
|
||||||
|
}
|
||||||
|
|
||||||
|
func searchSuggestions(query _: String) -> Resource {
|
||||||
|
resource(baseURL: Self.url, path: "/suggestions.json")
|
||||||
|
}
|
||||||
|
|
||||||
|
func video(_ id: Video.ID) -> Resource {
|
||||||
|
resource(baseURL: Self.url, path: "/videos/\(id).json")
|
||||||
|
}
|
||||||
|
|
||||||
|
var subscriptions: Resource?
|
||||||
|
|
||||||
|
var feed: Resource?
|
||||||
|
|
||||||
|
var home: Resource?
|
||||||
|
|
||||||
|
var popular: Resource?
|
||||||
|
|
||||||
|
var playlists: Resource?
|
||||||
|
|
||||||
|
func subscribe(_: String, onCompletion _: @escaping () -> Void) {}
|
||||||
|
|
||||||
|
func unsubscribe(_: String, onCompletion _: @escaping () -> Void) {}
|
||||||
|
|
||||||
|
func playlist(_: String) -> Resource? {
|
||||||
|
resource(baseURL: Self.url, path: "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func playlistVideo(_: String, _: String) -> Resource? {
|
||||||
|
resource(baseURL: Self.url, path: "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func playlistVideos(_: String) -> Resource? {
|
||||||
|
resource(baseURL: Self.url, path: "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func addVideoToPlaylist(_: String, _: String, onFailure _: @escaping (RequestError) -> Void, onSuccess _: @escaping () -> Void) {}
|
||||||
|
|
||||||
|
func removeVideoFromPlaylist(_: String, _: String, onFailure _: @escaping (RequestError) -> Void, onSuccess _: @escaping () -> Void) {}
|
||||||
|
|
||||||
|
func playlistForm(_: String, _: String, playlist _: Playlist?, onFailure _: @escaping (RequestError) -> Void, onSuccess _: @escaping (Playlist?) -> Void) {}
|
||||||
|
|
||||||
|
func deletePlaylist(_: Playlist, onFailure _: @escaping (RequestError) -> Void, onSuccess _: @escaping () -> Void) {}
|
||||||
|
|
||||||
|
func channelPlaylist(_: String) -> Resource? {
|
||||||
|
resource(baseURL: Self.url, path: "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func comments(_: Video.ID, page _: String?) -> Resource? {
|
||||||
|
resource(baseURL: Self.url, path: "")
|
||||||
|
}
|
||||||
|
|
||||||
|
private func pathPattern(_ path: String) -> String {
|
||||||
|
"**\(Self.url)/\(path)"
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractChannel(from content: JSON) -> Channel? {
|
||||||
|
let attributes = content.dictionaryValue
|
||||||
|
guard let id = attributes["id"]?.string ??
|
||||||
|
(attributes["url"] ?? attributes["uploaderUrl"])?.string?.components(separatedBy: "/").last
|
||||||
|
else {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
let subscriptionsCount = attributes["subscriberCount"]?.int ?? attributes["subscribers"]?.int
|
||||||
|
|
||||||
|
var videos = [Video]()
|
||||||
|
if let relatedStreams = attributes["relatedStreams"] {
|
||||||
|
videos = extractVideos(from: relatedStreams)
|
||||||
|
}
|
||||||
|
|
||||||
|
let name = attributes["name"]?.string ??
|
||||||
|
attributes["uploaderName"]?.string ??
|
||||||
|
attributes["uploader"]?.string ?? ""
|
||||||
|
|
||||||
|
let thumbnailURL = attributes["avatarUrl"]?.url ??
|
||||||
|
attributes["uploaderAvatar"]?.url ??
|
||||||
|
attributes["avatar"]?.url ??
|
||||||
|
attributes["thumbnail"]?.url
|
||||||
|
|
||||||
|
return Channel(
|
||||||
|
id: id,
|
||||||
|
name: name,
|
||||||
|
thumbnailURL: thumbnailURL,
|
||||||
|
subscriptionsCount: subscriptionsCount,
|
||||||
|
videos: videos
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractVideos(from content: JSON) -> [Video] {
|
||||||
|
content.arrayValue.compactMap(extractVideo(from:))
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractVideo(from content: JSON) -> Video? {
|
||||||
|
let details = content.dictionaryValue
|
||||||
|
|
||||||
|
if let url = details["url"]?.string {
|
||||||
|
guard url.contains("/watch") else {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let channelId = details["uploaderUrl"]?.string?.components(separatedBy: "/").last ?? "unknown"
|
||||||
|
|
||||||
|
let thumbnails: [Thumbnail] = Thumbnail.Quality.allCases.compactMap {
|
||||||
|
if let url = buildThumbnailURL(from: content, quality: $0) {
|
||||||
|
return Thumbnail(url: url, quality: $0)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
let author = details["uploaderName"]?.string ?? details["uploader"]?.string ?? ""
|
||||||
|
let authorThumbnailURL = details["avatarUrl"]?.url ?? details["uploaderAvatar"]?.url ?? details["avatar"]?.url
|
||||||
|
let subscriptionsCount = details["uploaderSubscriberCount"]?.int
|
||||||
|
|
||||||
|
let uploaded = details["uploaded"]?.double
|
||||||
|
var published = (uploaded.isNil || uploaded == -1) ? nil : (uploaded! / 1000).formattedAsRelativeTime()
|
||||||
|
if published.isNil {
|
||||||
|
published = (details["uploadedDate"] ?? details["uploadDate"])?.string ?? ""
|
||||||
|
}
|
||||||
|
|
||||||
|
let live = details["livestream"]?.bool ?? (details["duration"]?.int == -1)
|
||||||
|
|
||||||
|
let description = extractDescription(from: content) ?? ""
|
||||||
|
|
||||||
|
return Video(
|
||||||
|
videoID: extractID(from: content),
|
||||||
|
title: details["title"]?.string ?? "",
|
||||||
|
author: author,
|
||||||
|
length: details["duration"]?.double ?? 0,
|
||||||
|
published: published ?? "",
|
||||||
|
views: details["views"]?.int ?? 0,
|
||||||
|
description: description,
|
||||||
|
channel: Channel(id: channelId, name: author, thumbnailURL: authorThumbnailURL, subscriptionsCount: subscriptionsCount),
|
||||||
|
thumbnails: thumbnails,
|
||||||
|
live: live,
|
||||||
|
likes: details["likes"]?.int,
|
||||||
|
dislikes: details["dislikes"]?.int,
|
||||||
|
streams: extractStreams(from: content),
|
||||||
|
related: extractRelated(from: content)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private func buildThumbnailURL(from content: JSON, quality: Thumbnail.Quality) -> URL? {
|
||||||
|
guard let thumbnailURL = extractThumbnailURL(from: content) else {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return URL(string: thumbnailURL
|
||||||
|
.absoluteString
|
||||||
|
.replacingOccurrences(of: "hqdefault", with: quality.filename)
|
||||||
|
.replacingOccurrences(of: "maxresdefault", with: quality.filename)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractID(from content: JSON) -> Video.ID {
|
||||||
|
content.dictionaryValue["url"]?.string?.components(separatedBy: "?v=").last ??
|
||||||
|
extractThumbnailURL(from: content)?.relativeString.components(separatedBy: "/")[5].replacingFirstOccurrence(of: ".png", with: "") ?? ""
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractDescription(from content: JSON) -> String? {
|
||||||
|
guard var description = content.dictionaryValue["description"]?.string else {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
description = description.replacingOccurrences(
|
||||||
|
of: "<br/>|<br />|<br>",
|
||||||
|
with: "\n",
|
||||||
|
options: .regularExpression,
|
||||||
|
range: nil
|
||||||
|
)
|
||||||
|
|
||||||
|
let linkRegex = #"(<a\s+(?:[^>]*?\s+)?href=\"[^"]*\">[^<]*<\/a>)"#
|
||||||
|
let hrefRegex = #"href=\"([^"]*)\">"#
|
||||||
|
guard let hrefRegex = try? NSRegularExpression(pattern: hrefRegex) else { return description }
|
||||||
|
|
||||||
|
description = description.replacingMatches(regex: linkRegex) { matchingGroup in
|
||||||
|
let results = hrefRegex.matches(in: matchingGroup, range: NSRange(matchingGroup.startIndex..., in: matchingGroup))
|
||||||
|
|
||||||
|
if let result = results.first {
|
||||||
|
if let swiftRange = Range(result.range(at: 1), in: matchingGroup) {
|
||||||
|
return String(matchingGroup[swiftRange])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return matchingGroup
|
||||||
|
}
|
||||||
|
|
||||||
|
description = description.replacingOccurrences(of: "&", with: "&")
|
||||||
|
|
||||||
|
description = description.replacingOccurrences(
|
||||||
|
of: "<[^>]+>",
|
||||||
|
with: "",
|
||||||
|
options: .regularExpression,
|
||||||
|
range: nil
|
||||||
|
)
|
||||||
|
|
||||||
|
return description
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractStreams(from content: JSON) -> [Stream] {
|
||||||
|
var streams = [Stream]()
|
||||||
|
|
||||||
|
if let hlsURL = content.dictionaryValue["hls"]?.url {
|
||||||
|
streams.append(Stream(instance: account.instance, hlsURL: hlsURL))
|
||||||
|
}
|
||||||
|
|
||||||
|
let audioStreams = content
|
||||||
|
.dictionaryValue["audioStreams"]?
|
||||||
|
.arrayValue
|
||||||
|
.filter { $0.dictionaryValue["format"]?.string == "M4A" }
|
||||||
|
.sorted {
|
||||||
|
$0.dictionaryValue["bitrate"]?.int ?? 0 >
|
||||||
|
$1.dictionaryValue["bitrate"]?.int ?? 0
|
||||||
|
} ?? []
|
||||||
|
|
||||||
|
guard let audioStream = audioStreams.first else {
|
||||||
|
return streams
|
||||||
|
}
|
||||||
|
|
||||||
|
let videoStreams = content.dictionaryValue["videoStreams"]?.arrayValue ?? []
|
||||||
|
|
||||||
|
videoStreams.forEach { videoStream in
|
||||||
|
let videoCodec = videoStream.dictionaryValue["codec"]?.string ?? ""
|
||||||
|
|
||||||
|
guard let audioAssetUrl = audioStream.dictionaryValue["url"]?.url,
|
||||||
|
let videoAssetUrl = videoStream.dictionaryValue["url"]?.url
|
||||||
|
else {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
let audioAsset = AVURLAsset(url: audioAssetUrl)
|
||||||
|
let videoAsset = AVURLAsset(url: videoAssetUrl)
|
||||||
|
|
||||||
|
let videoOnly = videoStream.dictionaryValue["videoOnly"]?.bool ?? true
|
||||||
|
let quality = videoStream.dictionaryValue["quality"]?.string ?? "unknown"
|
||||||
|
let qualityComponents = quality.components(separatedBy: "p")
|
||||||
|
let fps = qualityComponents.count > 1 ? Int(qualityComponents[1]) : 30
|
||||||
|
let resolution = Stream.Resolution.from(resolution: quality, fps: fps)
|
||||||
|
let videoFormat = videoStream.dictionaryValue["format"]?.string
|
||||||
|
|
||||||
|
if videoOnly {
|
||||||
|
streams.append(
|
||||||
|
Stream(
|
||||||
|
instance: account.instance,
|
||||||
|
audioAsset: audioAsset,
|
||||||
|
videoAsset: videoAsset,
|
||||||
|
resolution: resolution,
|
||||||
|
kind: .adaptive,
|
||||||
|
videoFormat: videoFormat
|
||||||
|
)
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
streams.append(
|
||||||
|
SingleAssetStream(
|
||||||
|
instance: account.instance,
|
||||||
|
avAsset: videoAsset,
|
||||||
|
resolution: resolution,
|
||||||
|
kind: .stream
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return streams
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractRelated(from content: JSON) -> [Video] {
|
||||||
|
content
|
||||||
|
.dictionaryValue["relatedStreams"]?
|
||||||
|
.arrayValue
|
||||||
|
.compactMap(extractVideo(from:)) ?? []
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractThumbnailURL(from content: JSON) -> URL? {
|
||||||
|
content.dictionaryValue["thumbnail"]?.url ?? content.dictionaryValue["thumbnailUrl"]?.url
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractContentItem(from content: JSON) -> ContentItem? {
|
||||||
|
let details = content.dictionaryValue
|
||||||
|
|
||||||
|
let contentType: ContentItem.ContentType
|
||||||
|
|
||||||
|
if let url = details["url"]?.string {
|
||||||
|
if url.contains("/playlist") {
|
||||||
|
contentType = .playlist
|
||||||
|
} else if url.contains("/channel") {
|
||||||
|
contentType = .channel
|
||||||
|
} else {
|
||||||
|
contentType = .video
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
contentType = .video
|
||||||
|
}
|
||||||
|
|
||||||
|
switch contentType {
|
||||||
|
case .video:
|
||||||
|
if let video = extractVideo(from: content) {
|
||||||
|
return ContentItem(video: video)
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
private func extractContentItems(from content: JSON) -> [ContentItem] {
|
||||||
|
content.arrayValue.compactMap { extractContentItem(from: $0) }
|
||||||
|
}
|
||||||
|
}
|
@ -2,6 +2,7 @@ import Foundation
|
|||||||
|
|
||||||
enum VideosApp: String, CaseIterable {
|
enum VideosApp: String, CaseIterable {
|
||||||
case invidious, piped
|
case invidious, piped
|
||||||
|
case demoApp
|
||||||
|
|
||||||
var name: String {
|
var name: String {
|
||||||
rawValue.capitalized
|
rawValue.capitalized
|
||||||
|
@ -4,7 +4,6 @@ import Siesta
|
|||||||
import SwiftyJSON
|
import SwiftyJSON
|
||||||
|
|
||||||
final class InstancesManifest: Service, ObservableObject {
|
final class InstancesManifest: Service, ObservableObject {
|
||||||
static let builtinManifestUrl = "https://r.yattee.stream/manifest.json"
|
|
||||||
static let shared = InstancesManifest()
|
static let shared = InstancesManifest()
|
||||||
|
|
||||||
@Published var instances = [ManifestedInstance]()
|
@Published var instances = [ManifestedInstance]()
|
||||||
@ -12,10 +11,17 @@ final class InstancesManifest: Service, ObservableObject {
|
|||||||
init() {
|
init() {
|
||||||
super.init()
|
super.init()
|
||||||
|
|
||||||
|
configure()
|
||||||
|
}
|
||||||
|
|
||||||
|
func configure() {
|
||||||
|
invalidateConfiguration()
|
||||||
|
|
||||||
configure {
|
configure {
|
||||||
$0.pipeline[.parsing].add(SwiftyJSONTransformer, contentTypes: ["*/json"])
|
$0.pipeline[.parsing].add(SwiftyJSONTransformer, contentTypes: ["*/json"])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let manifestURL {
|
||||||
configureTransformer(
|
configureTransformer(
|
||||||
manifestURL,
|
manifestURL,
|
||||||
requestMethods: [.get]
|
requestMethods: [.get]
|
||||||
@ -26,6 +32,7 @@ final class InstancesManifest: Service, ObservableObject {
|
|||||||
return instances.arrayValue.compactMap(self.extractInstance)
|
return instances.arrayValue.compactMap(self.extractInstance)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func setPublicAccount(_ country: String?, accounts: AccountsModel, asCurrent: Bool = true) {
|
func setPublicAccount(_ country: String?, accounts: AccountsModel, asCurrent: Bool = true) {
|
||||||
guard let country else {
|
guard let country else {
|
||||||
@ -36,7 +43,7 @@ final class InstancesManifest: Service, ObservableObject {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
instancesList.load().onSuccess { response in
|
instancesList?.load().onSuccess { response in
|
||||||
if let instances: [ManifestedInstance] = response.typedContent() {
|
if let instances: [ManifestedInstance] = response.typedContent() {
|
||||||
guard let instance = instances.filter { $0.country == country }.randomElement() else { return }
|
guard let instance = instances.filter { $0.country == country }.randomElement() else { return }
|
||||||
let account = instance.anonymousAccount
|
let account = instance.anonymousAccount
|
||||||
@ -49,7 +56,7 @@ final class InstancesManifest: Service, ObservableObject {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func changePublicAccount(_ accounts: AccountsModel, settings: SettingsModel) {
|
func changePublicAccount(_ accounts: AccountsModel, settings: SettingsModel) {
|
||||||
instancesList.load().onSuccess { response in
|
instancesList?.load().onSuccess { response in
|
||||||
if let instances: [ManifestedInstance] = response.typedContent() {
|
if let instances: [ManifestedInstance] = response.typedContent() {
|
||||||
var countryInstances = instances.filter { $0.country == Defaults[.countryOfPublicInstances] }
|
var countryInstances = instances.filter { $0.country == Defaults[.countryOfPublicInstances] }
|
||||||
let region = countryInstances.first?.region ?? "Europe"
|
let region = countryInstances.first?.region ?? "Europe"
|
||||||
@ -97,17 +104,12 @@ final class InstancesManifest: Service, ObservableObject {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
var manifestURL: String {
|
var manifestURL: String? {
|
||||||
var url = Defaults[.instancesManifest]
|
Defaults[.instancesManifest].isEmpty ? nil : Defaults[.instancesManifest]
|
||||||
|
|
||||||
if url.isEmpty {
|
|
||||||
url = Self.builtinManifestUrl
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return url
|
var instancesList: Resource? {
|
||||||
}
|
guard let manifestURL else { return nil }
|
||||||
|
return resource(absoluteURL: manifestURL)
|
||||||
var instancesList: Resource {
|
|
||||||
resource(absoluteURL: manifestURL)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -146,7 +146,7 @@ final class PlayerModel: ObservableObject {
|
|||||||
|
|
||||||
var playerError: Error? { didSet {
|
var playerError: Error? { didSet {
|
||||||
if let error = playerError {
|
if let error = playerError {
|
||||||
navigation.presentAlert(title: "Failed loading video", message: error.localizedDescription)
|
navigation.presentAlert(title: "Failed loading video".localized(), message: error.localizedDescription)
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ extension Defaults.Keys {
|
|||||||
static let instancesManifest = Key<String>("instancesManifest", default: "")
|
static let instancesManifest = Key<String>("instancesManifest", default: "")
|
||||||
static let countryOfPublicInstances = Key<String?>("countryOfPublicInstances")
|
static let countryOfPublicInstances = Key<String?>("countryOfPublicInstances")
|
||||||
|
|
||||||
static let instances = Key<[Instance]>("instances", default: [])
|
static let instances = Key<[Instance]>("instances", default: [.init(app: .demoApp, name: "Demo", apiURL: "")])
|
||||||
static let accounts = Key<[Account]>("accounts", default: [])
|
static let accounts = Key<[Account]>("accounts", default: [])
|
||||||
static let lastAccountID = Key<Account.ID?>("lastAccountID")
|
static let lastAccountID = Key<Account.ID?>("lastAccountID")
|
||||||
static let lastInstanceID = Key<Instance.ID?>("lastInstanceID")
|
static let lastInstanceID = Key<Instance.ID?>("lastInstanceID")
|
||||||
@ -21,9 +21,7 @@ extension Defaults.Keys {
|
|||||||
|
|
||||||
static let enableReturnYouTubeDislike = Key<Bool>("enableReturnYouTubeDislike", default: false)
|
static let enableReturnYouTubeDislike = Key<Bool>("enableReturnYouTubeDislike", default: false)
|
||||||
|
|
||||||
static let favorites = Key<[FavoriteItem]>("favorites", default: [
|
static let favorites = Key<[FavoriteItem]>("favorites", default: [])
|
||||||
.init(section: .channel("UCE_M8A5yxnLfW0KghEeajjw", "Apple"))
|
|
||||||
])
|
|
||||||
|
|
||||||
#if !os(tvOS)
|
#if !os(tvOS)
|
||||||
#if os(macOS)
|
#if os(macOS)
|
||||||
|
@ -54,7 +54,7 @@ struct FavoriteItemView: View {
|
|||||||
#endif
|
#endif
|
||||||
.onAppear {
|
.onAppear {
|
||||||
resource?.addObserver(store)
|
resource?.addObserver(store)
|
||||||
resource?.load()
|
resource?.loadIfNeeded()
|
||||||
}
|
}
|
||||||
#if !os(tvOS)
|
#if !os(tvOS)
|
||||||
.onDrag {
|
.onDrag {
|
||||||
|
@ -8,6 +8,7 @@ struct AccountsNavigationLink: View {
|
|||||||
NavigationLink(instance.longDescription) {
|
NavigationLink(instance.longDescription) {
|
||||||
InstanceSettings(instance: instance)
|
InstanceSettings(instance: instance)
|
||||||
}
|
}
|
||||||
|
.disabled(instance.app == .demoApp)
|
||||||
.buttonStyle(.plain)
|
.buttonStyle(.plain)
|
||||||
.contextMenu {
|
.contextMenu {
|
||||||
removeInstanceButton(instance)
|
removeInstanceButton(instance)
|
||||||
|
@ -41,6 +41,7 @@ struct LocationsSettings: View {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@ViewBuilder var settings: some View {
|
@ViewBuilder var settings: some View {
|
||||||
|
if !InstancesManifest.shared.manifestURL.isNil {
|
||||||
Section(header: SettingsHeader(text: "Public Locations".localized()), footer: countryFooter) {
|
Section(header: SettingsHeader(text: "Public Locations".localized()), footer: countryFooter) {
|
||||||
Picker("Country", selection: $countryOfPublicInstances) {
|
Picker("Country", selection: $countryOfPublicInstances) {
|
||||||
Text("Don't use public locations").tag(String?.none)
|
Text("Don't use public locations").tag(String?.none)
|
||||||
@ -64,6 +65,7 @@ struct LocationsSettings: View {
|
|||||||
}
|
}
|
||||||
.disabled(countryOfPublicInstances.isNil)
|
.disabled(countryOfPublicInstances.isNil)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Section(header: SettingsHeader(text: "Custom Locations".localized())) {
|
Section(header: SettingsHeader(text: "Custom Locations".localized())) {
|
||||||
#if os(macOS)
|
#if os(macOS)
|
||||||
@ -92,7 +94,8 @@ struct LocationsSettings: View {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func loadCountries() {
|
func loadCountries() {
|
||||||
InstancesManifest.shared.instancesList.load()
|
InstancesManifest.shared.configure()
|
||||||
|
InstancesManifest.shared.instancesList?.load()
|
||||||
.onSuccess { response in
|
.onSuccess { response in
|
||||||
if let instances: [ManifestedInstance] = response.typedContent() {
|
if let instances: [ManifestedInstance] = response.typedContent() {
|
||||||
self.countries = instances.map(\.country).unique().sorted()
|
self.countries = instances.map(\.country).unique().sorted()
|
||||||
|
@ -31,6 +31,7 @@ struct PlayerSettings: View {
|
|||||||
@Default(.enableReturnYouTubeDislike) private var enableReturnYouTubeDislike
|
@Default(.enableReturnYouTubeDislike) private var enableReturnYouTubeDislike
|
||||||
@Default(.systemControlsCommands) private var systemControlsCommands
|
@Default(.systemControlsCommands) private var systemControlsCommands
|
||||||
|
|
||||||
|
@EnvironmentObject<AccountsModel> private var accounts
|
||||||
@EnvironmentObject<PlayerModel> private var player
|
@EnvironmentObject<PlayerModel> private var player
|
||||||
|
|
||||||
#if os(iOS)
|
#if os(iOS)
|
||||||
@ -99,8 +100,11 @@ struct PlayerSettings: View {
|
|||||||
|
|
||||||
keywordsToggle
|
keywordsToggle
|
||||||
showHistoryToggle
|
showHistoryToggle
|
||||||
|
|
||||||
|
if !accounts.isDemo {
|
||||||
returnYouTubeDislikeToggle
|
returnYouTubeDislikeToggle
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#if os(iOS)
|
#if os(iOS)
|
||||||
Section(header: SettingsHeader(text: "Orientation".localized())) {
|
Section(header: SettingsHeader(text: "Orientation".localized())) {
|
||||||
|
@ -15,7 +15,9 @@ struct ShareButton: View {
|
|||||||
Menu {
|
Menu {
|
||||||
instanceActions
|
instanceActions
|
||||||
Divider()
|
Divider()
|
||||||
|
if !accounts.isDemo {
|
||||||
youtubeActions
|
youtubeActions
|
||||||
|
}
|
||||||
} label: {
|
} label: {
|
||||||
Label("Share...", systemImage: "square.and.arrow.up")
|
Label("Share...", systemImage: "square.and.arrow.up")
|
||||||
}
|
}
|
||||||
|
@ -67,7 +67,7 @@ struct WelcomeScreen: View {
|
|||||||
.foregroundColor(.secondary)
|
.foregroundColor(.secondary)
|
||||||
}
|
}
|
||||||
.onAppear {
|
.onAppear {
|
||||||
resource.load().onSuccess { response in
|
resource?.load().onSuccess { response in
|
||||||
if let instances: [ManifestedInstance] = response.typedContent() {
|
if let instances: [ManifestedInstance] = response.typedContent() {
|
||||||
store = instances
|
store = instances
|
||||||
}
|
}
|
||||||
@ -85,7 +85,7 @@ struct WelcomeScreen: View {
|
|||||||
store.first { $0.country == country }?.flag
|
store.first { $0.country == country }?.flag
|
||||||
}
|
}
|
||||||
|
|
||||||
var resource: Resource {
|
var resource: Resource? {
|
||||||
InstancesManifest.shared.instancesList
|
InstancesManifest.shared.instancesList
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -174,13 +174,8 @@ struct YatteeApp: App {
|
|||||||
accounts.configureAccount()
|
accounts.configureAccount()
|
||||||
}
|
}
|
||||||
|
|
||||||
let countryOfPublicInstances = Defaults[.countryOfPublicInstances]
|
if let countryOfPublicInstances = Defaults[.countryOfPublicInstances] {
|
||||||
if accounts.current.isNil, countryOfPublicInstances.isNil {
|
InstancesManifest.shared.setPublicAccount(countryOfPublicInstances, accounts: accounts, asCurrent: accounts.current.isNil)
|
||||||
navigation.presentingWelcomeScreen = true
|
|
||||||
}
|
|
||||||
|
|
||||||
if !countryOfPublicInstances.isNil {
|
|
||||||
InstancesManifest.shared.setPublicAccount(countryOfPublicInstances!, accounts: accounts, asCurrent: accounts.current.isNil)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
playlists.accounts = accounts
|
playlists.accounts = accounts
|
||||||
|
@ -434,6 +434,9 @@
|
|||||||
376CD21726FBE18D001E1AC1 /* Instance+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 376CD21526FBE18D001E1AC1 /* Instance+Fixtures.swift */; };
|
376CD21726FBE18D001E1AC1 /* Instance+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 376CD21526FBE18D001E1AC1 /* Instance+Fixtures.swift */; };
|
||||||
376CD21826FBE18D001E1AC1 /* Instance+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 376CD21526FBE18D001E1AC1 /* Instance+Fixtures.swift */; };
|
376CD21826FBE18D001E1AC1 /* Instance+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 376CD21526FBE18D001E1AC1 /* Instance+Fixtures.swift */; };
|
||||||
376E331228AD3B320070E30C /* ScrollDismissesKeyboard+Backport.swift in Sources */ = {isa = PBXBuildFile; fileRef = 376E331128AD3B320070E30C /* ScrollDismissesKeyboard+Backport.swift */; };
|
376E331228AD3B320070E30C /* ScrollDismissesKeyboard+Backport.swift in Sources */ = {isa = PBXBuildFile; fileRef = 376E331128AD3B320070E30C /* ScrollDismissesKeyboard+Backport.swift */; };
|
||||||
|
3771429829087DFC00306CEA /* DemoAppAPI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3771429729087DFC00306CEA /* DemoAppAPI.swift */; };
|
||||||
|
3771429929087DFC00306CEA /* DemoAppAPI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3771429729087DFC00306CEA /* DemoAppAPI.swift */; };
|
||||||
|
3771429A29087DFC00306CEA /* DemoAppAPI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3771429729087DFC00306CEA /* DemoAppAPI.swift */; };
|
||||||
3772003827E8EEB100CB2475 /* AudioToolbox.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3772003227E8EEA100CB2475 /* AudioToolbox.framework */; };
|
3772003827E8EEB100CB2475 /* AudioToolbox.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3772003227E8EEA100CB2475 /* AudioToolbox.framework */; };
|
||||||
3772003927E8EEB700CB2475 /* AVFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3772003427E8EEA100CB2475 /* AVFoundation.framework */; };
|
3772003927E8EEB700CB2475 /* AVFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3772003427E8EEA100CB2475 /* AVFoundation.framework */; };
|
||||||
3772003A27E8EEBE00CB2475 /* CoreMedia.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3772003527E8EEA100CB2475 /* CoreMedia.framework */; };
|
3772003A27E8EEBE00CB2475 /* CoreMedia.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3772003527E8EEA100CB2475 /* CoreMedia.framework */; };
|
||||||
@ -1151,6 +1154,7 @@
|
|||||||
376E331128AD3B320070E30C /* ScrollDismissesKeyboard+Backport.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ScrollDismissesKeyboard+Backport.swift"; sourceTree = "<group>"; };
|
376E331128AD3B320070E30C /* ScrollDismissesKeyboard+Backport.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ScrollDismissesKeyboard+Backport.swift"; sourceTree = "<group>"; };
|
||||||
3771429529087BE100306CEA /* it */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = it; path = it.lproj/Localizable.strings; sourceTree = "<group>"; };
|
3771429529087BE100306CEA /* it */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = it; path = it.lproj/Localizable.strings; sourceTree = "<group>"; };
|
||||||
3771429629087BF000306CEA /* az */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = az; path = az.lproj/Localizable.strings; sourceTree = "<group>"; };
|
3771429629087BF000306CEA /* az */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = az; path = az.lproj/Localizable.strings; sourceTree = "<group>"; };
|
||||||
|
3771429729087DFC00306CEA /* DemoAppAPI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DemoAppAPI.swift; sourceTree = "<group>"; };
|
||||||
3772002527E8ED2600CB2475 /* BridgingHeader.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = BridgingHeader.h; sourceTree = "<group>"; };
|
3772002527E8ED2600CB2475 /* BridgingHeader.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = BridgingHeader.h; sourceTree = "<group>"; };
|
||||||
3772003127E8EEA100CB2475 /* libz.tbd */ = {isa = PBXFileReference; lastKnownFileType = "sourcecode.text-based-dylib-definition"; name = libz.tbd; path = Platforms/AppleTVOS.platform/Developer/SDKs/AppleTVOS15.4.sdk/usr/lib/libz.tbd; sourceTree = DEVELOPER_DIR; };
|
3772003127E8EEA100CB2475 /* libz.tbd */ = {isa = PBXFileReference; lastKnownFileType = "sourcecode.text-based-dylib-definition"; name = libz.tbd; path = Platforms/AppleTVOS.platform/Developer/SDKs/AppleTVOS15.4.sdk/usr/lib/libz.tbd; sourceTree = DEVELOPER_DIR; };
|
||||||
3772003227E8EEA100CB2475 /* AudioToolbox.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = AudioToolbox.framework; path = Platforms/AppleTVOS.platform/Developer/SDKs/AppleTVOS15.4.sdk/System/Library/Frameworks/AudioToolbox.framework; sourceTree = DEVELOPER_DIR; };
|
3772003227E8EEA100CB2475 /* AudioToolbox.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = AudioToolbox.framework; path = Platforms/AppleTVOS.platform/Developer/SDKs/AppleTVOS15.4.sdk/System/Library/Frameworks/AudioToolbox.framework; sourceTree = DEVELOPER_DIR; };
|
||||||
@ -1739,6 +1743,7 @@
|
|||||||
3700155A271B0D4D0049C794 /* PipedAPI.swift */,
|
3700155A271B0D4D0049C794 /* PipedAPI.swift */,
|
||||||
37D526DD2720AC4400ED2F5E /* VideosAPI.swift */,
|
37D526DD2720AC4400ED2F5E /* VideosAPI.swift */,
|
||||||
376A33DF2720CAD6000C1D6B /* VideosApp.swift */,
|
376A33DF2720CAD6000C1D6B /* VideosApp.swift */,
|
||||||
|
3771429729087DFC00306CEA /* DemoAppAPI.swift */,
|
||||||
);
|
);
|
||||||
path = Applications;
|
path = Applications;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
@ -2884,6 +2889,7 @@
|
|||||||
3711403F26B206A6005B3555 /* SearchModel.swift in Sources */,
|
3711403F26B206A6005B3555 /* SearchModel.swift in Sources */,
|
||||||
3729037E2739E47400EA99F6 /* MenuCommands.swift in Sources */,
|
3729037E2739E47400EA99F6 /* MenuCommands.swift in Sources */,
|
||||||
37F64FE426FE70A60081B69E /* RedrawOnModifier.swift in Sources */,
|
37F64FE426FE70A60081B69E /* RedrawOnModifier.swift in Sources */,
|
||||||
|
3771429829087DFC00306CEA /* DemoAppAPI.swift in Sources */,
|
||||||
37EBD8C427AF0DA800F1C24B /* PlayerBackend.swift in Sources */,
|
37EBD8C427AF0DA800F1C24B /* PlayerBackend.swift in Sources */,
|
||||||
376A33E02720CAD6000C1D6B /* VideosApp.swift in Sources */,
|
376A33E02720CAD6000C1D6B /* VideosApp.swift in Sources */,
|
||||||
374AB3DB28BCAF7E00DF56FB /* SeekType.swift in Sources */,
|
374AB3DB28BCAF7E00DF56FB /* SeekType.swift in Sources */,
|
||||||
@ -3274,6 +3280,7 @@
|
|||||||
373CFAF02697A78B003CB2C6 /* AddToPlaylistView.swift in Sources */,
|
373CFAF02697A78B003CB2C6 /* AddToPlaylistView.swift in Sources */,
|
||||||
3763495226DFF59D00B9A393 /* AppSidebarRecents.swift in Sources */,
|
3763495226DFF59D00B9A393 /* AppSidebarRecents.swift in Sources */,
|
||||||
370015AA28BBAE7F000149FD /* ProgressBar.swift in Sources */,
|
370015AA28BBAE7F000149FD /* ProgressBar.swift in Sources */,
|
||||||
|
3771429929087DFC00306CEA /* DemoAppAPI.swift in Sources */,
|
||||||
371B7E672759786B00D21217 /* Comment+Fixtures.swift in Sources */,
|
371B7E672759786B00D21217 /* Comment+Fixtures.swift in Sources */,
|
||||||
3769C02F2779F18600DDB3EA /* PlaceholderProgressView.swift in Sources */,
|
3769C02F2779F18600DDB3EA /* PlaceholderProgressView.swift in Sources */,
|
||||||
37BA794426DBA973002A0235 /* PlaylistsModel.swift in Sources */,
|
37BA794426DBA973002A0235 /* PlaylistsModel.swift in Sources */,
|
||||||
@ -3418,6 +3425,7 @@
|
|||||||
3752069B285E8DD300CA655F /* Chapter.swift in Sources */,
|
3752069B285E8DD300CA655F /* Chapter.swift in Sources */,
|
||||||
37B044B926F7AB9000E1419D /* SettingsView.swift in Sources */,
|
37B044B926F7AB9000E1419D /* SettingsView.swift in Sources */,
|
||||||
3743B86A27216D3600261544 /* ChannelCell.swift in Sources */,
|
3743B86A27216D3600261544 /* ChannelCell.swift in Sources */,
|
||||||
|
3771429A29087DFC00306CEA /* DemoAppAPI.swift in Sources */,
|
||||||
37E80F47287B7B9400561799 /* VideoDetailsOverlay.swift in Sources */,
|
37E80F47287B7B9400561799 /* VideoDetailsOverlay.swift in Sources */,
|
||||||
37E80F44287B7AB400561799 /* VideoDetails.swift in Sources */,
|
37E80F44287B7AB400561799 /* VideoDetails.swift in Sources */,
|
||||||
3751BA8527E6914F007B1A60 /* ReturnYouTubeDislikeAPI.swift in Sources */,
|
3751BA8527E6914F007B1A60 /* ReturnYouTubeDislikeAPI.swift in Sources */,
|
||||||
|
Loading…
Reference in New Issue
Block a user