From 88a733da34ce769a1173800e11ec7de57491b753 Mon Sep 17 00:00:00 2001 From: Arkadiusz Fal Date: Thu, 8 Jul 2021 17:14:54 +0200 Subject: [PATCH] Creating playlists --- ...owView.swift => CoverSectionRowView.swift} | 8 +- ...ctionView.swift => CoverSectionView.swift} | 2 +- Apple TV/NewPlaylistView.swift | 74 +++++++++++++++++++ Apple TV/OptionsView.swift | 8 +- Apple TV/PlaylistsView.swift | 30 +++++++- Apple TV/SearchOptionsView.swift | 16 ++-- Model/InvidiousAPI.swift | 15 ++-- Model/PlaylistVisibility.swift | 13 ++++ Model/SearchQuery.swift | 4 +- Pearvidious.xcodeproj/project.pbxproj | 48 ++++++++---- 10 files changed, 175 insertions(+), 43 deletions(-) rename Apple TV/{OptionRowView.swift => CoverSectionRowView.swift} (54%) rename Apple TV/{OptionsSectionView.swift => CoverSectionView.swift} (93%) create mode 100644 Apple TV/NewPlaylistView.swift create mode 100644 Model/PlaylistVisibility.swift diff --git a/Apple TV/OptionRowView.swift b/Apple TV/CoverSectionRowView.swift similarity index 54% rename from Apple TV/OptionRowView.swift rename to Apple TV/CoverSectionRowView.swift index f94c1cd3..681a0549 100644 --- a/Apple TV/OptionRowView.swift +++ b/Apple TV/CoverSectionRowView.swift @@ -1,17 +1,17 @@ import SwiftUI -struct OptionRowView: View { - let label: String +struct CoverSectionRowView: View { + let label: String? let controlView: Content - init(_ label: String, @ViewBuilder controlView: @escaping () -> Content) { + init(_ label: String? = nil, @ViewBuilder controlView: @escaping () -> Content) { self.label = label self.controlView = controlView() } var body: some View { HStack { - Text(label) + Text(label ?? "") Spacer() controlView } diff --git a/Apple TV/OptionsSectionView.swift b/Apple TV/CoverSectionView.swift similarity index 93% rename from Apple TV/OptionsSectionView.swift rename to Apple TV/CoverSectionView.swift index e859965e..37113421 100644 --- a/Apple TV/OptionsSectionView.swift +++ b/Apple TV/CoverSectionView.swift @@ -1,6 +1,6 @@ import SwiftUI -struct OptionsSectionView: View { +struct CoverSectionView: View { let title: String? let rowsView: Content diff --git a/Apple TV/NewPlaylistView.swift b/Apple TV/NewPlaylistView.swift new file mode 100644 index 00000000..1c7ef71d --- /dev/null +++ b/Apple TV/NewPlaylistView.swift @@ -0,0 +1,74 @@ +import SwiftUI +import SwiftyJSON + +struct NewPlaylistView: View { + @State private var name = "" + @State private var visibility = PlaylistVisibility.public + + @State private var valid = false + + @Binding var createdPlaylist: Playlist? + + @Environment(\.dismiss) private var dismiss + + var body: some View { + HStack { + Spacer() + + VStack { + Spacer() + + CoverSectionView("New Playlist") { + CoverSectionRowView("Name") { + TextField("Playlist Name", text: $name, onCommit: validate) + .frame(maxWidth: 450) + } + + CoverSectionRowView("Visibility") { visibilityButton } + } + + CoverSectionRowView { + Button("Create", action: createPlaylistAndDismiss).disabled(!valid) + } + + Spacer() + } + .frame(maxWidth: 800) + + Spacer() + } + .background(.thinMaterial) + .onAppear { + createdPlaylist = nil + } + } + + func validate() { + valid = !name.isEmpty + } + + func createPlaylistAndDismiss() { + let resource = InvidiousAPI.shared.playlists + let body = ["title": name, "privacy": visibility.rawValue] + + resource.request(.post, json: body).onSuccess { response in + if let playlist: Playlist = response.typedContent() { + createdPlaylist = playlist + dismiss() + } + } + } + + var visibilityButton: some View { + Button(self.visibility.name) { + self.visibility = self.visibility.next() + } + .contextMenu { + ForEach(PlaylistVisibility.allCases) { visibility in + Button(visibility.name) { + self.visibility = visibility + } + } + } + } +} diff --git a/Apple TV/OptionsView.swift b/Apple TV/OptionsView.swift index 14eb4e0e..97212486 100644 --- a/Apple TV/OptionsView.swift +++ b/Apple TV/OptionsView.swift @@ -18,12 +18,12 @@ struct OptionsView: View { tabSelectionOptions - OptionsSectionView("View Options") { - OptionRowView("Show videos as") { nextLayoutButton } + CoverSectionView("View Options") { + CoverSectionRowView("Show videos as") { nextLayoutButton } } - OptionsSectionView(divider: false) { - OptionRowView("Close View Options") { Button("Close") { dismiss() } } + CoverSectionView(divider: false) { + CoverSectionRowView("Close View Options") { Button("Close") { dismiss() } } } Spacer() diff --git a/Apple TV/PlaylistsView.swift b/Apple TV/PlaylistsView.swift index cc2b49c0..358a64a9 100644 --- a/Apple TV/PlaylistsView.swift +++ b/Apple TV/PlaylistsView.swift @@ -6,6 +6,9 @@ struct PlaylistsView: View { @State private var selectedPlaylist: Playlist? + @State private var showingNewPlaylist = false + @State private var createdPlaylist: Playlist? + var resource: Resource { InvidiousAPI.shared.playlists } @@ -17,8 +20,12 @@ struct PlaylistsView: View { var body: some View { Section { VStack(alignment: .center, spacing: 2) { - selectPlaylistButton - .scaleEffect(0.85) + HStack { + selectPlaylistButton + + newPlaylistButton + } + .scaleEffect(0.85) if currentPlaylist != nil { VideosView(videos: currentPlaylist!.videos) @@ -27,11 +34,24 @@ struct PlaylistsView: View { } } } + .fullScreenCover(isPresented: $showingNewPlaylist, onDismiss: selectCreatedPlaylist) { + NewPlaylistView(createdPlaylist: $createdPlaylist) + } .onAppear { resource.loadIfNeeded() } } + func selectCreatedPlaylist() { + guard createdPlaylist != nil else { + return + } + + resource.load().onSuccess { _ in + self.selectedPlaylist = store.collection.first { $0 == createdPlaylist } + } + } + var currentPlaylist: Playlist? { selectedPlaylist ?? store.collection.first } @@ -52,4 +72,10 @@ struct PlaylistsView: View { } } } + + var newPlaylistButton: some View { + Button(action: { self.showingNewPlaylist = true }) { + Image(systemName: "plus") + } + } } diff --git a/Apple TV/SearchOptionsView.swift b/Apple TV/SearchOptionsView.swift index 5b6440a3..79b512d9 100644 --- a/Apple TV/SearchOptionsView.swift +++ b/Apple TV/SearchOptionsView.swift @@ -7,10 +7,10 @@ struct SearchOptionsView: View { @Default(.searchDuration) private var searchDuration var body: some View { - OptionsSectionView("Search Options") { - OptionRowView("Sort By") { searchSortOrderButton } - OptionRowView("Upload date") { searchDateButton } - OptionRowView("Duration") { searchDurationButton } + CoverSectionView("Search Options") { + CoverSectionRowView("Sort By") { searchSortOrderButton } + CoverSectionRowView("Upload date") { searchDateButton } + CoverSectionRowView("Duration") { searchDurationButton } } } @@ -47,19 +47,17 @@ struct SearchOptionsView: View { var searchDurationButton: some View { Button(self.searchDuration?.name ?? "All") { - let duration = Defaults[.searchDuration] - - Defaults[.searchDuration] = duration == nil ? SearchDuration.allCases.first : duration!.next(nilAtEnd: true) + self.searchDuration = self.searchDuration == nil ? SearchDuration.allCases.first : self.searchDuration!.next(nilAtEnd: true) } .contextMenu { ForEach(SearchDuration.allCases) { searchDuration in Button(searchDuration.name) { - Defaults[.searchDuration] = searchDuration + self.searchDuration = searchDuration } } Button("Reset") { - Defaults.reset(.searchDuration) + self.searchDuration = nil } } } diff --git a/Model/InvidiousAPI.swift b/Model/InvidiousAPI.swift index c2f38292..92449b30 100644 --- a/Model/InvidiousAPI.swift +++ b/Model/InvidiousAPI.swift @@ -54,6 +54,11 @@ final class InvidiousAPI: Service { content.json.arrayValue.map(Playlist.init) } + configureTransformer("/auth/playlists", requestMethods: [.post]) { (content: Entity) -> Playlist in + // hacky, to verify if possible to get it in easier way + Playlist(JSON(parseJSON: String(data: content.content, encoding: .utf8)!)) + } + configureTransformer("/auth/feed", requestMethods: [.get]) { (content: Entity) -> [Video] in if let feedVideos = content.json.dictionaryValue["videos"] { return feedVideos.arrayValue.map { Video($0) } @@ -108,12 +113,12 @@ final class InvidiousAPI: Service { .withParam("q", searchQuery(query.query)) .withParam("sort_by", query.sortBy.parameter) - if let date = query.date { - resource = resource.withParam("date", date.rawValue) + if let date = query.date?.rawValue { + resource = resource.withParam("date", date) } - if let duration = query.duration { - resource = resource.withParam("duration", duration.rawValue) + if let duration = query.duration?.rawValue { + resource = resource.withParam("duration", duration) } return resource @@ -135,6 +140,6 @@ final class InvidiousAPI: Service { searchQuery = id } - return searchQuery.addingPercentEncoding(withAllowedCharacters: .alphanumerics)! + return searchQuery } } diff --git a/Model/PlaylistVisibility.swift b/Model/PlaylistVisibility.swift new file mode 100644 index 00000000..ed152a1f --- /dev/null +++ b/Model/PlaylistVisibility.swift @@ -0,0 +1,13 @@ +import Foundation + +enum PlaylistVisibility: String, CaseIterable, Identifiable { + case `public`, unlisted, `private` + + var id: String { + rawValue + } + + var name: String { + rawValue.capitalized + } +} diff --git a/Model/SearchQuery.swift b/Model/SearchQuery.swift index dde2f5fc..d88af146 100644 --- a/Model/SearchQuery.swift +++ b/Model/SearchQuery.swift @@ -2,8 +2,8 @@ import Foundation final class SearchQuery: ObservableObject { @Published var query: String - @Published var sortBy = SearchSortOrder.relevance - @Published var date: SearchDate? = SearchDate.month + @Published var sortBy: SearchSortOrder = .relevance + @Published var date: SearchDate? = .month @Published var duration: SearchDuration? @Published var page = 1 diff --git a/Pearvidious.xcodeproj/project.pbxproj b/Pearvidious.xcodeproj/project.pbxproj index b3a66e64..d47d9c8b 100644 --- a/Pearvidious.xcodeproj/project.pbxproj +++ b/Pearvidious.xcodeproj/project.pbxproj @@ -30,12 +30,12 @@ 372915EA2687EBA500F5A35B /* ListingLayout.swift in Sources */ = {isa = PBXBuildFile; fileRef = 372915E92687EBA500F5A35B /* ListingLayout.swift */; }; 372915EB2687EBA500F5A35B /* ListingLayout.swift in Sources */ = {isa = PBXBuildFile; fileRef = 372915E92687EBA500F5A35B /* ListingLayout.swift */; }; 372915EC2687EBA500F5A35B /* ListingLayout.swift in Sources */ = {isa = PBXBuildFile; fileRef = 372915E92687EBA500F5A35B /* ListingLayout.swift */; }; - 373CFABE26966148003CB2C6 /* OptionsSectionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFABD26966115003CB2C6 /* OptionsSectionView.swift */; }; - 373CFABF26966149003CB2C6 /* OptionsSectionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFABD26966115003CB2C6 /* OptionsSectionView.swift */; }; - 373CFAC026966149003CB2C6 /* OptionsSectionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFABD26966115003CB2C6 /* OptionsSectionView.swift */; }; - 373CFAC226966159003CB2C6 /* OptionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* OptionRowView.swift */; }; - 373CFAC32696616C003CB2C6 /* OptionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* OptionRowView.swift */; }; - 373CFAC42696616C003CB2C6 /* OptionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* OptionRowView.swift */; }; + 373CFABE26966148003CB2C6 /* CoverSectionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFABD26966115003CB2C6 /* CoverSectionView.swift */; }; + 373CFABF26966149003CB2C6 /* CoverSectionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFABD26966115003CB2C6 /* CoverSectionView.swift */; }; + 373CFAC026966149003CB2C6 /* CoverSectionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFABD26966115003CB2C6 /* CoverSectionView.swift */; }; + 373CFAC226966159003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; }; + 373CFAC32696616C003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; }; + 373CFAC42696616C003CB2C6 /* CoverSectionRowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */; }; 373CFAC62696617C003CB2C6 /* SearchOptionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */; }; 373CFAC726966187003CB2C6 /* SearchOptionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */; }; 373CFAC926966188003CB2C6 /* SearchOptionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */; }; @@ -54,6 +54,12 @@ 373CFADB269663F1003CB2C6 /* Thumbnail.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFADA269663F1003CB2C6 /* Thumbnail.swift */; }; 373CFADC269663F1003CB2C6 /* Thumbnail.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFADA269663F1003CB2C6 /* Thumbnail.swift */; }; 373CFADD269663F1003CB2C6 /* Thumbnail.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFADA269663F1003CB2C6 /* Thumbnail.swift */; }; + 373CFADF2696F88B003CB2C6 /* NewPlaylistView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFADE2696F861003CB2C6 /* NewPlaylistView.swift */; }; + 373CFAE02696F88B003CB2C6 /* NewPlaylistView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFADE2696F861003CB2C6 /* NewPlaylistView.swift */; }; + 373CFAE12696F88B003CB2C6 /* NewPlaylistView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFADE2696F861003CB2C6 /* NewPlaylistView.swift */; }; + 373CFAE326974812003CB2C6 /* PlaylistVisibility.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAE226974812003CB2C6 /* PlaylistVisibility.swift */; }; + 373CFAE426974812003CB2C6 /* PlaylistVisibility.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAE226974812003CB2C6 /* PlaylistVisibility.swift */; }; + 373CFAE526974812003CB2C6 /* PlaylistVisibility.swift in Sources */ = {isa = PBXBuildFile; fileRef = 373CFAE226974812003CB2C6 /* PlaylistVisibility.swift */; }; 3741B5302676213400125C5E /* PlayerViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3741B52F2676213400125C5E /* PlayerViewController.swift */; }; 376578852685429C00D4EA09 /* CaseIterable+Next.swift in Sources */ = {isa = PBXBuildFile; fileRef = 376578842685429C00D4EA09 /* CaseIterable+Next.swift */; }; 376578862685429C00D4EA09 /* CaseIterable+Next.swift in Sources */ = {isa = PBXBuildFile; fileRef = 376578842685429C00D4EA09 /* CaseIterable+Next.swift */; }; @@ -215,14 +221,16 @@ 37141672267A8E10006CA35D /* Country.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Country.swift; sourceTree = ""; }; 372915E52687E3B900F5A35B /* Defaults.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Defaults.swift; sourceTree = ""; }; 372915E92687EBA500F5A35B /* ListingLayout.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ListingLayout.swift; sourceTree = ""; }; - 373CFABD26966115003CB2C6 /* OptionsSectionView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OptionsSectionView.swift; sourceTree = ""; }; - 373CFAC126966159003CB2C6 /* OptionRowView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OptionRowView.swift; sourceTree = ""; }; + 373CFABD26966115003CB2C6 /* CoverSectionView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CoverSectionView.swift; sourceTree = ""; }; + 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CoverSectionRowView.swift; sourceTree = ""; }; 373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SearchOptionsView.swift; sourceTree = ""; }; 373CFACA26966264003CB2C6 /* SearchQuery.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SearchQuery.swift; sourceTree = ""; }; 373CFACE26966290003CB2C6 /* SearchSortOrder.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SearchSortOrder.swift; sourceTree = ""; }; 373CFAD2269662AB003CB2C6 /* SearchDate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SearchDate.swift; sourceTree = ""; }; 373CFAD6269662CD003CB2C6 /* SearchDuration.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SearchDuration.swift; sourceTree = ""; }; 373CFADA269663F1003CB2C6 /* Thumbnail.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Thumbnail.swift; sourceTree = ""; }; + 373CFADE2696F861003CB2C6 /* NewPlaylistView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NewPlaylistView.swift; sourceTree = ""; }; + 373CFAE226974812003CB2C6 /* PlaylistVisibility.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlaylistVisibility.swift; sourceTree = ""; }; 3741B52F2676213400125C5E /* PlayerViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerViewController.swift; sourceTree = ""; }; 376578842685429C00D4EA09 /* CaseIterable+Next.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "CaseIterable+Next.swift"; sourceTree = ""; }; 376578882685471400D4EA09 /* Playlist.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Playlist.swift; sourceTree = ""; }; @@ -413,8 +421,9 @@ isa = PBXGroup; children = ( 37AAF2892673AB89007FC770 /* ChannelView.swift */, - 373CFAC126966159003CB2C6 /* OptionRowView.swift */, - 373CFABD26966115003CB2C6 /* OptionsSectionView.swift */, + 373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */, + 373CFABD26966115003CB2C6 /* CoverSectionView.swift */, + 373CFADE2696F861003CB2C6 /* NewPlaylistView.swift */, 37B76E95268747C900CE5671 /* OptionsView.swift */, 37D4B1822671681B00C925CA /* PlayerView.swift */, 3741B52F2676213400125C5E /* PlayerViewController.swift */, @@ -455,6 +464,7 @@ 37977582268922F600DD52A8 /* InvidiousAPI.swift */, 37B767DA2677C3CA0098BAA8 /* PlayerState.swift */, 376578882685471400D4EA09 /* Playlist.swift */, + 373CFAE226974812003CB2C6 /* PlaylistVisibility.swift */, 37C7A1DB267CE9D90010EAD6 /* Profile.swift */, 373CFAD2269662AB003CB2C6 /* SearchDate.swift */, 373CFAD6269662CD003CB2C6 /* SearchDuration.swift */, @@ -732,6 +742,7 @@ 37977583268922F600DD52A8 /* InvidiousAPI.swift in Sources */, 37CEE4C12677B697005A1EFE /* Stream.swift in Sources */, 37F4AE7226828F0900BD60EA /* VideosCellsView.swift in Sources */, + 373CFAE326974812003CB2C6 /* PlaylistVisibility.swift in Sources */, 376578852685429C00D4EA09 /* CaseIterable+Next.swift in Sources */, 37D4B0E62671614900C925CA /* ContentView.swift in Sources */, 377FC7DC267A081800A6BBAF /* PopularVideosView.swift in Sources */, @@ -742,9 +753,10 @@ 37EAD86F267B9ED100D9E01B /* Segment.swift in Sources */, 376578892685471400D4EA09 /* Playlist.swift in Sources */, 373CFADB269663F1003CB2C6 /* Thumbnail.swift in Sources */, + 373CFAE12696F88B003CB2C6 /* NewPlaylistView.swift in Sources */, 37CEE4B52677B628005A1EFE /* StreamType.swift in Sources */, 37C7A1DC267CE9D90010EAD6 /* Profile.swift in Sources */, - 373CFAC026966149003CB2C6 /* OptionsSectionView.swift in Sources */, + 373CFAC026966149003CB2C6 /* CoverSectionView.swift in Sources */, 3714166F267A8ACC006CA35D /* TrendingView.swift in Sources */, 377FC7E3267A084A00A6BBAF /* VideoListRowView.swift in Sources */, 37AAF29026740715007FC770 /* Channel.swift in Sources */, @@ -761,7 +773,7 @@ 37B767DB2677C3CA0098BAA8 /* PlayerState.swift in Sources */, 373CFACB26966264003CB2C6 /* SearchQuery.swift in Sources */, 372915EA2687EBA500F5A35B /* ListingLayout.swift in Sources */, - 373CFAC226966159003CB2C6 /* OptionRowView.swift in Sources */, + 373CFAC226966159003CB2C6 /* CoverSectionRowView.swift in Sources */, 37141673267A8E10006CA35D /* Country.swift in Sources */, 37AAF2A026741C97007FC770 /* SubscriptionsView.swift in Sources */, 373CFAD7269662CD003CB2C6 /* SearchDuration.swift in Sources */, @@ -783,7 +795,7 @@ 37CEE4BE2677B670005A1EFE /* AudioVideoStream.swift in Sources */, 37B17DA5268A285E006AEE9B /* VideoDetailsView.swift in Sources */, 37F4AE772682908700BD60EA /* VideoCellView.swift in Sources */, - 373CFABF26966149003CB2C6 /* OptionsSectionView.swift in Sources */, + 373CFABF26966149003CB2C6 /* CoverSectionView.swift in Sources */, 37141669267A83F9006CA35D /* StreamAVPlayerViewController.swift in Sources */, 37EAD86C267B9C5600D9E01B /* SponsorBlockAPI.swift in Sources */, 377FC7E7267A085600A6BBAF /* PlayerView.swift in Sources */, @@ -798,7 +810,7 @@ 377FC7E2267A084A00A6BBAF /* VideoListRowView.swift in Sources */, 3765788A2685471400D4EA09 /* Playlist.swift in Sources */, 373CFACC26966264003CB2C6 /* SearchQuery.swift in Sources */, - 373CFAC32696616C003CB2C6 /* OptionRowView.swift in Sources */, + 373CFAC32696616C003CB2C6 /* CoverSectionRowView.swift in Sources */, 37AAF29126740715007FC770 /* Channel.swift in Sources */, 373CFAC726966187003CB2C6 /* SearchOptionsView.swift in Sources */, 37AAF2952674086B007FC770 /* TabSelection.swift in Sources */, @@ -824,7 +836,9 @@ 377FC7DE267A082100A6BBAF /* VideosListView.swift in Sources */, 37D4B19826717E1500C925CA /* Video.swift in Sources */, 37D4B0E52671614900C925CA /* PearvidiousApp.swift in Sources */, + 373CFAE426974812003CB2C6 /* PlaylistVisibility.swift in Sources */, 37CEE4BA2677B63F005A1EFE /* StreamResolution.swift in Sources */, + 373CFAE02696F88B003CB2C6 /* NewPlaylistView.swift in Sources */, 37977584268922F600DD52A8 /* InvidiousAPI.swift in Sources */, 37B17DA1268A1F89006AEE9B /* VideoContextMenuView.swift in Sources */, 373CFAD4269662AB003CB2C6 /* SearchDate.swift in Sources */, @@ -861,6 +875,7 @@ 37977585268922F600DD52A8 /* InvidiousAPI.swift in Sources */, 37F4AE7426828F0900BD60EA /* VideosCellsView.swift in Sources */, 376578872685429C00D4EA09 /* CaseIterable+Next.swift in Sources */, + 373CFAE526974812003CB2C6 /* PlaylistVisibility.swift in Sources */, 37D4B1842671684E00C925CA /* PlayerView.swift in Sources */, 37D4B1802671650A00C925CA /* PearvidiousApp.swift in Sources */, 371231852683E7820000B307 /* VideosView.swift in Sources */, @@ -871,9 +886,10 @@ 37EAD86D267B9C5600D9E01B /* SponsorBlockAPI.swift in Sources */, 3765788B2685471400D4EA09 /* Playlist.swift in Sources */, 373CFADD269663F1003CB2C6 /* Thumbnail.swift in Sources */, + 373CFADF2696F88B003CB2C6 /* NewPlaylistView.swift in Sources */, 37C7A1DE267CE9D90010EAD6 /* Profile.swift in Sources */, 3741B5302676213400125C5E /* PlayerViewController.swift in Sources */, - 373CFABE26966148003CB2C6 /* OptionsSectionView.swift in Sources */, + 373CFABE26966148003CB2C6 /* CoverSectionView.swift in Sources */, 37B767DD2677C3CA0098BAA8 /* PlayerState.swift in Sources */, 37D4B18E26717B3800C925CA /* VideoListRowView.swift in Sources */, 37AAF27E26737323007FC770 /* PopularVideosView.swift in Sources */, @@ -890,7 +906,7 @@ 3705B180267B4DFB00704544 /* TrendingCountrySelectionView.swift in Sources */, 373CFACD26966264003CB2C6 /* SearchQuery.swift in Sources */, 37141675267A8E10006CA35D /* Country.swift in Sources */, - 373CFAC42696616C003CB2C6 /* OptionRowView.swift in Sources */, + 373CFAC42696616C003CB2C6 /* CoverSectionRowView.swift in Sources */, 372915EC2687EBA500F5A35B /* ListingLayout.swift in Sources */, 37D4B19926717E1500C925CA /* Video.swift in Sources */, 373CFAD9269662CD003CB2C6 /* SearchDuration.swift in Sources */,