mirror of
https://github.com/yattee/yattee.git
synced 2024-12-22 21:43:41 +00:00
Channel playlists support
This commit is contained in:
parent
4307da57c5
commit
734bb31260
12
Fixtures/ChannelPlaylist+Fixtures.swift
Normal file
12
Fixtures/ChannelPlaylist+Fixtures.swift
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import Foundation
|
||||||
|
|
||||||
|
extension ChannelPlaylist {
|
||||||
|
static var fixture: ChannelPlaylist {
|
||||||
|
ChannelPlaylist(
|
||||||
|
title: "Playlist with a very long title that will not fit easily in the screen",
|
||||||
|
thumbnailURL: URL(string: "https://i.ytimg.com/vi/hT_nvWreIhg/hqdefault.jpg?sqp=-oaymwEWCKgBEF5IWvKriqkDCQgBFQAAiEIYAQ==&rs=AOn4CLAAD21_-Bo6Td1z3cV-UFyoi1flEg")!,
|
||||||
|
channel: Video.fixture.channel,
|
||||||
|
videos: Video.allFixtures
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
@ -95,8 +95,7 @@ final class InvidiousAPI: Service, ObservableObject, VideosAPI {
|
|||||||
if type == "channel" {
|
if type == "channel" {
|
||||||
return ContentItem(channel: InvidiousAPI.extractChannel(from: $0))
|
return ContentItem(channel: InvidiousAPI.extractChannel(from: $0))
|
||||||
} else if type == "playlist" {
|
} else if type == "playlist" {
|
||||||
// TODO: fix playlists
|
return ContentItem(playlist: InvidiousAPI.extractChannelPlaylist(from: $0))
|
||||||
return ContentItem(playlist: Playlist(JSON(parseJSON: "{}")))
|
|
||||||
}
|
}
|
||||||
return ContentItem(video: InvidiousAPI.extractVideo($0))
|
return ContentItem(video: InvidiousAPI.extractVideo($0))
|
||||||
}
|
}
|
||||||
@ -143,6 +142,10 @@ final class InvidiousAPI: Service, ObservableObject, VideosAPI {
|
|||||||
content.json.arrayValue.map(InvidiousAPI.extractVideo)
|
content.json.arrayValue.map(InvidiousAPI.extractVideo)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
configureTransformer(pathPattern("playlists/*"), requestMethods: [.get]) { (content: Entity<JSON>) -> ChannelPlaylist in
|
||||||
|
InvidiousAPI.extractChannelPlaylist(from: content.json)
|
||||||
|
}
|
||||||
|
|
||||||
configureTransformer(pathPattern("videos/*"), requestMethods: [.get]) { (content: Entity<JSON>) -> Video in
|
configureTransformer(pathPattern("videos/*"), requestMethods: [.get]) { (content: Entity<JSON>) -> Video in
|
||||||
InvidiousAPI.extractVideo(content.json)
|
InvidiousAPI.extractVideo(content.json)
|
||||||
}
|
}
|
||||||
@ -214,6 +217,10 @@ final class InvidiousAPI: Service, ObservableObject, VideosAPI {
|
|||||||
playlist(playlistID)?.child("videos").child(videoID)
|
playlist(playlistID)?.child("videos").child(videoID)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func channelPlaylist(_ id: String) -> Resource? {
|
||||||
|
resource(baseURL: account.url, path: basePathAppending("playlists/\(id)"))
|
||||||
|
}
|
||||||
|
|
||||||
func search(_ query: SearchQuery) -> Resource {
|
func search(_ query: SearchQuery) -> Resource {
|
||||||
var resource = resource(baseURL: account.url, path: basePathAppending("search"))
|
var resource = resource(baseURL: account.url, path: basePathAppending("search"))
|
||||||
.withParam("q", searchQuery(query.query))
|
.withParam("q", searchQuery(query.query))
|
||||||
@ -325,6 +332,21 @@ final class InvidiousAPI: Service, ObservableObject, VideosAPI {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static func extractChannelPlaylist(from json: JSON) -> ChannelPlaylist {
|
||||||
|
let details = json.dictionaryValue
|
||||||
|
return ChannelPlaylist(
|
||||||
|
id: details["playlistId"]!.stringValue,
|
||||||
|
title: details["title"]!.stringValue,
|
||||||
|
thumbnailURL: details["playlistThumbnail"]?.url,
|
||||||
|
channel: extractChannel(from: json),
|
||||||
|
videos: details["videos"]?.arrayValue.compactMap(InvidiousAPI.extractVideo) ?? []
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
static func extractChannelPlaylists(from json: JSON) -> [ChannelPlaylist] {
|
||||||
|
json.arrayValue.map(InvidiousAPI.extractChannelPlaylist)
|
||||||
|
}
|
||||||
|
|
||||||
private static func extractThumbnails(from details: JSON) -> [Thumbnail] {
|
private static func extractThumbnails(from details: JSON) -> [Thumbnail] {
|
||||||
details["videoThumbnails"].arrayValue.map { json in
|
details["videoThumbnails"].arrayValue.map { json in
|
||||||
Thumbnail(url: json["url"].url!, quality: .init(rawValue: json["quality"].string!)!)
|
Thumbnail(url: json["url"].url!, quality: .init(rawValue: json["quality"].string!)!)
|
||||||
|
@ -35,6 +35,10 @@ final class PipedAPI: Service, ObservableObject, VideosAPI {
|
|||||||
PipedAPI.extractChannel(content.json)
|
PipedAPI.extractChannel(content.json)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
configureTransformer(pathPattern("playlists/*")) { (content: Entity<JSON>) -> ChannelPlaylist? in
|
||||||
|
PipedAPI.extractChannelPlaylist(from: content.json)
|
||||||
|
}
|
||||||
|
|
||||||
configureTransformer(pathPattern("streams/*")) { (content: Entity<JSON>) -> Video? in
|
configureTransformer(pathPattern("streams/*")) { (content: Entity<JSON>) -> Video? in
|
||||||
PipedAPI.extractVideo(content.json)
|
PipedAPI.extractVideo(content.json)
|
||||||
}
|
}
|
||||||
@ -56,6 +60,10 @@ final class PipedAPI: Service, ObservableObject, VideosAPI {
|
|||||||
resource(baseURL: account.url, path: "channel/\(id)")
|
resource(baseURL: account.url, path: "channel/\(id)")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func channelPlaylist(_ id: String) -> Resource? {
|
||||||
|
resource(baseURL: account.url, path: "playlists/\(id)")
|
||||||
|
}
|
||||||
|
|
||||||
func trending(country: Country, category _: TrendingCategory? = nil) -> Resource {
|
func trending(country: Country, category _: TrendingCategory? = nil) -> Resource {
|
||||||
resource(baseURL: account.instance.url, path: "trending")
|
resource(baseURL: account.instance.url, path: "trending")
|
||||||
.withParam("region", country.rawValue)
|
.withParam("region", country.rawValue)
|
||||||
@ -118,7 +126,9 @@ final class PipedAPI: Service, ObservableObject, VideosAPI {
|
|||||||
}
|
}
|
||||||
|
|
||||||
case .playlist:
|
case .playlist:
|
||||||
return nil
|
if let playlist = PipedAPI.extractChannelPlaylist(from: content) {
|
||||||
|
return ContentItem(playlist: playlist)
|
||||||
|
}
|
||||||
|
|
||||||
case .channel:
|
case .channel:
|
||||||
if let channel = PipedAPI.extractChannel(content) {
|
if let channel = PipedAPI.extractChannel(content) {
|
||||||
@ -136,7 +146,7 @@ final class PipedAPI: Service, ObservableObject, VideosAPI {
|
|||||||
private static func extractChannel(_ content: JSON) -> Channel? {
|
private static func extractChannel(_ content: JSON) -> Channel? {
|
||||||
let attributes = content.dictionaryValue
|
let attributes = content.dictionaryValue
|
||||||
guard let id = attributes["id"]?.stringValue ??
|
guard let id = attributes["id"]?.stringValue ??
|
||||||
attributes["url"]?.stringValue.components(separatedBy: "/").last
|
(attributes["url"] ?? attributes["uploaderUrl"])?.stringValue.components(separatedBy: "/").last
|
||||||
else {
|
else {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -157,6 +167,28 @@ final class PipedAPI: Service, ObservableObject, VideosAPI {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static func extractChannelPlaylist(from json: JSON) -> ChannelPlaylist? {
|
||||||
|
let details = json.dictionaryValue
|
||||||
|
let id = details["url"]?.stringValue.components(separatedBy: "?list=").last ?? UUID().uuidString
|
||||||
|
let thumbnailURL = details["thumbnail"]?.url ?? details["thumbnailUrl"]?.url
|
||||||
|
var videos = [Video]()
|
||||||
|
if let relatedStreams = details["relatedStreams"] {
|
||||||
|
videos = PipedAPI.extractVideos(relatedStreams)
|
||||||
|
}
|
||||||
|
return ChannelPlaylist(
|
||||||
|
id: id,
|
||||||
|
title: details["name"]!.stringValue,
|
||||||
|
thumbnailURL: thumbnailURL,
|
||||||
|
channel: extractChannel(json)!,
|
||||||
|
videos: videos,
|
||||||
|
videosCount: details["videos"]?.int
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
static func extractChannelPlaylists(from json: JSON) -> [ChannelPlaylist] {
|
||||||
|
json.arrayValue.compactMap(PipedAPI.extractChannelPlaylist)
|
||||||
|
}
|
||||||
|
|
||||||
private static func extractVideo(_ content: JSON) -> Video? {
|
private static func extractVideo(_ content: JSON) -> Video? {
|
||||||
let details = content.dictionaryValue
|
let details = content.dictionaryValue
|
||||||
let url = details["url"]?.string
|
let url = details["url"]?.string
|
||||||
|
@ -21,4 +21,6 @@ protocol VideosAPI {
|
|||||||
|
|
||||||
func playlistVideo(_ playlistID: String, _ videoID: String) -> Resource?
|
func playlistVideo(_ playlistID: String, _ videoID: String) -> Resource?
|
||||||
func playlistVideos(_ id: String) -> Resource?
|
func playlistVideos(_ id: String) -> Resource?
|
||||||
|
|
||||||
|
func channelPlaylist(_ id: String) -> Resource?
|
||||||
}
|
}
|
||||||
|
10
Model/ChannelPlaylist.swift
Normal file
10
Model/ChannelPlaylist.swift
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
import Foundation
|
||||||
|
|
||||||
|
struct ChannelPlaylist: Identifiable {
|
||||||
|
var id: String = UUID().uuidString
|
||||||
|
var title: String
|
||||||
|
var thumbnailURL: URL?
|
||||||
|
var channel: Channel?
|
||||||
|
var videos = [Video]()
|
||||||
|
var videosCount: Int?
|
||||||
|
}
|
@ -8,7 +8,7 @@ struct ContentItem: Identifiable {
|
|||||||
switch self {
|
switch self {
|
||||||
case .channel:
|
case .channel:
|
||||||
return 1
|
return 1
|
||||||
case .video:
|
case .playlist:
|
||||||
return 2
|
return 2
|
||||||
default:
|
default:
|
||||||
return 3
|
return 3
|
||||||
@ -21,7 +21,7 @@ struct ContentItem: Identifiable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var video: Video!
|
var video: Video!
|
||||||
var playlist: Playlist!
|
var playlist: ChannelPlaylist!
|
||||||
var channel: Channel!
|
var channel: Channel!
|
||||||
|
|
||||||
static func array(of videos: [Video]) -> [ContentItem] {
|
static func array(of videos: [Video]) -> [ContentItem] {
|
||||||
|
@ -26,7 +26,8 @@ final class NavigationModel: ObservableObject {
|
|||||||
@Published var presentingUnsubscribeAlert = false
|
@Published var presentingUnsubscribeAlert = false
|
||||||
@Published var channelToUnsubscribe: Channel!
|
@Published var channelToUnsubscribe: Channel!
|
||||||
|
|
||||||
@Published var isChannelOpen = false
|
@Published var presentingChannel = false
|
||||||
|
@Published var presentingPlaylist = false
|
||||||
@Published var sidebarSectionChanged = false
|
@Published var sidebarSectionChanged = false
|
||||||
|
|
||||||
@Published var presentingSettings = false
|
@Published var presentingSettings = false
|
||||||
|
@ -27,17 +27,16 @@ final class PlayerModel: ObservableObject {
|
|||||||
|
|
||||||
@Published var queue = [PlayerQueueItem]()
|
@Published var queue = [PlayerQueueItem]()
|
||||||
@Published var currentItem: PlayerQueueItem!
|
@Published var currentItem: PlayerQueueItem!
|
||||||
@Published var live = false
|
|
||||||
|
|
||||||
@Published var history = [PlayerQueueItem]()
|
@Published var history = [PlayerQueueItem]()
|
||||||
|
|
||||||
@Published var savedTime: CMTime?
|
@Published var savedTime: CMTime?
|
||||||
|
|
||||||
@Published var composition = AVMutableComposition()
|
@Published var playerNavigationLinkActive = false
|
||||||
|
|
||||||
var accounts: AccountsModel
|
var accounts: AccountsModel
|
||||||
var instances: InstancesModel
|
var instances: InstancesModel
|
||||||
|
|
||||||
|
var composition = AVMutableComposition()
|
||||||
var timeObserver: Any?
|
var timeObserver: Any?
|
||||||
private var shouldResumePlaying = true
|
private var shouldResumePlaying = true
|
||||||
private var statusObservation: NSKeyValueObservation?
|
private var statusObservation: NSKeyValueObservation?
|
||||||
@ -61,6 +60,10 @@ final class PlayerModel: ObservableObject {
|
|||||||
currentItem?.playbackTime
|
currentItem?.playbackTime
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var live: Bool {
|
||||||
|
currentItem?.video.live ?? false
|
||||||
|
}
|
||||||
|
|
||||||
var playerItemDuration: CMTime? {
|
var playerItemDuration: CMTime? {
|
||||||
player.currentItem?.asset.duration
|
player.currentItem?.asset.duration
|
||||||
}
|
}
|
||||||
@ -335,7 +338,6 @@ final class PlayerModel: ObservableObject {
|
|||||||
|
|
||||||
timeObserver = player.addPeriodicTimeObserver(forInterval: interval, queue: .main) { _ in
|
timeObserver = player.addPeriodicTimeObserver(forInterval: interval, queue: .main) { _ in
|
||||||
self.currentRate = self.player.rate
|
self.currentRate = self.player.rate
|
||||||
self.live = self.currentVideo?.live ?? false
|
|
||||||
self.currentItem?.playbackTime = self.player.currentTime()
|
self.currentItem?.playbackTime = self.player.currentTime()
|
||||||
self.currentItem?.videoDuration = self.player.currentItem?.asset.duration.seconds
|
self.currentItem?.videoDuration = self.player.currentItem?.asset.duration.seconds
|
||||||
}
|
}
|
||||||
|
@ -39,13 +39,21 @@ final class RecentsModel: ObservableObject {
|
|||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var presentedPlaylist: ChannelPlaylist? {
|
||||||
|
if let recent = items.last(where: { $0.type == .playlist }) {
|
||||||
|
return recent.playlist
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct RecentItem: Defaults.Serializable, Identifiable {
|
struct RecentItem: Defaults.Serializable, Identifiable {
|
||||||
static var bridge = RecentItemBridge()
|
static var bridge = RecentItemBridge()
|
||||||
|
|
||||||
enum ItemType: String {
|
enum ItemType: String {
|
||||||
case channel, query
|
case channel, playlist, query
|
||||||
}
|
}
|
||||||
|
|
||||||
var type: ItemType
|
var type: ItemType
|
||||||
@ -72,6 +80,14 @@ struct RecentItem: Defaults.Serializable, Identifiable {
|
|||||||
return Channel(id: id, name: title)
|
return Channel(id: id, name: title)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var playlist: ChannelPlaylist? {
|
||||||
|
guard type == .playlist else {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return ChannelPlaylist(id: id, title: title)
|
||||||
|
}
|
||||||
|
|
||||||
init(type: ItemType, identifier: String, title: String) {
|
init(type: ItemType, identifier: String, title: String) {
|
||||||
self.type = type
|
self.type = type
|
||||||
id = identifier
|
id = identifier
|
||||||
@ -89,6 +105,12 @@ struct RecentItem: Defaults.Serializable, Identifiable {
|
|||||||
id = query
|
id = query
|
||||||
title = query
|
title = query
|
||||||
}
|
}
|
||||||
|
|
||||||
|
init(from playlist: ChannelPlaylist) {
|
||||||
|
type = .playlist
|
||||||
|
id = playlist.id
|
||||||
|
title = playlist.title
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct RecentItemBridge: Defaults.Bridge {
|
struct RecentItemBridge: Defaults.Bridge {
|
||||||
|
@ -268,6 +268,18 @@
|
|||||||
37C3A241272359900087A57A /* Double+Format.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A240272359900087A57A /* Double+Format.swift */; };
|
37C3A241272359900087A57A /* Double+Format.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A240272359900087A57A /* Double+Format.swift */; };
|
||||||
37C3A242272359900087A57A /* Double+Format.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A240272359900087A57A /* Double+Format.swift */; };
|
37C3A242272359900087A57A /* Double+Format.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A240272359900087A57A /* Double+Format.swift */; };
|
||||||
37C3A243272359900087A57A /* Double+Format.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A240272359900087A57A /* Double+Format.swift */; };
|
37C3A243272359900087A57A /* Double+Format.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A240272359900087A57A /* Double+Format.swift */; };
|
||||||
|
37C3A24527235DA70087A57A /* ChannelPlaylist.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24427235DA70087A57A /* ChannelPlaylist.swift */; };
|
||||||
|
37C3A24627235DA70087A57A /* ChannelPlaylist.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24427235DA70087A57A /* ChannelPlaylist.swift */; };
|
||||||
|
37C3A24727235DA70087A57A /* ChannelPlaylist.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24427235DA70087A57A /* ChannelPlaylist.swift */; };
|
||||||
|
37C3A24927235FAA0087A57A /* ChannelPlaylistCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24827235FAA0087A57A /* ChannelPlaylistCell.swift */; };
|
||||||
|
37C3A24A27235FAA0087A57A /* ChannelPlaylistCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24827235FAA0087A57A /* ChannelPlaylistCell.swift */; };
|
||||||
|
37C3A24B27235FAA0087A57A /* ChannelPlaylistCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24827235FAA0087A57A /* ChannelPlaylistCell.swift */; };
|
||||||
|
37C3A24D272360470087A57A /* ChannelPlaylist+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24C272360470087A57A /* ChannelPlaylist+Fixtures.swift */; };
|
||||||
|
37C3A24E272360470087A57A /* ChannelPlaylist+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24C272360470087A57A /* ChannelPlaylist+Fixtures.swift */; };
|
||||||
|
37C3A24F272360470087A57A /* ChannelPlaylist+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A24C272360470087A57A /* ChannelPlaylist+Fixtures.swift */; };
|
||||||
|
37C3A251272366440087A57A /* ChannelPlaylistView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A250272366440087A57A /* ChannelPlaylistView.swift */; };
|
||||||
|
37C3A252272366440087A57A /* ChannelPlaylistView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A250272366440087A57A /* ChannelPlaylistView.swift */; };
|
||||||
|
37C3A253272366440087A57A /* ChannelPlaylistView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C3A250272366440087A57A /* ChannelPlaylistView.swift */; };
|
||||||
37C7A1D5267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C7A1D4267BFD9D0010EAD6 /* SponsorBlockSegment.swift */; };
|
37C7A1D5267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C7A1D4267BFD9D0010EAD6 /* SponsorBlockSegment.swift */; };
|
||||||
37C7A1D6267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C7A1D4267BFD9D0010EAD6 /* SponsorBlockSegment.swift */; };
|
37C7A1D6267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C7A1D4267BFD9D0010EAD6 /* SponsorBlockSegment.swift */; };
|
||||||
37C7A1D7267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C7A1D4267BFD9D0010EAD6 /* SponsorBlockSegment.swift */; };
|
37C7A1D7267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37C7A1D4267BFD9D0010EAD6 /* SponsorBlockSegment.swift */; };
|
||||||
@ -477,6 +489,10 @@
|
|||||||
37BE0BDB26A2367F0092E2DB /* Player.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Player.swift; sourceTree = "<group>"; };
|
37BE0BDB26A2367F0092E2DB /* Player.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Player.swift; sourceTree = "<group>"; };
|
||||||
37C194C626F6A9C8005D3B96 /* RecentsModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RecentsModel.swift; sourceTree = "<group>"; };
|
37C194C626F6A9C8005D3B96 /* RecentsModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RecentsModel.swift; sourceTree = "<group>"; };
|
||||||
37C3A240272359900087A57A /* Double+Format.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Double+Format.swift"; sourceTree = "<group>"; };
|
37C3A240272359900087A57A /* Double+Format.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Double+Format.swift"; sourceTree = "<group>"; };
|
||||||
|
37C3A24427235DA70087A57A /* ChannelPlaylist.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ChannelPlaylist.swift; sourceTree = "<group>"; };
|
||||||
|
37C3A24827235FAA0087A57A /* ChannelPlaylistCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ChannelPlaylistCell.swift; sourceTree = "<group>"; };
|
||||||
|
37C3A24C272360470087A57A /* ChannelPlaylist+Fixtures.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ChannelPlaylist+Fixtures.swift"; sourceTree = "<group>"; };
|
||||||
|
37C3A250272366440087A57A /* ChannelPlaylistView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ChannelPlaylistView.swift; sourceTree = "<group>"; };
|
||||||
37C7A1D4267BFD9D0010EAD6 /* SponsorBlockSegment.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SponsorBlockSegment.swift; sourceTree = "<group>"; };
|
37C7A1D4267BFD9D0010EAD6 /* SponsorBlockSegment.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SponsorBlockSegment.swift; sourceTree = "<group>"; };
|
||||||
37CC3F44270CE30600608308 /* PlayerQueueItem.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerQueueItem.swift; sourceTree = "<group>"; };
|
37CC3F44270CE30600608308 /* PlayerQueueItem.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerQueueItem.swift; sourceTree = "<group>"; };
|
||||||
37CC3F4B270CFE1700608308 /* PlayerQueueView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerQueueView.swift; sourceTree = "<group>"; };
|
37CC3F4B270CFE1700608308 /* PlayerQueueView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerQueueView.swift; sourceTree = "<group>"; };
|
||||||
@ -660,6 +676,8 @@
|
|||||||
isa = PBXGroup;
|
isa = PBXGroup;
|
||||||
children = (
|
children = (
|
||||||
3743B86727216D3600261544 /* ChannelCell.swift */,
|
3743B86727216D3600261544 /* ChannelCell.swift */,
|
||||||
|
37C3A24827235FAA0087A57A /* ChannelPlaylistCell.swift */,
|
||||||
|
37C3A250272366440087A57A /* ChannelPlaylistView.swift */,
|
||||||
37BA793E26DB8F97002A0235 /* ChannelVideosView.swift */,
|
37BA793E26DB8F97002A0235 /* ChannelVideosView.swift */,
|
||||||
37FB285D272225E800A57617 /* ContentItemView.swift */,
|
37FB285D272225E800A57617 /* ContentItemView.swift */,
|
||||||
3748186D26A769D60084E870 /* DetailBadge.swift */,
|
3748186D26A769D60084E870 /* DetailBadge.swift */,
|
||||||
@ -715,6 +733,7 @@
|
|||||||
3748186426A762300084E870 /* Fixtures */ = {
|
3748186426A762300084E870 /* Fixtures */ = {
|
||||||
isa = PBXGroup;
|
isa = PBXGroup;
|
||||||
children = (
|
children = (
|
||||||
|
37C3A24C272360470087A57A /* ChannelPlaylist+Fixtures.swift */,
|
||||||
376CD21526FBE18D001E1AC1 /* Instance+Fixtures.swift */,
|
376CD21526FBE18D001E1AC1 /* Instance+Fixtures.swift */,
|
||||||
37F49BA226CAA59B00304AC0 /* Playlist+Fixtures.swift */,
|
37F49BA226CAA59B00304AC0 /* Playlist+Fixtures.swift */,
|
||||||
3748186926A764FB0084E870 /* Thumbnail+Fixtures.swift */,
|
3748186926A764FB0084E870 /* Thumbnail+Fixtures.swift */,
|
||||||
@ -927,6 +946,7 @@
|
|||||||
373CFADA269663F1003CB2C6 /* Thumbnail.swift */,
|
373CFADA269663F1003CB2C6 /* Thumbnail.swift */,
|
||||||
3705B181267B4E4900704544 /* TrendingCategory.swift */,
|
3705B181267B4E4900704544 /* TrendingCategory.swift */,
|
||||||
37D4B19626717E1500C925CA /* Video.swift */,
|
37D4B19626717E1500C925CA /* Video.swift */,
|
||||||
|
37C3A24427235DA70087A57A /* ChannelPlaylist.swift */,
|
||||||
);
|
);
|
||||||
path = Model;
|
path = Model;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
@ -1400,6 +1420,7 @@
|
|||||||
3700155B271B0D4D0049C794 /* PipedAPI.swift in Sources */,
|
3700155B271B0D4D0049C794 /* PipedAPI.swift in Sources */,
|
||||||
37B044B726F7AB9000E1419D /* SettingsView.swift in Sources */,
|
37B044B726F7AB9000E1419D /* SettingsView.swift in Sources */,
|
||||||
377FC7E3267A084A00A6BBAF /* VideoCell.swift in Sources */,
|
377FC7E3267A084A00A6BBAF /* VideoCell.swift in Sources */,
|
||||||
|
37C3A251272366440087A57A /* ChannelPlaylistView.swift in Sources */,
|
||||||
37E2EEAB270656EC00170416 /* PlayerControlsView.swift in Sources */,
|
37E2EEAB270656EC00170416 /* PlayerControlsView.swift in Sources */,
|
||||||
376A33E42720CB35000C1D6B /* Account.swift in Sources */,
|
376A33E42720CB35000C1D6B /* Account.swift in Sources */,
|
||||||
37BA794326DBA973002A0235 /* PlaylistsModel.swift in Sources */,
|
37BA794326DBA973002A0235 /* PlaylistsModel.swift in Sources */,
|
||||||
@ -1409,7 +1430,9 @@
|
|||||||
37B81AFF26D2CA3700675966 /* VideoDetails.swift in Sources */,
|
37B81AFF26D2CA3700675966 /* VideoDetails.swift in Sources */,
|
||||||
377FC7E5267A084E00A6BBAF /* SearchView.swift in Sources */,
|
377FC7E5267A084E00A6BBAF /* SearchView.swift in Sources */,
|
||||||
376578912685490700D4EA09 /* PlaylistsView.swift in Sources */,
|
376578912685490700D4EA09 /* PlaylistsView.swift in Sources */,
|
||||||
|
37C3A24527235DA70087A57A /* ChannelPlaylist.swift in Sources */,
|
||||||
37FB28412721B22200A57617 /* ContentItem.swift in Sources */,
|
37FB28412721B22200A57617 /* ContentItem.swift in Sources */,
|
||||||
|
37C3A24D272360470087A57A /* ChannelPlaylist+Fixtures.swift in Sources */,
|
||||||
37484C2D26FC844700287258 /* AccountsSettingsView.swift in Sources */,
|
37484C2D26FC844700287258 /* AccountsSettingsView.swift in Sources */,
|
||||||
377A20A92693C9A2002842B8 /* TypedContentAccessors.swift in Sources */,
|
377A20A92693C9A2002842B8 /* TypedContentAccessors.swift in Sources */,
|
||||||
37484C3126FCB8F900287258 /* AccountValidator.swift in Sources */,
|
37484C3126FCB8F900287258 /* AccountValidator.swift in Sources */,
|
||||||
@ -1429,6 +1452,7 @@
|
|||||||
37B767DB2677C3CA0098BAA8 /* PlayerModel.swift in Sources */,
|
37B767DB2677C3CA0098BAA8 /* PlayerModel.swift in Sources */,
|
||||||
3788AC2726F6840700F6BAA9 /* WatchNowSection.swift in Sources */,
|
3788AC2726F6840700F6BAA9 /* WatchNowSection.swift in Sources */,
|
||||||
375DFB5826F9DA010013F468 /* InstancesModel.swift in Sources */,
|
375DFB5826F9DA010013F468 /* InstancesModel.swift in Sources */,
|
||||||
|
37C3A24927235FAA0087A57A /* ChannelPlaylistCell.swift in Sources */,
|
||||||
373CFACB26966264003CB2C6 /* SearchQuery.swift in Sources */,
|
373CFACB26966264003CB2C6 /* SearchQuery.swift in Sources */,
|
||||||
37141673267A8E10006CA35D /* Country.swift in Sources */,
|
37141673267A8E10006CA35D /* Country.swift in Sources */,
|
||||||
3748186E26A769D60084E870 /* DetailBadge.swift in Sources */,
|
3748186E26A769D60084E870 /* DetailBadge.swift in Sources */,
|
||||||
@ -1465,11 +1489,13 @@
|
|||||||
37C194C826F6A9C8005D3B96 /* RecentsModel.swift in Sources */,
|
37C194C826F6A9C8005D3B96 /* RecentsModel.swift in Sources */,
|
||||||
37BE0BDC26A2367F0092E2DB /* Player.swift in Sources */,
|
37BE0BDC26A2367F0092E2DB /* Player.swift in Sources */,
|
||||||
3743CA53270F284F00E4D32B /* View+Borders.swift in Sources */,
|
3743CA53270F284F00E4D32B /* View+Borders.swift in Sources */,
|
||||||
|
37C3A24627235DA70087A57A /* ChannelPlaylist.swift in Sources */,
|
||||||
3788AC2C26F6842D00F6BAA9 /* WatchNowSectionBody.swift in Sources */,
|
3788AC2C26F6842D00F6BAA9 /* WatchNowSectionBody.swift in Sources */,
|
||||||
37CEE4BE2677B670005A1EFE /* SingleAssetStream.swift in Sources */,
|
37CEE4BE2677B670005A1EFE /* SingleAssetStream.swift in Sources */,
|
||||||
37BA794826DC2E56002A0235 /* AppSidebarSubscriptions.swift in Sources */,
|
37BA794826DC2E56002A0235 /* AppSidebarSubscriptions.swift in Sources */,
|
||||||
37E70928271CDDAE00D34DDE /* OpenSettingsButton.swift in Sources */,
|
37E70928271CDDAE00D34DDE /* OpenSettingsButton.swift in Sources */,
|
||||||
37EAD86C267B9C5600D9E01B /* SponsorBlockAPI.swift in Sources */,
|
37EAD86C267B9C5600D9E01B /* SponsorBlockAPI.swift in Sources */,
|
||||||
|
37C3A24E272360470087A57A /* ChannelPlaylist+Fixtures.swift in Sources */,
|
||||||
37CEE4C22677B697005A1EFE /* Stream.swift in Sources */,
|
37CEE4C22677B697005A1EFE /* Stream.swift in Sources */,
|
||||||
371F2F1B269B43D300E4A7AB /* NavigationModel.swift in Sources */,
|
371F2F1B269B43D300E4A7AB /* NavigationModel.swift in Sources */,
|
||||||
37001564271B1F250049C794 /* AccountsModel.swift in Sources */,
|
37001564271B1F250049C794 /* AccountsModel.swift in Sources */,
|
||||||
@ -1557,7 +1583,9 @@
|
|||||||
37B17DA1268A1F89006AEE9B /* VideoContextMenuView.swift in Sources */,
|
37B17DA1268A1F89006AEE9B /* VideoContextMenuView.swift in Sources */,
|
||||||
3743B86927216D3600261544 /* ChannelCell.swift in Sources */,
|
3743B86927216D3600261544 /* ChannelCell.swift in Sources */,
|
||||||
3748186B26A764FB0084E870 /* Thumbnail+Fixtures.swift in Sources */,
|
3748186B26A764FB0084E870 /* Thumbnail+Fixtures.swift in Sources */,
|
||||||
|
37C3A252272366440087A57A /* ChannelPlaylistView.swift in Sources */,
|
||||||
373CFADC269663F1003CB2C6 /* Thumbnail.swift in Sources */,
|
373CFADC269663F1003CB2C6 /* Thumbnail.swift in Sources */,
|
||||||
|
37C3A24A27235FAA0087A57A /* ChannelPlaylistCell.swift in Sources */,
|
||||||
373CFAF02697A78B003CB2C6 /* AddToPlaylistView.swift in Sources */,
|
373CFAF02697A78B003CB2C6 /* AddToPlaylistView.swift in Sources */,
|
||||||
3763495226DFF59D00B9A393 /* AppSidebarRecents.swift in Sources */,
|
3763495226DFF59D00B9A393 /* AppSidebarRecents.swift in Sources */,
|
||||||
37BA794426DBA973002A0235 /* PlaylistsModel.swift in Sources */,
|
37BA794426DBA973002A0235 /* PlaylistsModel.swift in Sources */,
|
||||||
@ -1603,6 +1631,7 @@
|
|||||||
37BD07C92698FBDB003EBB87 /* ContentView.swift in Sources */,
|
37BD07C92698FBDB003EBB87 /* ContentView.swift in Sources */,
|
||||||
376B2E0926F920D600B1D64D /* SignInRequiredView.swift in Sources */,
|
376B2E0926F920D600B1D64D /* SignInRequiredView.swift in Sources */,
|
||||||
37141671267A8ACC006CA35D /* TrendingView.swift in Sources */,
|
37141671267A8ACC006CA35D /* TrendingView.swift in Sources */,
|
||||||
|
37C3A24727235DA70087A57A /* ChannelPlaylist.swift in Sources */,
|
||||||
3788AC2926F6840700F6BAA9 /* WatchNowSection.swift in Sources */,
|
3788AC2926F6840700F6BAA9 /* WatchNowSection.swift in Sources */,
|
||||||
37319F0727103F94004ECCD0 /* PlayerQueue.swift in Sources */,
|
37319F0727103F94004ECCD0 /* PlayerQueue.swift in Sources */,
|
||||||
37E70925271CD43000D34DDE /* WelcomeScreen.swift in Sources */,
|
37E70925271CD43000D34DDE /* WelcomeScreen.swift in Sources */,
|
||||||
@ -1641,6 +1670,7 @@
|
|||||||
3748186C26A764FB0084E870 /* Thumbnail+Fixtures.swift in Sources */,
|
3748186C26A764FB0084E870 /* Thumbnail+Fixtures.swift in Sources */,
|
||||||
377A20AB2693C9A2002842B8 /* TypedContentAccessors.swift in Sources */,
|
377A20AB2693C9A2002842B8 /* TypedContentAccessors.swift in Sources */,
|
||||||
3748186826A7627F0084E870 /* Video+Fixtures.swift in Sources */,
|
3748186826A7627F0084E870 /* Video+Fixtures.swift in Sources */,
|
||||||
|
37C3A253272366440087A57A /* ChannelPlaylistView.swift in Sources */,
|
||||||
3743CA54270F284F00E4D32B /* View+Borders.swift in Sources */,
|
3743CA54270F284F00E4D32B /* View+Borders.swift in Sources */,
|
||||||
371F2F1C269B43D300E4A7AB /* NavigationModel.swift in Sources */,
|
371F2F1C269B43D300E4A7AB /* NavigationModel.swift in Sources */,
|
||||||
37E2EEAD270656EC00170416 /* PlayerControlsView.swift in Sources */,
|
37E2EEAD270656EC00170416 /* PlayerControlsView.swift in Sources */,
|
||||||
@ -1658,6 +1688,7 @@
|
|||||||
379775952689365600DD52A8 /* Array+Next.swift in Sources */,
|
379775952689365600DD52A8 /* Array+Next.swift in Sources */,
|
||||||
3705B180267B4DFB00704544 /* TrendingCountry.swift in Sources */,
|
3705B180267B4DFB00704544 /* TrendingCountry.swift in Sources */,
|
||||||
373CFACD26966264003CB2C6 /* SearchQuery.swift in Sources */,
|
373CFACD26966264003CB2C6 /* SearchQuery.swift in Sources */,
|
||||||
|
37C3A24B27235FAA0087A57A /* ChannelPlaylistCell.swift in Sources */,
|
||||||
37FD43E52704847C0073EE42 /* View+Fixtures.swift in Sources */,
|
37FD43E52704847C0073EE42 /* View+Fixtures.swift in Sources */,
|
||||||
37141675267A8E10006CA35D /* Country.swift in Sources */,
|
37141675267A8E10006CA35D /* Country.swift in Sources */,
|
||||||
37152EEC26EFEB95004FB96D /* LazyView.swift in Sources */,
|
37152EEC26EFEB95004FB96D /* LazyView.swift in Sources */,
|
||||||
@ -1668,6 +1699,7 @@
|
|||||||
37D526E02720AC4400ED2F5E /* VideosAPI.swift in Sources */,
|
37D526E02720AC4400ED2F5E /* VideosAPI.swift in Sources */,
|
||||||
3705B184267B4E4900704544 /* TrendingCategory.swift in Sources */,
|
3705B184267B4E4900704544 /* TrendingCategory.swift in Sources */,
|
||||||
3761ABFF26F0F8DE00AA496F /* EnvironmentValues.swift in Sources */,
|
3761ABFF26F0F8DE00AA496F /* EnvironmentValues.swift in Sources */,
|
||||||
|
37C3A24F272360470087A57A /* ChannelPlaylist+Fixtures.swift in Sources */,
|
||||||
37FB28432721B22200A57617 /* ContentItem.swift in Sources */,
|
37FB28432721B22200A57617 /* ContentItem.swift in Sources */,
|
||||||
37AAF2A226741C97007FC770 /* SubscriptionsView.swift in Sources */,
|
37AAF2A226741C97007FC770 /* SubscriptionsView.swift in Sources */,
|
||||||
37484C1B26FC837400287258 /* PlaybackSettingsView.swift in Sources */,
|
37484C1B26FC837400287258 /* PlaybackSettingsView.swift in Sources */,
|
||||||
|
@ -17,6 +17,12 @@ struct AppSidebarRecents: View {
|
|||||||
RecentNavigationLink(recent: recent) {
|
RecentNavigationLink(recent: recent) {
|
||||||
LazyView(ChannelVideosView(channel: recent.channel!))
|
LazyView(ChannelVideosView(channel: recent.channel!))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case .playlist:
|
||||||
|
RecentNavigationLink(recent: recent, systemImage: "list.and.film") {
|
||||||
|
LazyView(ChannelPlaylistView(playlist: recent.playlist!))
|
||||||
|
}
|
||||||
|
|
||||||
case .query:
|
case .query:
|
||||||
RecentNavigationLink(recent: recent, systemImage: "magnifyingglass") {
|
RecentNavigationLink(recent: recent, systemImage: "magnifyingglass") {
|
||||||
LazyView(SearchView(recent.query!))
|
LazyView(SearchView(recent.query!))
|
||||||
@ -64,6 +70,7 @@ struct RecentNavigationLink<DestinationContent: View>: View {
|
|||||||
} label: {
|
} label: {
|
||||||
HStack {
|
HStack {
|
||||||
Label(recent.title, systemImage: labelSystemImage)
|
Label(recent.title, systemImage: labelSystemImage)
|
||||||
|
.lineLimit(1)
|
||||||
|
|
||||||
Spacer()
|
Spacer()
|
||||||
|
|
||||||
|
@ -2,9 +2,11 @@ import Defaults
|
|||||||
import SwiftUI
|
import SwiftUI
|
||||||
|
|
||||||
struct AppTabNavigation: View {
|
struct AppTabNavigation: View {
|
||||||
|
@EnvironmentObject<AccountsModel> private var accounts
|
||||||
@EnvironmentObject<NavigationModel> private var navigation
|
@EnvironmentObject<NavigationModel> private var navigation
|
||||||
@EnvironmentObject<SearchModel> private var search
|
@EnvironmentObject<PlayerModel> private var player
|
||||||
@EnvironmentObject<RecentsModel> private var recents
|
@EnvironmentObject<RecentsModel> private var recents
|
||||||
|
@EnvironmentObject<SearchModel> private var search
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
TabView(selection: navigation.tabSelectionBinding) {
|
TabView(selection: navigation.tabSelectionBinding) {
|
||||||
@ -18,27 +20,30 @@ struct AppTabNavigation: View {
|
|||||||
}
|
}
|
||||||
.tag(TabSelection.watchNow)
|
.tag(TabSelection.watchNow)
|
||||||
|
|
||||||
NavigationView {
|
if accounts.app.supportsSubscriptions {
|
||||||
LazyView(SubscriptionsView())
|
NavigationView {
|
||||||
.toolbar { toolbarContent }
|
LazyView(SubscriptionsView())
|
||||||
|
.toolbar { toolbarContent }
|
||||||
|
}
|
||||||
|
.tabItem {
|
||||||
|
Label("Subscriptions", systemImage: "star.circle.fill")
|
||||||
|
.accessibility(label: Text("Subscriptions"))
|
||||||
|
}
|
||||||
|
.tag(TabSelection.subscriptions)
|
||||||
}
|
}
|
||||||
.tabItem {
|
|
||||||
Label("Subscriptions", systemImage: "star.circle.fill")
|
|
||||||
.accessibility(label: Text("Subscriptions"))
|
|
||||||
}
|
|
||||||
.tag(TabSelection.subscriptions)
|
|
||||||
|
|
||||||
// TODO: reenable with settings
|
// TODO: reenable with settings
|
||||||
// ============================
|
if accounts.app.supportsPopular && false {
|
||||||
// NavigationView {
|
NavigationView {
|
||||||
// LazyView(PopularView())
|
LazyView(PopularView())
|
||||||
// .toolbar { toolbarContent }
|
.toolbar { toolbarContent }
|
||||||
// }
|
}
|
||||||
// .tabItem {
|
.tabItem {
|
||||||
// Label("Popular", systemImage: "chart.bar")
|
Label("Popular", systemImage: "chart.bar")
|
||||||
// .accessibility(label: Text("Popular"))
|
.accessibility(label: Text("Popular"))
|
||||||
// }
|
}
|
||||||
// .tag(TabSelection.popular)
|
.tag(TabSelection.popular)
|
||||||
|
}
|
||||||
|
|
||||||
NavigationView {
|
NavigationView {
|
||||||
LazyView(TrendingView())
|
LazyView(TrendingView())
|
||||||
@ -50,15 +55,17 @@ struct AppTabNavigation: View {
|
|||||||
}
|
}
|
||||||
.tag(TabSelection.trending)
|
.tag(TabSelection.trending)
|
||||||
|
|
||||||
NavigationView {
|
if accounts.app.supportsUserPlaylists {
|
||||||
LazyView(PlaylistsView())
|
NavigationView {
|
||||||
.toolbar { toolbarContent }
|
LazyView(PlaylistsView())
|
||||||
|
.toolbar { toolbarContent }
|
||||||
|
}
|
||||||
|
.tabItem {
|
||||||
|
Label("Playlists", systemImage: "list.and.film")
|
||||||
|
.accessibility(label: Text("Playlists"))
|
||||||
|
}
|
||||||
|
.tag(TabSelection.playlists)
|
||||||
}
|
}
|
||||||
.tabItem {
|
|
||||||
Label("Playlists", systemImage: "list.and.film")
|
|
||||||
.accessibility(label: Text("Playlists"))
|
|
||||||
}
|
|
||||||
.tag(TabSelection.playlists)
|
|
||||||
|
|
||||||
NavigationView {
|
NavigationView {
|
||||||
LazyView(
|
LazyView(
|
||||||
@ -89,19 +96,41 @@ struct AppTabNavigation: View {
|
|||||||
.tag(TabSelection.search)
|
.tag(TabSelection.search)
|
||||||
}
|
}
|
||||||
.environment(\.navigationStyle, .tab)
|
.environment(\.navigationStyle, .tab)
|
||||||
.sheet(isPresented: $navigation.isChannelOpen, onDismiss: {
|
.sheet(isPresented: $navigation.presentingChannel, onDismiss: {
|
||||||
if let channel = recents.presentedChannel {
|
if let channel = recents.presentedChannel {
|
||||||
let recent = RecentItem(from: channel)
|
recents.close(RecentItem(from: channel))
|
||||||
recents.close(recent)
|
|
||||||
}
|
}
|
||||||
}) {
|
}) {
|
||||||
if recents.presentedChannel != nil {
|
if let channel = recents.presentedChannel {
|
||||||
NavigationView {
|
NavigationView {
|
||||||
ChannelVideosView(channel: recents.presentedChannel!)
|
ChannelVideosView(channel: channel)
|
||||||
.environment(\.inNavigationView, true)
|
.environment(\.inNavigationView, true)
|
||||||
|
.background(playerNavigationLink)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
.sheet(isPresented: $navigation.presentingPlaylist, onDismiss: {
|
||||||
|
if let playlist = recents.presentedPlaylist {
|
||||||
|
recents.close(RecentItem(from: playlist))
|
||||||
|
}
|
||||||
|
}) {
|
||||||
|
if let playlist = recents.presentedPlaylist {
|
||||||
|
NavigationView {
|
||||||
|
ChannelPlaylistView(playlist: playlist)
|
||||||
|
.environment(\.inNavigationView, true)
|
||||||
|
.background(playerNavigationLink)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private var playerNavigationLink: some View {
|
||||||
|
NavigationLink(isActive: $player.playerNavigationLinkActive, destination: {
|
||||||
|
VideoPlayerView()
|
||||||
|
.environment(\.inNavigationView, true)
|
||||||
|
}) {
|
||||||
|
EmptyView()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var toolbarContent: some ToolbarContent {
|
var toolbarContent: some ToolbarContent {
|
||||||
|
@ -4,8 +4,6 @@ import SwiftUI
|
|||||||
|
|
||||||
struct VideoCell: View {
|
struct VideoCell: View {
|
||||||
var video: Video
|
var video: Video
|
||||||
|
|
||||||
@State private var playerNavigationLinkActive = false
|
|
||||||
@State private var lowQualityThumbnail = false
|
@State private var lowQualityThumbnail = false
|
||||||
|
|
||||||
@Environment(\.inNavigationView) private var inNavigationView
|
@Environment(\.inNavigationView) private var inNavigationView
|
||||||
@ -23,24 +21,17 @@ struct VideoCell: View {
|
|||||||
player.playNow(video)
|
player.playNow(video)
|
||||||
|
|
||||||
if inNavigationView {
|
if inNavigationView {
|
||||||
playerNavigationLinkActive = true
|
player.playerNavigationLinkActive = true
|
||||||
} else {
|
} else {
|
||||||
player.presentPlayer()
|
player.presentPlayer()
|
||||||
}
|
}
|
||||||
}) {
|
}) {
|
||||||
content
|
content
|
||||||
}
|
}
|
||||||
|
|
||||||
NavigationLink(isActive: $playerNavigationLinkActive, destination: {
|
|
||||||
VideoPlayerView()
|
|
||||||
.environment(\.inNavigationView, true)
|
|
||||||
}) {
|
|
||||||
EmptyView()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
.buttonStyle(.plain)
|
.buttonStyle(.plain)
|
||||||
.contentShape(RoundedRectangle(cornerRadius: 12))
|
.contentShape(RoundedRectangle(cornerRadius: 12))
|
||||||
.contextMenu { VideoContextMenuView(video: video, playerNavigationLinkActive: $playerNavigationLinkActive) }
|
.contextMenu { VideoContextMenuView(video: video, playerNavigationLinkActive: $player.playerNavigationLinkActive) }
|
||||||
}
|
}
|
||||||
|
|
||||||
var content: some View {
|
var content: some View {
|
||||||
@ -90,7 +81,7 @@ struct VideoCell: View {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if video.views != 0 {
|
if video.views > 0 {
|
||||||
VStack {
|
VStack {
|
||||||
Image(systemName: "eye")
|
Image(systemName: "eye")
|
||||||
Text(video.viewsCount!)
|
Text(video.viewsCount!)
|
||||||
@ -125,6 +116,7 @@ struct VideoCell: View {
|
|||||||
|
|
||||||
Spacer()
|
Spacer()
|
||||||
}
|
}
|
||||||
|
.lineLimit(1)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
@ -154,7 +146,7 @@ struct VideoCell: View {
|
|||||||
Text(date)
|
Text(date)
|
||||||
}
|
}
|
||||||
|
|
||||||
if video.views != 0 {
|
if video.views > 0 {
|
||||||
Image(systemName: "eye")
|
Image(systemName: "eye")
|
||||||
Text(video.viewsCount!)
|
Text(video.viewsCount!)
|
||||||
}
|
}
|
||||||
@ -210,6 +202,7 @@ struct VideoCell: View {
|
|||||||
}
|
}
|
||||||
.padding(10)
|
.padding(10)
|
||||||
}
|
}
|
||||||
|
.lineLimit(1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -253,7 +246,7 @@ struct VideoCell: View {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct VideoView_Preview: PreviewProvider {
|
struct VideoCell_Preview: PreviewProvider {
|
||||||
static var previews: some View {
|
static var previews: some View {
|
||||||
Group {
|
Group {
|
||||||
VideoCell(video: Video.fixture)
|
VideoCell(video: Video.fixture)
|
||||||
|
@ -14,7 +14,7 @@ struct ChannelCell: View {
|
|||||||
Button {
|
Button {
|
||||||
let recent = RecentItem(from: channel)
|
let recent = RecentItem(from: channel)
|
||||||
recents.add(recent)
|
recents.add(recent)
|
||||||
navigation.isChannelOpen = true
|
navigation.presentingChannel = true
|
||||||
|
|
||||||
if navigationStyle == .sidebar {
|
if navigationStyle == .sidebar {
|
||||||
navigation.sidebarSectionChanged.toggle()
|
navigation.sidebarSectionChanged.toggle()
|
||||||
@ -30,10 +30,13 @@ struct ChannelCell: View {
|
|||||||
|
|
||||||
var content: some View {
|
var content: some View {
|
||||||
VStack {
|
VStack {
|
||||||
Text("Channel".uppercased())
|
HStack(alignment: .top, spacing: 3) {
|
||||||
.foregroundColor(.secondary)
|
Image(systemName: "person.crop.rectangle")
|
||||||
.fontWeight(.light)
|
Text("Channel".uppercased())
|
||||||
.opacity(0.6)
|
.fontWeight(.light)
|
||||||
|
.opacity(0.6)
|
||||||
|
}
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
|
||||||
WebImage(url: channel.thumbnailURL)
|
WebImage(url: channel.thumbnailURL)
|
||||||
.resizable()
|
.resizable()
|
||||||
@ -44,20 +47,17 @@ struct ChannelCell: View {
|
|||||||
.frame(width: 88, height: 88)
|
.frame(width: 88, height: 88)
|
||||||
.clipShape(Circle())
|
.clipShape(Circle())
|
||||||
|
|
||||||
Group {
|
DetailBadge(text: channel.name, style: .prominent)
|
||||||
DetailBadge(text: channel.name, style: .prominent)
|
|
||||||
|
|
||||||
Group {
|
Group {
|
||||||
if let subscriptions = channel.subscriptionsString {
|
if let subscriptions = channel.subscriptionsString {
|
||||||
Text("\(subscriptions) subscribers")
|
Text("\(subscriptions) subscribers")
|
||||||
.foregroundColor(.secondary)
|
.foregroundColor(.secondary)
|
||||||
} else {
|
} else {
|
||||||
Text("")
|
Text("")
|
||||||
}
|
|
||||||
}
|
}
|
||||||
.frame(height: 20)
|
|
||||||
}
|
}
|
||||||
.offset(x: 0, y: -15)
|
.frame(height: 20)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
68
Shared/Views/ChannelPlaylistCell.swift
Normal file
68
Shared/Views/ChannelPlaylistCell.swift
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import SDWebImageSwiftUI
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct ChannelPlaylistCell: View {
|
||||||
|
let playlist: ChannelPlaylist
|
||||||
|
|
||||||
|
@Environment(\.navigationStyle) private var navigationStyle
|
||||||
|
|
||||||
|
@EnvironmentObject<NavigationModel> private var navigation
|
||||||
|
@EnvironmentObject<RecentsModel> private var recents
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
Button {
|
||||||
|
let recent = RecentItem(from: playlist)
|
||||||
|
recents.add(recent)
|
||||||
|
navigation.presentingPlaylist = true
|
||||||
|
|
||||||
|
if navigationStyle == .sidebar {
|
||||||
|
navigation.sidebarSectionChanged.toggle()
|
||||||
|
navigation.tabSelection = .recentlyOpened(recent.tag)
|
||||||
|
}
|
||||||
|
} label: {
|
||||||
|
content
|
||||||
|
.frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: .infinity)
|
||||||
|
.contentShape(RoundedRectangle(cornerRadius: 12))
|
||||||
|
}
|
||||||
|
.buttonStyle(.plain)
|
||||||
|
}
|
||||||
|
|
||||||
|
var content: some View {
|
||||||
|
VStack {
|
||||||
|
HStack(alignment: .top, spacing: 3) {
|
||||||
|
Image(systemName: "list.and.film")
|
||||||
|
Text("Playlist".uppercased())
|
||||||
|
.fontWeight(.light)
|
||||||
|
.opacity(0.6)
|
||||||
|
}
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
|
||||||
|
WebImage(url: playlist.thumbnailURL)
|
||||||
|
.resizable()
|
||||||
|
.placeholder {
|
||||||
|
Rectangle().fill(Color("PlaceholderColor"))
|
||||||
|
}
|
||||||
|
.indicator(.progress)
|
||||||
|
.frame(width: 165, height: 88)
|
||||||
|
.clipShape(RoundedRectangle(cornerRadius: 10))
|
||||||
|
|
||||||
|
Group {
|
||||||
|
DetailBadge(text: playlist.title, style: .prominent)
|
||||||
|
.lineLimit(2)
|
||||||
|
|
||||||
|
Text("\(playlist.videosCount ?? playlist.videos.count) videos")
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
|
||||||
|
.frame(height: 20)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ChannelPlaylistCell_Previews: PreviewProvider {
|
||||||
|
static var previews: some View {
|
||||||
|
ChannelPlaylistCell(playlist: ChannelPlaylist.fixture)
|
||||||
|
.frame(maxWidth: 320)
|
||||||
|
.injectFixtureEnvironmentObjects()
|
||||||
|
}
|
||||||
|
}
|
74
Shared/Views/ChannelPlaylistView.swift
Normal file
74
Shared/Views/ChannelPlaylistView.swift
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
import Siesta
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct ChannelPlaylistView: View {
|
||||||
|
var playlist: ChannelPlaylist
|
||||||
|
|
||||||
|
@StateObject private var store = Store<ChannelPlaylist>()
|
||||||
|
|
||||||
|
@Environment(\.dismiss) private var dismiss
|
||||||
|
@Environment(\.inNavigationView) private var inNavigationView
|
||||||
|
|
||||||
|
@EnvironmentObject<AccountsModel> private var accounts
|
||||||
|
|
||||||
|
var items: [ContentItem] {
|
||||||
|
ContentItem.array(of: store.item?.videos ?? [])
|
||||||
|
}
|
||||||
|
|
||||||
|
var resource: Resource? {
|
||||||
|
accounts.api.channelPlaylist(playlist.id)
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
#if os(iOS)
|
||||||
|
if inNavigationView {
|
||||||
|
content
|
||||||
|
} else {
|
||||||
|
PlayerControlsView {
|
||||||
|
content
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
PlayerControlsView {
|
||||||
|
content
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
var content: some View {
|
||||||
|
VStack(alignment: .leading) {
|
||||||
|
#if os(tvOS)
|
||||||
|
Text(playlist.title)
|
||||||
|
.font(.title2)
|
||||||
|
.frame(alignment: .leading)
|
||||||
|
#endif
|
||||||
|
VerticalCells(items: items)
|
||||||
|
}
|
||||||
|
.onAppear {
|
||||||
|
resource?.addObserver(store)
|
||||||
|
resource?.loadIfNeeded()
|
||||||
|
}
|
||||||
|
#if !os(tvOS)
|
||||||
|
.toolbar {
|
||||||
|
ToolbarItem(placement: .cancellationAction) {
|
||||||
|
if inNavigationView {
|
||||||
|
Button("Done") {
|
||||||
|
dismiss()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.navigationTitle(playlist.title)
|
||||||
|
|
||||||
|
#else
|
||||||
|
.background(.thickMaterial)
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ChannelPlaylistView_Previews: PreviewProvider {
|
||||||
|
static var previews: some View {
|
||||||
|
ChannelPlaylistView(playlist: ChannelPlaylist.fixture)
|
||||||
|
.injectFixtureEnvironmentObjects()
|
||||||
|
}
|
||||||
|
}
|
@ -6,17 +6,17 @@ struct ChannelVideosView: View {
|
|||||||
|
|
||||||
@StateObject private var store = Store<Channel>()
|
@StateObject private var store = Store<Channel>()
|
||||||
|
|
||||||
@EnvironmentObject<AccountsModel> private var accounts
|
@Environment(\.dismiss) private var dismiss
|
||||||
@EnvironmentObject<NavigationModel> private var navigation
|
|
||||||
@EnvironmentObject<SubscriptionsModel> private var subscriptions
|
|
||||||
|
|
||||||
@Environment(\.inNavigationView) private var inNavigationView
|
@Environment(\.inNavigationView) private var inNavigationView
|
||||||
|
|
||||||
@Environment(\.dismiss) private var dismiss
|
|
||||||
#if os(iOS)
|
#if os(iOS)
|
||||||
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
|
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@EnvironmentObject<AccountsModel> private var accounts
|
||||||
|
@EnvironmentObject<NavigationModel> private var navigation
|
||||||
|
@EnvironmentObject<SubscriptionsModel> private var subscriptions
|
||||||
|
|
||||||
@Namespace private var focusNamespace
|
@Namespace private var focusNamespace
|
||||||
|
|
||||||
var videos: [ContentItem] {
|
var videos: [ContentItem] {
|
||||||
@ -88,8 +88,7 @@ struct ChannelVideosView: View {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#else
|
||||||
#if os(tvOS)
|
|
||||||
.background(.thickMaterial)
|
.background(.thickMaterial)
|
||||||
#endif
|
#endif
|
||||||
.modifier(UnsubscribeAlertModifier())
|
.modifier(UnsubscribeAlertModifier())
|
||||||
|
@ -8,7 +8,7 @@ struct ContentItemView: View {
|
|||||||
Group {
|
Group {
|
||||||
switch item.contentType {
|
switch item.contentType {
|
||||||
case .playlist:
|
case .playlist:
|
||||||
VideoCell(video: item.video)
|
ChannelPlaylistCell(playlist: item.playlist)
|
||||||
case .channel:
|
case .channel:
|
||||||
ChannelCell(channel: item.channel)
|
ChannelCell(channel: item.channel)
|
||||||
default:
|
default:
|
||||||
|
@ -73,7 +73,6 @@ struct DetailBadge: View {
|
|||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
Text(text)
|
Text(text)
|
||||||
.lineLimit(1)
|
|
||||||
.truncationMode(.middle)
|
.truncationMode(.middle)
|
||||||
.padding(10)
|
.padding(10)
|
||||||
.modifier(StyleModifier(style: style))
|
.modifier(StyleModifier(style: style))
|
||||||
|
@ -25,6 +25,10 @@ struct SearchView: View {
|
|||||||
|
|
||||||
private var videos = [Video]()
|
private var videos = [Video]()
|
||||||
|
|
||||||
|
var items: [ContentItem] {
|
||||||
|
state.store.collection.sorted { $0 < $1 }
|
||||||
|
}
|
||||||
|
|
||||||
init(_ query: SearchQuery? = nil, videos: [Video] = [Video]()) {
|
init(_ query: SearchQuery? = nil, videos: [Video] = [Video]()) {
|
||||||
self.query = query
|
self.query = query
|
||||||
self.videos = videos
|
self.videos = videos
|
||||||
@ -42,11 +46,11 @@ struct SearchView: View {
|
|||||||
filtersHorizontalStack
|
filtersHorizontalStack
|
||||||
}
|
}
|
||||||
|
|
||||||
HorizontalCells(items: state.store.collection)
|
HorizontalCells(items: items)
|
||||||
}
|
}
|
||||||
.edgesIgnoringSafeArea(.horizontal)
|
.edgesIgnoringSafeArea(.horizontal)
|
||||||
#else
|
#else
|
||||||
VerticalCells(items: state.store.collection)
|
VerticalCells(items: items)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if noResults {
|
if noResults {
|
||||||
@ -173,7 +177,7 @@ struct SearchView: View {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fileprivate var noResults: Bool {
|
fileprivate var noResults: Bool {
|
||||||
state.store.collection.isEmpty && !state.isLoading && !state.query.isEmpty
|
items.isEmpty && !state.isLoading && !state.query.isEmpty
|
||||||
}
|
}
|
||||||
|
|
||||||
var recentQueries: some View {
|
var recentQueries: some View {
|
||||||
|
@ -85,7 +85,7 @@ struct VideoContextMenuView: View {
|
|||||||
Button {
|
Button {
|
||||||
let recent = RecentItem(from: video.channel)
|
let recent = RecentItem(from: video.channel)
|
||||||
recents.add(recent)
|
recents.add(recent)
|
||||||
navigation.isChannelOpen = true
|
navigation.presentingChannel = true
|
||||||
|
|
||||||
if navigationStyle == .sidebar {
|
if navigationStyle == .sidebar {
|
||||||
navigation.sidebarSectionChanged.toggle()
|
navigation.sidebarSectionChanged.toggle()
|
||||||
|
@ -53,11 +53,16 @@ struct TVNavigationView: View {
|
|||||||
.fullScreenCover(isPresented: $player.presentingPlayer) {
|
.fullScreenCover(isPresented: $player.presentingPlayer) {
|
||||||
VideoPlayerView()
|
VideoPlayerView()
|
||||||
}
|
}
|
||||||
.fullScreenCover(isPresented: $navigation.isChannelOpen) {
|
.fullScreenCover(isPresented: $navigation.presentingChannel) {
|
||||||
if let channel = recents.presentedChannel {
|
if let channel = recents.presentedChannel {
|
||||||
ChannelVideosView(channel: channel)
|
ChannelVideosView(channel: channel)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
.fullScreenCover(isPresented: $navigation.presentingPlaylist) {
|
||||||
|
if let playlist = recents.presentedPlaylist {
|
||||||
|
ChannelPlaylistView(playlist: playlist)
|
||||||
|
}
|
||||||
|
}
|
||||||
.onPlayPauseCommand { navigation.presentingSettings.toggle() }
|
.onPlayPauseCommand { navigation.presentingSettings.toggle() }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user