mirror of
https://github.com/yattee/yattee.git
synced 2025-01-08 22:07:10 +00:00
Forms improvements for trending and playlists
This commit is contained in:
parent
2942119136
commit
09c3947fef
7
Fixtures/Playlist+Fixtures.swift
Normal file
7
Fixtures/Playlist+Fixtures.swift
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
import Foundation
|
||||||
|
|
||||||
|
extension Playlist {
|
||||||
|
static var fixture: Playlist {
|
||||||
|
Playlist(id: "ABC", title: "The Playlist", visibility: .public, updated: 1)
|
||||||
|
}
|
||||||
|
}
|
@ -22,6 +22,13 @@ struct Playlist: Identifiable, Equatable, Hashable {
|
|||||||
|
|
||||||
var videos = [Video]()
|
var videos = [Video]()
|
||||||
|
|
||||||
|
init(id: String, title: String, visibility: Visibility, updated: TimeInterval) {
|
||||||
|
self.id = id
|
||||||
|
self.title = title
|
||||||
|
self.visibility = visibility
|
||||||
|
self.updated = updated
|
||||||
|
}
|
||||||
|
|
||||||
init(_ json: JSON) {
|
init(_ json: JSON) {
|
||||||
id = json["playlistId"].stringValue
|
id = json["playlistId"].stringValue
|
||||||
title = json["title"].stringValue
|
title = json["title"].stringValue
|
||||||
|
@ -174,6 +174,9 @@
|
|||||||
37EAD86F267B9ED100D9E01B /* Segment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37EAD86E267B9ED100D9E01B /* Segment.swift */; };
|
37EAD86F267B9ED100D9E01B /* Segment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37EAD86E267B9ED100D9E01B /* Segment.swift */; };
|
||||||
37EAD870267B9ED100D9E01B /* Segment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37EAD86E267B9ED100D9E01B /* Segment.swift */; };
|
37EAD870267B9ED100D9E01B /* Segment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37EAD86E267B9ED100D9E01B /* Segment.swift */; };
|
||||||
37EAD871267B9ED100D9E01B /* Segment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37EAD86E267B9ED100D9E01B /* Segment.swift */; };
|
37EAD871267B9ED100D9E01B /* Segment.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37EAD86E267B9ED100D9E01B /* Segment.swift */; };
|
||||||
|
37F49BA326CAA59B00304AC0 /* Playlist+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F49BA226CAA59B00304AC0 /* Playlist+Fixtures.swift */; };
|
||||||
|
37F49BA426CAA59B00304AC0 /* Playlist+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F49BA226CAA59B00304AC0 /* Playlist+Fixtures.swift */; };
|
||||||
|
37F49BA526CAA59B00304AC0 /* Playlist+Fixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F49BA226CAA59B00304AC0 /* Playlist+Fixtures.swift */; };
|
||||||
37F4AE7226828F0900BD60EA /* VideosCellsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F4AE7126828F0900BD60EA /* VideosCellsView.swift */; };
|
37F4AE7226828F0900BD60EA /* VideosCellsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F4AE7126828F0900BD60EA /* VideosCellsView.swift */; };
|
||||||
37F4AE7326828F0900BD60EA /* VideosCellsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F4AE7126828F0900BD60EA /* VideosCellsView.swift */; };
|
37F4AE7326828F0900BD60EA /* VideosCellsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F4AE7126828F0900BD60EA /* VideosCellsView.swift */; };
|
||||||
37F4AE7426828F0900BD60EA /* VideosCellsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F4AE7126828F0900BD60EA /* VideosCellsView.swift */; };
|
37F4AE7426828F0900BD60EA /* VideosCellsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 37F4AE7126828F0900BD60EA /* VideosCellsView.swift */; };
|
||||||
@ -270,6 +273,7 @@
|
|||||||
37D4B1AE26729DEB00C925CA /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
|
37D4B1AE26729DEB00C925CA /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
|
||||||
37EAD86A267B9C5600D9E01B /* SponsorBlockAPI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SponsorBlockAPI.swift; sourceTree = "<group>"; };
|
37EAD86A267B9C5600D9E01B /* SponsorBlockAPI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SponsorBlockAPI.swift; sourceTree = "<group>"; };
|
||||||
37EAD86E267B9ED100D9E01B /* Segment.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Segment.swift; sourceTree = "<group>"; };
|
37EAD86E267B9ED100D9E01B /* Segment.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Segment.swift; sourceTree = "<group>"; };
|
||||||
|
37F49BA226CAA59B00304AC0 /* Playlist+Fixtures.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Playlist+Fixtures.swift"; sourceTree = "<group>"; };
|
||||||
37F4AE7126828F0900BD60EA /* VideosCellsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VideosCellsView.swift; sourceTree = "<group>"; };
|
37F4AE7126828F0900BD60EA /* VideosCellsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VideosCellsView.swift; sourceTree = "<group>"; };
|
||||||
/* End PBXFileReference section */
|
/* End PBXFileReference section */
|
||||||
|
|
||||||
@ -340,6 +344,7 @@
|
|||||||
children = (
|
children = (
|
||||||
3748186926A764FB0084E870 /* Thumbnail+Fixtures.swift */,
|
3748186926A764FB0084E870 /* Thumbnail+Fixtures.swift */,
|
||||||
3748186526A7627F0084E870 /* Video+Fixtures.swift */,
|
3748186526A7627F0084E870 /* Video+Fixtures.swift */,
|
||||||
|
37F49BA226CAA59B00304AC0 /* Playlist+Fixtures.swift */,
|
||||||
);
|
);
|
||||||
path = Fixtures;
|
path = Fixtures;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
@ -394,11 +399,14 @@
|
|||||||
37D4B0C32671614700C925CA /* AppTabNavigation.swift */,
|
37D4B0C32671614700C925CA /* AppTabNavigation.swift */,
|
||||||
37BD07B42698AA4D003EBB87 /* ContentView.swift */,
|
37BD07B42698AA4D003EBB87 /* ContentView.swift */,
|
||||||
37141672267A8E10006CA35D /* Country.swift */,
|
37141672267A8E10006CA35D /* Country.swift */,
|
||||||
|
373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */,
|
||||||
|
373CFABD26966115003CB2C6 /* CoverSectionView.swift */,
|
||||||
372915E52687E3B900F5A35B /* Defaults.swift */,
|
372915E52687E3B900F5A35B /* Defaults.swift */,
|
||||||
3748186D26A769D60084E870 /* DetailBadge.swift */,
|
3748186D26A769D60084E870 /* DetailBadge.swift */,
|
||||||
37D4B0C22671614700C925CA /* PearvidiousApp.swift */,
|
37D4B0C22671614700C925CA /* PearvidiousApp.swift */,
|
||||||
37BE0BD226A1D4780092E2DB /* Player.swift */,
|
37BE0BD226A1D4780092E2DB /* Player.swift */,
|
||||||
37BE0BD526A1D4A90092E2DB /* PlayerViewController.swift */,
|
37BE0BD526A1D4A90092E2DB /* PlayerViewController.swift */,
|
||||||
|
373CFAEA26975CBF003CB2C6 /* PlaylistFormView.swift */,
|
||||||
376578902685490700D4EA09 /* PlaylistsView.swift */,
|
376578902685490700D4EA09 /* PlaylistsView.swift */,
|
||||||
37AAF27D26737323007FC770 /* PopularView.swift */,
|
37AAF27D26737323007FC770 /* PopularView.swift */,
|
||||||
37AAF27F26737550007FC770 /* SearchView.swift */,
|
37AAF27F26737550007FC770 /* SearchView.swift */,
|
||||||
@ -450,10 +458,7 @@
|
|||||||
children = (
|
children = (
|
||||||
373CFAEE2697A78B003CB2C6 /* AddToPlaylistView.swift */,
|
373CFAEE2697A78B003CB2C6 /* AddToPlaylistView.swift */,
|
||||||
37AAF2892673AB89007FC770 /* ChannelView.swift */,
|
37AAF2892673AB89007FC770 /* ChannelView.swift */,
|
||||||
373CFAC126966159003CB2C6 /* CoverSectionRowView.swift */,
|
|
||||||
373CFABD26966115003CB2C6 /* CoverSectionView.swift */,
|
|
||||||
37B76E95268747C900CE5671 /* OptionsView.swift */,
|
37B76E95268747C900CE5671 /* OptionsView.swift */,
|
||||||
373CFAEA26975CBF003CB2C6 /* PlaylistFormView.swift */,
|
|
||||||
373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */,
|
373CFAC52696617C003CB2C6 /* SearchOptionsView.swift */,
|
||||||
37BAB54B269B39FD00E75ED1 /* TVNavigationView.swift */,
|
37BAB54B269B39FD00E75ED1 /* TVNavigationView.swift */,
|
||||||
37B17D9F268A1F25006AEE9B /* VideoContextMenuView.swift */,
|
37B17D9F268A1F25006AEE9B /* VideoContextMenuView.swift */,
|
||||||
@ -778,6 +783,7 @@
|
|||||||
379775932689365600DD52A8 /* Array+Next.swift in Sources */,
|
379775932689365600DD52A8 /* Array+Next.swift in Sources */,
|
||||||
377FC7E1267A082600A6BBAF /* ChannelView.swift in Sources */,
|
377FC7E1267A082600A6BBAF /* ChannelView.swift in Sources */,
|
||||||
37C7A1D5267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */,
|
37C7A1D5267BFD9D0010EAD6 /* SponsorBlockSegment.swift in Sources */,
|
||||||
|
37F49BA326CAA59B00304AC0 /* Playlist+Fixtures.swift in Sources */,
|
||||||
37B767DB2677C3CA0098BAA8 /* PlayerState.swift in Sources */,
|
37B767DB2677C3CA0098BAA8 /* PlayerState.swift in Sources */,
|
||||||
373CFACB26966264003CB2C6 /* SearchQuery.swift in Sources */,
|
373CFACB26966264003CB2C6 /* SearchQuery.swift in Sources */,
|
||||||
373CFAC226966159003CB2C6 /* CoverSectionRowView.swift in Sources */,
|
373CFAC226966159003CB2C6 /* CoverSectionRowView.swift in Sources */,
|
||||||
@ -808,6 +814,7 @@
|
|||||||
377FC7DD267A081A00A6BBAF /* PopularView.swift in Sources */,
|
377FC7DD267A081A00A6BBAF /* PopularView.swift in Sources */,
|
||||||
3705B183267B4E4900704544 /* TrendingCategory.swift in Sources */,
|
3705B183267B4E4900704544 /* TrendingCategory.swift in Sources */,
|
||||||
37BD07C32698AD4F003EBB87 /* ContentView.swift in Sources */,
|
37BD07C32698AD4F003EBB87 /* ContentView.swift in Sources */,
|
||||||
|
37F49BA426CAA59B00304AC0 /* Playlist+Fixtures.swift in Sources */,
|
||||||
37EAD870267B9ED100D9E01B /* Segment.swift in Sources */,
|
37EAD870267B9ED100D9E01B /* Segment.swift in Sources */,
|
||||||
37141670267A8ACC006CA35D /* TrendingView.swift in Sources */,
|
37141670267A8ACC006CA35D /* TrendingView.swift in Sources */,
|
||||||
377FC7E2267A084A00A6BBAF /* VideoView.swift in Sources */,
|
377FC7E2267A084A00A6BBAF /* VideoView.swift in Sources */,
|
||||||
@ -872,6 +879,7 @@
|
|||||||
37754C9F26B7500000DBD602 /* VideosView.swift in Sources */,
|
37754C9F26B7500000DBD602 /* VideosView.swift in Sources */,
|
||||||
37AAF28026737550007FC770 /* SearchView.swift in Sources */,
|
37AAF28026737550007FC770 /* SearchView.swift in Sources */,
|
||||||
37EAD871267B9ED100D9E01B /* Segment.swift in Sources */,
|
37EAD871267B9ED100D9E01B /* Segment.swift in Sources */,
|
||||||
|
37F49BA526CAA59B00304AC0 /* Playlist+Fixtures.swift in Sources */,
|
||||||
37CEE4BF2677B670005A1EFE /* SingleAssetStream.swift in Sources */,
|
37CEE4BF2677B670005A1EFE /* SingleAssetStream.swift in Sources */,
|
||||||
37BE0BD426A1D47D0092E2DB /* Player.swift in Sources */,
|
37BE0BD426A1D47D0092E2DB /* Player.swift in Sources */,
|
||||||
37977585268922F600DD52A8 /* InvidiousAPI.swift in Sources */,
|
37977585268922F600DD52A8 /* InvidiousAPI.swift in Sources */,
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
import SwiftUI
|
import SwiftUI
|
||||||
|
|
||||||
struct CoverSectionRowView<Content: View>: View {
|
struct CoverSectionRowView<ControlContent: View>: View {
|
||||||
let label: String?
|
let label: String?
|
||||||
let controlView: Content
|
let controlView: ControlContent
|
||||||
|
|
||||||
init(_ label: String? = nil, @ViewBuilder controlView: @escaping () -> Content) {
|
init(_ label: String? = nil, @ViewBuilder controlView: @escaping () -> ControlContent) {
|
||||||
self.label = label
|
self.label = label
|
||||||
self.controlView = controlView()
|
self.controlView = controlView()
|
||||||
}
|
}
|
||||||
@ -12,6 +12,7 @@ struct CoverSectionRowView<Content: View>: View {
|
|||||||
var body: some View {
|
var body: some View {
|
||||||
HStack {
|
HStack {
|
||||||
Text(label ?? "")
|
Text(label ?? "")
|
||||||
|
|
||||||
Spacer()
|
Spacer()
|
||||||
controlView
|
controlView
|
||||||
}
|
}
|
@ -42,7 +42,11 @@ struct CoverSectionView<Content: View>: View {
|
|||||||
|
|
||||||
var sectionTitle: some View {
|
var sectionTitle: some View {
|
||||||
Text(title ?? "")
|
Text(title ?? "")
|
||||||
.font(.title3)
|
|
||||||
|
.font(.title2)
|
||||||
|
#if os(macOS)
|
||||||
|
.bold()
|
||||||
|
#endif
|
||||||
.padding(.bottom)
|
.padding(.bottom)
|
||||||
}
|
}
|
||||||
}
|
}
|
194
Shared/PlaylistFormView.swift
Normal file
194
Shared/PlaylistFormView.swift
Normal file
@ -0,0 +1,194 @@
|
|||||||
|
import Siesta
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct PlaylistFormView: View {
|
||||||
|
@State private var name = ""
|
||||||
|
@State private var visibility = Playlist.Visibility.public
|
||||||
|
|
||||||
|
@State private var valid = false
|
||||||
|
@State private var showingDeleteConfirmation = false
|
||||||
|
|
||||||
|
@Binding var playlist: Playlist!
|
||||||
|
|
||||||
|
@Environment(\.dismiss) private var dismiss
|
||||||
|
|
||||||
|
var editing: Bool {
|
||||||
|
playlist != nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
#if os(macOS) || os(iOS)
|
||||||
|
VStack(alignment: .leading) {
|
||||||
|
HStack(alignment: .center) {
|
||||||
|
Text(editing ? "Edit Playlist" : "Create Playlist")
|
||||||
|
.font(.title2.bold())
|
||||||
|
|
||||||
|
Spacer()
|
||||||
|
|
||||||
|
Button("Cancel") {
|
||||||
|
dismiss()
|
||||||
|
}.keyboardShortcut(.cancelAction)
|
||||||
|
}
|
||||||
|
Form {
|
||||||
|
TextField("Name", text: $name, onCommit: validate)
|
||||||
|
.frame(maxWidth: 450)
|
||||||
|
.padding(.leading, 10)
|
||||||
|
|
||||||
|
Picker("Visibility", selection: $visibility) {
|
||||||
|
ForEach(Playlist.Visibility.allCases, id: \.self) { visibility in
|
||||||
|
Text(visibility.name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.pickerStyle(SegmentedPickerStyle())
|
||||||
|
}
|
||||||
|
Divider()
|
||||||
|
.padding(.vertical, 4)
|
||||||
|
HStack {
|
||||||
|
if editing {
|
||||||
|
deletePlaylistButton
|
||||||
|
}
|
||||||
|
|
||||||
|
Spacer()
|
||||||
|
|
||||||
|
Button("Save", action: submitForm).disabled(!valid)
|
||||||
|
.keyboardShortcut(.defaultAction)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.onChange(of: name) { _ in validate() }
|
||||||
|
.onAppear(perform: setFieldsFromPlaylist)
|
||||||
|
.padding(.horizontal)
|
||||||
|
#if !os(iOS)
|
||||||
|
.frame(width: 400, height: 150)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#else
|
||||||
|
HStack {
|
||||||
|
Spacer()
|
||||||
|
|
||||||
|
VStack {
|
||||||
|
Spacer()
|
||||||
|
|
||||||
|
CoverSectionView(editing ? "Edit Playlist" : "Create Playlist") {
|
||||||
|
CoverSectionRowView("Name") {
|
||||||
|
TextField("Playlist Name", text: $name, onCommit: validate)
|
||||||
|
.frame(maxWidth: 450)
|
||||||
|
}
|
||||||
|
|
||||||
|
CoverSectionRowView("Visibility") { visibilityButton }
|
||||||
|
}
|
||||||
|
|
||||||
|
CoverSectionRowView {
|
||||||
|
Button("Save", action: submitForm).disabled(!valid)
|
||||||
|
}
|
||||||
|
|
||||||
|
if editing {
|
||||||
|
CoverSectionView("Delete Playlist", divider: false, inline: true) { deletePlaylistButton }
|
||||||
|
.padding(.top, 50)
|
||||||
|
}
|
||||||
|
|
||||||
|
Spacer()
|
||||||
|
}
|
||||||
|
.frame(maxWidth: 800)
|
||||||
|
|
||||||
|
Spacer()
|
||||||
|
}
|
||||||
|
.background(.thinMaterial)
|
||||||
|
.onAppear {
|
||||||
|
guard editing else {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
self.name = self.playlist.title
|
||||||
|
self.visibility = self.playlist.visibility
|
||||||
|
|
||||||
|
validate()
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
func setFieldsFromPlaylist() {
|
||||||
|
guard editing else {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
name = playlist.title
|
||||||
|
visibility = playlist.visibility
|
||||||
|
|
||||||
|
validate()
|
||||||
|
}
|
||||||
|
|
||||||
|
func validate() {
|
||||||
|
valid = !name.isEmpty
|
||||||
|
}
|
||||||
|
|
||||||
|
func submitForm() {
|
||||||
|
guard valid else {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
let body = ["title": name, "privacy": visibility.rawValue]
|
||||||
|
|
||||||
|
resource.request(editing ? .patch : .post, json: body).onSuccess { response in
|
||||||
|
if let modifiedPlaylist: Playlist = response.typedContent() {
|
||||||
|
playlist = modifiedPlaylist
|
||||||
|
}
|
||||||
|
|
||||||
|
dismiss()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var resource: Resource {
|
||||||
|
editing ? InvidiousAPI.shared.playlist(playlist.id) : InvidiousAPI.shared.playlists
|
||||||
|
}
|
||||||
|
|
||||||
|
var visibilityButton: some View {
|
||||||
|
#if os(macOS)
|
||||||
|
Picker("Visibility", selection: $visibility) {
|
||||||
|
ForEach(Playlist.Visibility.allCases) { visibility in
|
||||||
|
Text(visibility.name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
Button(self.visibility.name) {
|
||||||
|
self.visibility = self.visibility.next()
|
||||||
|
}
|
||||||
|
.contextMenu {
|
||||||
|
ForEach(Playlist.Visibility.allCases) { visibility in
|
||||||
|
Button(visibility.name) {
|
||||||
|
self.visibility = visibility
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
var deletePlaylistButton: some View {
|
||||||
|
Button("Delete", role: .destructive) {
|
||||||
|
showingDeleteConfirmation = true
|
||||||
|
}.alert(isPresented: $showingDeleteConfirmation) {
|
||||||
|
Alert(
|
||||||
|
title: Text("Are you sure you want to delete playlist?"),
|
||||||
|
message: Text("Playlist \"\(playlist.title)\" will be deleted.\nIt cannot be undone."),
|
||||||
|
primaryButton: .destructive(Text("Delete"), action: deletePlaylistAndDismiss),
|
||||||
|
secondaryButton: .cancel()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
#if os(macOS)
|
||||||
|
.foregroundColor(.red)
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
func deletePlaylistAndDismiss() {
|
||||||
|
let resource = InvidiousAPI.shared.playlist(playlist.id)
|
||||||
|
resource.request(.delete).onSuccess { _ in
|
||||||
|
playlist = nil
|
||||||
|
dismiss()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct PlaylistFormView_Previews: PreviewProvider {
|
||||||
|
static var previews: some View {
|
||||||
|
PlaylistFormView(playlist: .constant(Playlist.fixture))
|
||||||
|
}
|
||||||
|
}
|
@ -21,10 +21,88 @@ struct PlaylistsView: View {
|
|||||||
resource.addObserver(store)
|
resource.addObserver(store)
|
||||||
}
|
}
|
||||||
|
|
||||||
var body: some View {
|
var videos: [Video] {
|
||||||
Section {
|
currentPlaylist?.videos ?? []
|
||||||
VStack(alignment: .center, spacing: 2) {
|
}
|
||||||
|
|
||||||
|
var videosViewMaxHeight: CGFloat {
|
||||||
#if os(tvOS)
|
#if os(tvOS)
|
||||||
|
videos.isEmpty ? 150 : .infinity
|
||||||
|
#else
|
||||||
|
videos.isEmpty ? 0 : .infinity
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
VStack {
|
||||||
|
#if os(tvOS)
|
||||||
|
toolbar
|
||||||
|
.font(.system(size: 28))
|
||||||
|
|
||||||
|
#endif
|
||||||
|
if currentPlaylist != nil, videos.isEmpty {
|
||||||
|
hintText("Playlist is empty\n\nTap and hold on a video and then tap \"Add to Playlist\"")
|
||||||
|
} else if store.collection.isEmpty {
|
||||||
|
hintText("You have no playlists\n\nTap on \"New Playlist\" to create one")
|
||||||
|
} else {
|
||||||
|
VideosView(videos: videos)
|
||||||
|
}
|
||||||
|
|
||||||
|
#if os(iOS)
|
||||||
|
toolbar
|
||||||
|
.font(.system(size: 14))
|
||||||
|
.animation(nil)
|
||||||
|
.padding(.horizontal)
|
||||||
|
.padding(.vertical, 10)
|
||||||
|
.overlay(Divider().offset(x: 0, y: -2), alignment: .topTrailing)
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
#if os(tvOS)
|
||||||
|
.fullScreenCover(isPresented: $showingNewPlaylist, onDismiss: selectCreatedPlaylist) {
|
||||||
|
PlaylistFormView(playlist: $createdPlaylist)
|
||||||
|
}
|
||||||
|
.fullScreenCover(isPresented: $showingEditPlaylist, onDismiss: selectEditedPlaylist) {
|
||||||
|
PlaylistFormView(playlist: $editedPlaylist)
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
.sheet(isPresented: $showingNewPlaylist, onDismiss: selectCreatedPlaylist) {
|
||||||
|
PlaylistFormView(playlist: $createdPlaylist)
|
||||||
|
}
|
||||||
|
.sheet(isPresented: $showingEditPlaylist, onDismiss: selectEditedPlaylist) {
|
||||||
|
PlaylistFormView(playlist: $editedPlaylist)
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
.toolbar {
|
||||||
|
ToolbarItemGroup {
|
||||||
|
#if !os(iOS)
|
||||||
|
if !store.collection.isEmpty {
|
||||||
|
selectPlaylistButton
|
||||||
|
}
|
||||||
|
|
||||||
|
if currentPlaylist != nil {
|
||||||
|
editPlaylistButton
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
newPlaylistButton
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.onAppear {
|
||||||
|
resource.loadIfNeeded()?.onSuccess { _ in
|
||||||
|
selectPlaylist(selectedPlaylistID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#if !os(tvOS)
|
||||||
|
.navigationTitle("Playlists")
|
||||||
|
#elseif os(iOS)
|
||||||
|
.navigationBarItems(trailing: newPlaylistButton)
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
var scaledToolbar: some View {
|
||||||
|
toolbar.scaleEffect(0.85)
|
||||||
|
}
|
||||||
|
|
||||||
|
var toolbar: some View {
|
||||||
HStack {
|
HStack {
|
||||||
if store.collection.isEmpty {
|
if store.collection.isEmpty {
|
||||||
Text("No Playlists")
|
Text("No Playlists")
|
||||||
@ -36,50 +114,32 @@ struct PlaylistsView: View {
|
|||||||
selectPlaylistButton
|
selectPlaylistButton
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if os(iOS)
|
||||||
|
Spacer()
|
||||||
|
#endif
|
||||||
|
|
||||||
if currentPlaylist != nil {
|
if currentPlaylist != nil {
|
||||||
editPlaylistButton
|
editPlaylistButton
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if !os(iOS)
|
||||||
newPlaylistButton
|
newPlaylistButton
|
||||||
.padding(.leading, 40)
|
.padding(.leading, 40)
|
||||||
}
|
|
||||||
.scaleEffect(0.85)
|
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if currentPlaylist != nil {
|
func hintText(_ text: String) -> some View {
|
||||||
if currentPlaylist!.videos.isEmpty {
|
VStack {
|
||||||
Spacer()
|
Spacer()
|
||||||
|
Text(text)
|
||||||
Text("Playlist is empty\n\nTap and hold on a video and then tap \"Add to Playlist\"")
|
|
||||||
.foregroundColor(.secondary)
|
.foregroundColor(.secondary)
|
||||||
.multilineTextAlignment(.center)
|
.multilineTextAlignment(.center)
|
||||||
|
|
||||||
Spacer()
|
|
||||||
} else {
|
|
||||||
VideosView(videos: currentPlaylist!.videos)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Spacer()
|
Spacer()
|
||||||
}
|
}
|
||||||
}
|
.frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: .infinity)
|
||||||
}
|
#if os(macOS)
|
||||||
#if !os(macOS)
|
.background()
|
||||||
.fullScreenCover(isPresented: $showingNewPlaylist, onDismiss: selectCreatedPlaylist) {
|
|
||||||
PlaylistFormView(playlist: $createdPlaylist)
|
|
||||||
}
|
|
||||||
.fullScreenCover(isPresented: $showingEditPlaylist, onDismiss: selectEditedPlaylist) {
|
|
||||||
PlaylistFormView(playlist: $editedPlaylist)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
.onAppear {
|
|
||||||
resource.loadIfNeeded()?.onSuccess { _ in
|
|
||||||
selectPlaylist(selectedPlaylistID)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#if !os(tvOS)
|
|
||||||
.navigationTitle("Playlists")
|
|
||||||
#elseif os(iOS)
|
|
||||||
.navigationBarItems(trailing: newPlaylistButton)
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -116,6 +176,7 @@ struct PlaylistsView: View {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var selectPlaylistButton: some View {
|
var selectPlaylistButton: some View {
|
||||||
|
#if os(tvOS)
|
||||||
Button(currentPlaylist?.title ?? "Select playlist") {
|
Button(currentPlaylist?.title ?? "Select playlist") {
|
||||||
guard currentPlaylist != nil else {
|
guard currentPlaylist != nil else {
|
||||||
return
|
return
|
||||||
@ -130,6 +191,19 @@ struct PlaylistsView: View {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
Menu(currentPlaylist?.title ?? "Select playlist") {
|
||||||
|
ForEach(store.collection) { playlist in
|
||||||
|
Button(action: { selectPlaylist(playlist.id) }) {
|
||||||
|
if playlist == self.currentPlaylist {
|
||||||
|
Label(playlist.title, systemImage: "checkmark")
|
||||||
|
} else {
|
||||||
|
Text(playlist.title)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
var editPlaylistButton: some View {
|
var editPlaylistButton: some View {
|
||||||
@ -138,7 +212,7 @@ struct PlaylistsView: View {
|
|||||||
self.showingEditPlaylist = true
|
self.showingEditPlaylist = true
|
||||||
}) {
|
}) {
|
||||||
HStack(spacing: 8) {
|
HStack(spacing: 8) {
|
||||||
Image(systemName: "pencil")
|
Image(systemName: "slider.horizontal.3")
|
||||||
Text("Edit")
|
Text("Edit")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -155,3 +229,10 @@ struct PlaylistsView: View {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct PlaylistsView_Provider: PreviewProvider {
|
||||||
|
static var previews: some View {
|
||||||
|
PlaylistsView()
|
||||||
|
.environmentObject(NavigationState())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -18,7 +18,10 @@ struct TrendingCountrySelection: View {
|
|||||||
HStack {
|
HStack {
|
||||||
TextField("Country", text: $query, prompt: Text(TrendingCountrySelection.prompt))
|
TextField("Country", text: $query, prompt: Text(TrendingCountrySelection.prompt))
|
||||||
.focused($countryIsFocused)
|
.focused($countryIsFocused)
|
||||||
|
|
||||||
Button("Done") { selectCountryAndDismiss() }
|
Button("Done") { selectCountryAndDismiss() }
|
||||||
|
.keyboardShortcut(.defaultAction)
|
||||||
|
.keyboardShortcut(.cancelAction)
|
||||||
}
|
}
|
||||||
.padding([.horizontal, .top])
|
.padding([.horizontal, .top])
|
||||||
|
|
||||||
|
@ -188,7 +188,7 @@ struct VideoView: View {
|
|||||||
}
|
}
|
||||||
.frame(minWidth: 320, maxWidth: .infinity, minHeight: 180, maxHeight: .infinity)
|
.frame(minWidth: 320, maxWidth: .infinity, minHeight: 180, maxHeight: .infinity)
|
||||||
#if os(tvOS)
|
#if os(tvOS)
|
||||||
.frame(minHeight: 320)
|
.frame(minHeight: layout == .cells ? 320 : 200)
|
||||||
#endif
|
#endif
|
||||||
.aspectRatio(1.777, contentMode: .fit)
|
.aspectRatio(1.777, contentMode: .fit)
|
||||||
}
|
}
|
||||||
|
@ -22,11 +22,13 @@ struct VideosCellsView: View {
|
|||||||
.padding()
|
.padding()
|
||||||
}
|
}
|
||||||
.onChange(of: videos) { [videos] newVideos in
|
.onChange(of: videos) { [videos] newVideos in
|
||||||
|
#if !os(tvOS)
|
||||||
guard !videos.isEmpty, let video = newVideos.first else {
|
guard !videos.isEmpty, let video = newVideos.first else {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
scrollView.scrollTo(video.id, anchor: .top)
|
scrollView.scrollTo(video.id, anchor: .top)
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
#if os(tvOS)
|
#if os(tvOS)
|
||||||
.padding(.horizontal, 10)
|
.padding(.horizontal, 10)
|
||||||
@ -37,7 +39,7 @@ struct VideosCellsView: View {
|
|||||||
|
|
||||||
var items: [GridItem] {
|
var items: [GridItem] {
|
||||||
#if os(tvOS)
|
#if os(tvOS)
|
||||||
videos.count < 3 ? Array(repeating: GridItem(.fixed(540)), count: videos.count) : adaptiveItem
|
videos.count < 3 ? Array(repeating: GridItem(.fixed(540)), count: [videos.count, 1].max()!) : adaptiveItem
|
||||||
#else
|
#else
|
||||||
adaptiveItem
|
adaptiveItem
|
||||||
#endif
|
#endif
|
||||||
|
@ -13,11 +13,14 @@ struct VideosListView: View {
|
|||||||
.listRowInsets(EdgeInsets())
|
.listRowInsets(EdgeInsets())
|
||||||
}
|
}
|
||||||
.onChange(of: videos) { videos in
|
.onChange(of: videos) { videos in
|
||||||
|
#if !os(tvOS)
|
||||||
|
|
||||||
guard let video = videos.first else {
|
guard let video = videos.first else {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
scrollView.scrollTo(video.id, anchor: .top)
|
scrollView.scrollTo(video.id, anchor: .top)
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,120 +0,0 @@
|
|||||||
import Siesta
|
|
||||||
import SwiftUI
|
|
||||||
|
|
||||||
struct PlaylistFormView: View {
|
|
||||||
@State private var name = ""
|
|
||||||
@State private var visibility = Playlist.Visibility.public
|
|
||||||
|
|
||||||
@State private var valid = false
|
|
||||||
@State private var showingDeleteConfirmation = false
|
|
||||||
|
|
||||||
@Binding var playlist: Playlist!
|
|
||||||
|
|
||||||
@Environment(\.dismiss) private var dismiss
|
|
||||||
|
|
||||||
var editing: Bool {
|
|
||||||
playlist != nil
|
|
||||||
}
|
|
||||||
|
|
||||||
var body: some View {
|
|
||||||
HStack {
|
|
||||||
Spacer()
|
|
||||||
|
|
||||||
VStack {
|
|
||||||
Spacer()
|
|
||||||
|
|
||||||
CoverSectionView(editing ? "Edit Playlist" : "Create Playlist") {
|
|
||||||
CoverSectionRowView("Name") {
|
|
||||||
TextField("Playlist Name", text: $name, onCommit: validate)
|
|
||||||
.frame(maxWidth: 450)
|
|
||||||
}
|
|
||||||
|
|
||||||
CoverSectionRowView("Visibility") { visibilityButton }
|
|
||||||
}
|
|
||||||
|
|
||||||
CoverSectionRowView {
|
|
||||||
Button("Save", action: submitForm).disabled(!valid)
|
|
||||||
}
|
|
||||||
|
|
||||||
if editing {
|
|
||||||
CoverSectionView("Delete Playlist", divider: false, inline: true) { deletePlaylistButton }
|
|
||||||
.padding(.top, 50)
|
|
||||||
}
|
|
||||||
|
|
||||||
Spacer()
|
|
||||||
}
|
|
||||||
.frame(maxWidth: 800)
|
|
||||||
|
|
||||||
Spacer()
|
|
||||||
}
|
|
||||||
.background(.thinMaterial)
|
|
||||||
.onAppear {
|
|
||||||
guard editing else {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
self.name = self.playlist.title
|
|
||||||
self.visibility = self.playlist.visibility
|
|
||||||
|
|
||||||
validate()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func validate() {
|
|
||||||
valid = !name.isEmpty
|
|
||||||
}
|
|
||||||
|
|
||||||
func submitForm() {
|
|
||||||
guard valid else {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let body = ["title": name, "privacy": visibility.rawValue]
|
|
||||||
|
|
||||||
resource.request(editing ? .patch : .post, json: body).onSuccess { response in
|
|
||||||
if let createdPlaylist: Playlist = response.typedContent() {
|
|
||||||
playlist = createdPlaylist
|
|
||||||
}
|
|
||||||
|
|
||||||
dismiss()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var resource: Resource {
|
|
||||||
editing ? InvidiousAPI.shared.playlist(playlist.id) : InvidiousAPI.shared.playlists
|
|
||||||
}
|
|
||||||
|
|
||||||
var visibilityButton: some View {
|
|
||||||
Button(self.visibility.name) {
|
|
||||||
self.visibility = self.visibility.next()
|
|
||||||
}
|
|
||||||
.contextMenu {
|
|
||||||
ForEach(Playlist.Visibility.allCases) { visibility in
|
|
||||||
Button(visibility.name) {
|
|
||||||
self.visibility = visibility
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var deletePlaylistButton: some View {
|
|
||||||
Button("Delete", role: .destructive) {
|
|
||||||
showingDeleteConfirmation = true
|
|
||||||
}.alert(isPresented: $showingDeleteConfirmation) {
|
|
||||||
Alert(
|
|
||||||
title: Text("Are you sure you want to delete playlist?"),
|
|
||||||
message: Text("Playlist \"\(playlist.title)\" will be deleted.\nIt cannot be undone."),
|
|
||||||
primaryButton: .destructive(Text("Delete"), action: deletePlaylistAndDismiss),
|
|
||||||
secondaryButton: .cancel()
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func deletePlaylistAndDismiss() {
|
|
||||||
let resource = InvidiousAPI.shared.playlist(playlist.id)
|
|
||||||
resource.request(.delete).onSuccess { _ in
|
|
||||||
playlist = nil
|
|
||||||
dismiss()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user