mirror of
https://github.com/yattee/yattee.git
synced 2024-12-22 21:43:41 +00:00
Simple view display switching
This commit is contained in:
parent
b336d2c512
commit
15bfaf7497
@ -8,15 +8,13 @@ struct PlayerViewController: UIViewControllerRepresentable {
|
|||||||
|
|
||||||
@ObservedObject private var state: PlayerState
|
@ObservedObject private var state: PlayerState
|
||||||
|
|
||||||
@ObservedObject private var profile = Profile()
|
|
||||||
|
|
||||||
var video: Video
|
var video: Video
|
||||||
|
|
||||||
init(video: Video) {
|
init(video: Video) {
|
||||||
self.video = video
|
self.video = video
|
||||||
state = PlayerState(video)
|
state = PlayerState(video)
|
||||||
|
|
||||||
loadStream(video.defaultStreamForProfile(profile), loadBest: profile.defaultStreamResolution == .hd720pFirstThenBest)
|
loadStream(video.defaultStreamForProfile(state.profile), loadBest: state.profile.defaultStreamResolution == .hd720pFirstThenBest)
|
||||||
}
|
}
|
||||||
|
|
||||||
fileprivate func loadStream(_ stream: Stream?, loadBest: Bool = false) {
|
fileprivate func loadStream(_ stream: Stream?, loadBest: Bool = false) {
|
||||||
|
@ -64,8 +64,13 @@ struct PlaylistsView: View {
|
|||||||
|
|
||||||
extension Array where Element: Equatable {
|
extension Array where Element: Equatable {
|
||||||
func next(after element: Element) -> Element? {
|
func next(after element: Element) -> Element? {
|
||||||
let idx = firstIndex(of: element)!
|
let idx = firstIndex(of: element)
|
||||||
let next = index(after: idx)
|
|
||||||
|
if idx == nil {
|
||||||
|
return first
|
||||||
|
}
|
||||||
|
|
||||||
|
let next = index(after: idx!)
|
||||||
|
|
||||||
return self[next == endIndex ? startIndex : next]
|
return self[next == endIndex ? startIndex : next]
|
||||||
}
|
}
|
||||||
|
@ -7,11 +7,6 @@ struct PopularVideosView: View {
|
|||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
VideosView(tabSelection: $tabSelection, videos: videos)
|
VideosView(tabSelection: $tabSelection, videos: videos)
|
||||||
.task {
|
|
||||||
Task {
|
|
||||||
provider.load()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var videos: [Video] {
|
var videos: [Video] {
|
||||||
|
@ -2,6 +2,7 @@ import SwiftUI
|
|||||||
|
|
||||||
struct SearchView: View {
|
struct SearchView: View {
|
||||||
@ObservedObject private var provider = SearchedVideosProvider()
|
@ObservedObject private var provider = SearchedVideosProvider()
|
||||||
|
@EnvironmentObject private var profile: Profile
|
||||||
@EnvironmentObject private var state: AppState
|
@EnvironmentObject private var state: AppState
|
||||||
|
|
||||||
@Binding var tabSelection: TabSelection
|
@Binding var tabSelection: TabSelection
|
||||||
@ -11,6 +12,7 @@ struct SearchView: View {
|
|||||||
var body: some View {
|
var body: some View {
|
||||||
VideosView(tabSelection: $tabSelection, videos: videos)
|
VideosView(tabSelection: $tabSelection, videos: videos)
|
||||||
.environmentObject(state)
|
.environmentObject(state)
|
||||||
|
.environmentObject(profile)
|
||||||
.searchable(text: $query)
|
.searchable(text: $query)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,21 +1,19 @@
|
|||||||
import SwiftUI
|
import SwiftUI
|
||||||
|
|
||||||
struct SubscriptionsView: View {
|
struct SubscriptionsView: View {
|
||||||
@ObservedObject private var provider = SubscriptionVideosProvider()
|
|
||||||
@EnvironmentObject private var state: AppState
|
|
||||||
|
|
||||||
@Binding var tabSelection: TabSelection
|
@Binding var tabSelection: TabSelection
|
||||||
|
|
||||||
|
@ObservedObject private var provider = SubscriptionVideosProvider()
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
VideosView(tabSelection: $tabSelection, videos: videos)
|
VideosView(tabSelection: $tabSelection, videos: videos)
|
||||||
.task {
|
|
||||||
Task {
|
|
||||||
provider.load()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var videos: [Video] {
|
var videos: [Video] {
|
||||||
provider.videos
|
if provider.videos.isEmpty {
|
||||||
|
provider.load()
|
||||||
|
}
|
||||||
|
|
||||||
|
return provider.videos
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,18 +1,22 @@
|
|||||||
import SwiftUI
|
import SwiftUI
|
||||||
|
|
||||||
struct VideosView: View {
|
struct VideosView: View {
|
||||||
@EnvironmentObject private var state: AppState
|
@EnvironmentObject private var profile: Profile
|
||||||
|
|
||||||
@Binding var tabSelection: TabSelection
|
@Binding var tabSelection: TabSelection
|
||||||
var videos: [Video]
|
var videos: [Video]
|
||||||
|
|
||||||
|
@State private var showingViewOptions = false
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
Group {
|
Section {
|
||||||
if state.profile.listing == .list {
|
if self.profile.listing == .list {
|
||||||
VideosListView(tabSelection: $tabSelection, videos: videos)
|
VideosListView(tabSelection: $tabSelection, videos: videos)
|
||||||
} else {
|
} else {
|
||||||
VideosCellsView(videos: videos, columns: state.profile.cellsColumns)
|
VideosCellsView(videos: videos, columns: self.profile.cellsColumns)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
.fullScreenCover(isPresented: $showingViewOptions) { ViewOptionsView() }
|
||||||
|
.onPlayPauseCommand { showingViewOptions.toggle() }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
32
Apple TV/ViewOptionsView.swift
Normal file
32
Apple TV/ViewOptionsView.swift
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct ViewOptionsView: View {
|
||||||
|
@EnvironmentObject private var profile: Profile
|
||||||
|
|
||||||
|
@Environment(\.presentationMode) private var presentationMode
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
ZStack {
|
||||||
|
VisualEffectView(effect: UIBlurEffect(style: .dark))
|
||||||
|
|
||||||
|
VStack {
|
||||||
|
Spacer()
|
||||||
|
|
||||||
|
ScrollView(.vertical) {
|
||||||
|
Button(profile.listing == .list ? "Cells" : "List") {
|
||||||
|
profile.listing = (profile.listing == .list ? .cells : .list)
|
||||||
|
presentationMode.wrappedValue.dismiss()
|
||||||
|
}
|
||||||
|
|
||||||
|
Button("Close") {
|
||||||
|
presentationMode.wrappedValue.dismiss()
|
||||||
|
}
|
||||||
|
.frame(width: 800)
|
||||||
|
}
|
||||||
|
Spacer()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
||||||
|
.edgesIgnoringSafeArea(.all)
|
||||||
|
}
|
||||||
|
}
|
@ -6,8 +6,6 @@ final class AppState: ObservableObject {
|
|||||||
@Published var channelID: String = ""
|
@Published var channelID: String = ""
|
||||||
@Published var channel: String = ""
|
@Published var channel: String = ""
|
||||||
|
|
||||||
@Published var profile = Profile()
|
|
||||||
|
|
||||||
func openChannel(from video: Video) {
|
func openChannel(from video: Video) {
|
||||||
channel = video.author
|
channel = video.author
|
||||||
channelID = video.channelID
|
channelID = video.channelID
|
||||||
|
@ -22,7 +22,7 @@ final class PlayerState: ObservableObject {
|
|||||||
|
|
||||||
@Published var currentSegment: Segment?
|
@Published var currentSegment: Segment?
|
||||||
|
|
||||||
private var profile = Profile()
|
private(set) var profile = Profile()
|
||||||
|
|
||||||
@Published private(set) var currentRate: Float = 0.0
|
@Published private(set) var currentRate: Float = 0.0
|
||||||
static let availablePlaybackRates: [Double] = [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2]
|
static let availablePlaybackRates: [Double] = [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2]
|
||||||
|
@ -1,16 +1,16 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
final class Profile: ObservableObject {
|
final class Profile: ObservableObject {
|
||||||
let defaultStreamResolution: DefaultStreamResolution = .hd720pFirstThenBest
|
var defaultStreamResolution: DefaultStreamResolution = .hd720pFirstThenBest
|
||||||
|
|
||||||
let skippedSegmentsCategories = [String]() // SponsorBlockSegmentsProvider.categories
|
var skippedSegmentsCategories = [String]() // SponsorBlockSegmentsProvider.categories
|
||||||
|
|
||||||
// let sid = "B3_WzklziGu8JKefihLrCsTNavdj73KMiPUBfN5HW2M="
|
// var sid = "B3_WzklziGu8JKefihLrCsTNavdj73KMiPUBfN5HW2M="
|
||||||
let sid = "RpoS7YPPK2-QS81jJF9z4KSQAjmzsOnMpn84c73-GQ8="
|
var sid = "RpoS7YPPK2-QS81jJF9z4KSQAjmzsOnMpn84c73-GQ8="
|
||||||
|
|
||||||
let listing = VideoListing.cells
|
@Published var listing = VideoListing.cells
|
||||||
|
|
||||||
let cellsColumns = 3
|
var cellsColumns = 3
|
||||||
}
|
}
|
||||||
|
|
||||||
enum VideoListing: String {
|
enum VideoListing: String {
|
||||||
|
@ -93,6 +93,9 @@
|
|||||||
37B767DC2677C3CA0098BAA8 /* PlayerState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37B767DA2677C3CA0098BAA8 /* PlayerState.swift */; };
|
37B767DC2677C3CA0098BAA8 /* PlayerState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37B767DA2677C3CA0098BAA8 /* PlayerState.swift */; };
|
||||||
37B767DD2677C3CA0098BAA8 /* PlayerState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37B767DA2677C3CA0098BAA8 /* PlayerState.swift */; };
|
37B767DD2677C3CA0098BAA8 /* PlayerState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37B767DA2677C3CA0098BAA8 /* PlayerState.swift */; };
|
||||||
37B767E02678C5BF0098BAA8 /* Logging in Frameworks */ = {isa = PBXBuildFile; productRef = 37B767DF2678C5BF0098BAA8 /* Logging */; };
|
37B767E02678C5BF0098BAA8 /* Logging in Frameworks */ = {isa = PBXBuildFile; productRef = 37B767DF2678C5BF0098BAA8 /* Logging */; };
|
||||||
|
37B76E96268747C900CE5671 /* ViewOptionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37B76E95268747C900CE5671 /* ViewOptionsView.swift */; };
|
||||||
|
37B76E97268747C900CE5671 /* ViewOptionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37B76E95268747C900CE5671 /* ViewOptionsView.swift */; };
|
||||||
|
37B76E98268747C900CE5671 /* ViewOptionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37B76E95268747C900CE5671 /* ViewOptionsView.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 */; };
|
||||||
@ -211,6 +214,7 @@
|
|||||||
37AAF29B26741B5F007FC770 /* SubscriptionVideosProvider.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SubscriptionVideosProvider.swift; sourceTree = "<group>"; };
|
37AAF29B26741B5F007FC770 /* SubscriptionVideosProvider.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SubscriptionVideosProvider.swift; sourceTree = "<group>"; };
|
||||||
37AAF29F26741C97007FC770 /* SubscriptionsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SubscriptionsView.swift; sourceTree = "<group>"; };
|
37AAF29F26741C97007FC770 /* SubscriptionsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SubscriptionsView.swift; sourceTree = "<group>"; };
|
||||||
37B767DA2677C3CA0098BAA8 /* PlayerState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerState.swift; sourceTree = "<group>"; };
|
37B767DA2677C3CA0098BAA8 /* PlayerState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerState.swift; sourceTree = "<group>"; };
|
||||||
|
37B76E95268747C900CE5671 /* ViewOptionsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ViewOptionsView.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>"; };
|
||||||
37C7A1DB267CE9D90010EAD6 /* Profile.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Profile.swift; sourceTree = "<group>"; };
|
37C7A1DB267CE9D90010EAD6 /* Profile.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Profile.swift; sourceTree = "<group>"; };
|
||||||
37C7A9032679059200E721B4 /* AVKeyValueStatus+String.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "AVKeyValueStatus+String.swift"; sourceTree = "<group>"; };
|
37C7A9032679059200E721B4 /* AVKeyValueStatus+String.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "AVKeyValueStatus+String.swift"; sourceTree = "<group>"; };
|
||||||
@ -395,6 +399,7 @@
|
|||||||
37F4AE7126828F0900BD60EA /* VideosCellsView.swift */,
|
37F4AE7126828F0900BD60EA /* VideosCellsView.swift */,
|
||||||
37AAF29926740A01007FC770 /* VideosListView.swift */,
|
37AAF29926740A01007FC770 /* VideosListView.swift */,
|
||||||
371231832683E62F0000B307 /* VideosView.swift */,
|
371231832683E62F0000B307 /* VideosView.swift */,
|
||||||
|
37B76E95268747C900CE5671 /* ViewOptionsView.swift */,
|
||||||
3705B17B267B4D9A00704544 /* VisualEffectView.swift */,
|
3705B17B267B4D9A00704544 /* VisualEffectView.swift */,
|
||||||
37D4B15E267164AF00C925CA /* Assets.xcassets */,
|
37D4B15E267164AF00C925CA /* Assets.xcassets */,
|
||||||
37D4B1AE26729DEB00C925CA /* Info.plist */,
|
37D4B1AE26729DEB00C925CA /* Info.plist */,
|
||||||
@ -726,6 +731,7 @@
|
|||||||
3714167B267AA1CF006CA35D /* TrendingCountriesProvider.swift in Sources */,
|
3714167B267AA1CF006CA35D /* TrendingCountriesProvider.swift in Sources */,
|
||||||
377FC7DF267A082200A6BBAF /* VideosListView.swift in Sources */,
|
377FC7DF267A082200A6BBAF /* VideosListView.swift in Sources */,
|
||||||
37D4B19726717E1500C925CA /* Video.swift in Sources */,
|
37D4B19726717E1500C925CA /* Video.swift in Sources */,
|
||||||
|
37B76E96268747C900CE5671 /* ViewOptionsView.swift in Sources */,
|
||||||
37D4B0E42671614900C925CA /* PearvidiousApp.swift in Sources */,
|
37D4B0E42671614900C925CA /* PearvidiousApp.swift in Sources */,
|
||||||
37CEE4B92677B63F005A1EFE /* StreamResolution.swift in Sources */,
|
37CEE4B92677B63F005A1EFE /* StreamResolution.swift in Sources */,
|
||||||
);
|
);
|
||||||
@ -764,6 +770,7 @@
|
|||||||
376578862685429C00D4EA09 /* CaseIterable+Next.swift in Sources */,
|
376578862685429C00D4EA09 /* CaseIterable+Next.swift in Sources */,
|
||||||
37F4AE7326828F0900BD60EA /* VideosCellsView.swift in Sources */,
|
37F4AE7326828F0900BD60EA /* VideosCellsView.swift in Sources */,
|
||||||
377FC7E0267A082600A6BBAF /* ChannelView.swift in Sources */,
|
377FC7E0267A082600A6BBAF /* ChannelView.swift in Sources */,
|
||||||
|
37B76E97268747C900CE5671 /* ViewOptionsView.swift in Sources */,
|
||||||
371231862683E7820000B307 /* VideosView.swift in Sources */,
|
371231862683E7820000B307 /* VideosView.swift in Sources */,
|
||||||
37C7A1D6267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */,
|
37C7A1D6267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */,
|
||||||
37C7A1DD267CE9D90010EAD6 /* Profile.swift in Sources */,
|
37C7A1DD267CE9D90010EAD6 /* Profile.swift in Sources */,
|
||||||
@ -842,6 +849,7 @@
|
|||||||
3705B184267B4E4900704544 /* TrendingCategory.swift in Sources */,
|
3705B184267B4E4900704544 /* TrendingCategory.swift in Sources */,
|
||||||
37AAF2A226741C97007FC770 /* SubscriptionsView.swift in Sources */,
|
37AAF2A226741C97007FC770 /* SubscriptionsView.swift in Sources */,
|
||||||
37D4B1812671653A00C925CA /* ContentView.swift in Sources */,
|
37D4B1812671653A00C925CA /* ContentView.swift in Sources */,
|
||||||
|
37B76E98268747C900CE5671 /* ViewOptionsView.swift in Sources */,
|
||||||
37AAF2842673791F007FC770 /* SearchedVideosProvider.swift in Sources */,
|
37AAF2842673791F007FC770 /* SearchedVideosProvider.swift in Sources */,
|
||||||
37CEE4BB2677B63F005A1EFE /* StreamResolution.swift in Sources */,
|
37CEE4BB2677B63F005A1EFE /* StreamResolution.swift in Sources */,
|
||||||
);
|
);
|
||||||
|
@ -2,6 +2,7 @@ import SwiftUI
|
|||||||
|
|
||||||
struct ContentView: View {
|
struct ContentView: View {
|
||||||
@ObservedObject private var state = AppState()
|
@ObservedObject private var state = AppState()
|
||||||
|
@ObservedObject private var profile = Profile()
|
||||||
|
|
||||||
@SceneStorage("tabSelection") var tabSelection = TabSelection.subscriptions
|
@SceneStorage("tabSelection") var tabSelection = TabSelection.subscriptions
|
||||||
|
|
||||||
@ -36,6 +37,7 @@ struct ContentView: View {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
.environmentObject(state)
|
.environmentObject(state)
|
||||||
|
.environmentObject(profile)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user