flattened model into one playlist, added network functions on playlist change
This commit is contained in:
parent
228fdb3991
commit
3a68d3c892
@ -10,58 +10,130 @@ import Foundation
|
|||||||
import UIKit
|
import UIKit
|
||||||
import SwiftyJSON
|
import SwiftyJSON
|
||||||
|
|
||||||
class Playlist: Identifiable, Equatable, Codable {
|
class Playlist: Identifiable, Equatable, Codable, ObservableObject {
|
||||||
|
|
||||||
//MARK: Properties
|
//MARK: Properties
|
||||||
|
|
||||||
var name: String
|
@Published var name: String
|
||||||
var uri: String
|
@Published var uri: String
|
||||||
var username: String?
|
@Published var username: String?
|
||||||
|
|
||||||
var include_recommendations: Bool
|
@Published var type: String {
|
||||||
var recommendation_sample: Int
|
didSet {
|
||||||
var include_library_tracks: Bool
|
self.updatePlaylist(updates: JSON(["type": self.type]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var parts: Array<String>
|
@Published var include_recommendations: Bool {
|
||||||
var playlist_references: Array<String>
|
didSet {
|
||||||
var shuffle: Bool
|
self.updatePlaylist(updates: JSON(["include_recommendations": self.include_recommendations]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@Published var recommendation_sample: Int{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["recommendation_sample": self.recommendation_sample]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@Published var include_library_tracks: Bool{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["include_library_tracks": self.include_library_tracks]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var sort: String
|
@Published var parts: Array<String>{
|
||||||
var description_overwrite: String?
|
didSet {
|
||||||
var description_suffix: String?
|
self.updatePlaylist(updates: JSON(["parts": self.parts]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@Published var playlist_references: Array<String>{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["playlist_references": self.playlist_references]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@Published var shuffle: Bool{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["shuffle": self.shuffle]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var last_updated: String?
|
var sort: String?
|
||||||
|
@Published var description_overwrite: String?
|
||||||
|
@Published var description_suffix: String?
|
||||||
|
|
||||||
var lastfm_stat_count: Int
|
@Published var last_updated: String?
|
||||||
var lastfm_stat_album_count: Int
|
|
||||||
var lastfm_stat_artist_count: Int
|
|
||||||
|
|
||||||
var lastfm_stat_percent: Float
|
@Published var lastfm_stat_count: Int
|
||||||
|
@Published var lastfm_stat_album_count: Int
|
||||||
|
@Published var lastfm_stat_artist_count: Int
|
||||||
|
|
||||||
|
@Published var lastfm_stat_percent: Float
|
||||||
var lastfm_stat_percent_str: String {
|
var lastfm_stat_percent_str: String {
|
||||||
get {
|
get {
|
||||||
return String(format: "%.2f%%", lastfm_stat_percent)
|
return String(format: "%.2f%%", lastfm_stat_percent)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var lastfm_stat_album_percent: Float
|
@Published var lastfm_stat_album_percent: Float
|
||||||
var lastfm_stat_album_percent_str: String {
|
var lastfm_stat_album_percent_str: String {
|
||||||
get {
|
get {
|
||||||
return String(format: "%.2f%%", lastfm_stat_album_percent)
|
return String(format: "%.2f%%", lastfm_stat_album_percent)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var lastfm_stat_artist_percent: Float
|
@Published var lastfm_stat_artist_percent: Float
|
||||||
var lastfm_stat_artist_percent_str: String {
|
var lastfm_stat_artist_percent_str: String {
|
||||||
get {
|
get {
|
||||||
return String(format: "%.2f%%", lastfm_stat_artist_percent)
|
return String(format: "%.2f%%", lastfm_stat_artist_percent)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var lastfm_stat_last_refresh: String?
|
@Published var lastfm_stat_last_refresh: String?
|
||||||
|
|
||||||
|
@Published var add_last_month: Bool{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["add_last_month": self.add_last_month]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@Published var add_this_month: Bool{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["add_this_month": self.add_this_month]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@Published var day_boundary: Int{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["day_boundary": self.day_boundary]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Published var chart_range: LastFmRange{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["chart_range": self.chart_range.rawValue]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@Published var chart_limit: Int{
|
||||||
|
didSet {
|
||||||
|
self.updatePlaylist(updates: JSON(["chart_range": self.chart_range.rawValue]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func updatePlaylist(updates: JSON) {
|
||||||
|
let api = PlaylistApi.updatePlaylist(name: self.name, updates: updates)
|
||||||
|
RequestBuilder.buildRequest(apiRequest: api).responseJSON{ response in
|
||||||
|
switch response.result {
|
||||||
|
case .success:
|
||||||
|
break
|
||||||
|
case .failure:
|
||||||
|
debugPrint("error: \(self.name), \(updates)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//TODO: do better error checking
|
||||||
|
}
|
||||||
|
|
||||||
private enum CodingKeys: String, CodingKey {
|
private enum CodingKeys: String, CodingKey {
|
||||||
case name
|
case name
|
||||||
case uri
|
case uri
|
||||||
case username
|
case username
|
||||||
|
|
||||||
|
case type
|
||||||
|
|
||||||
case include_recommendations
|
case include_recommendations
|
||||||
case recommendation_sample
|
case recommendation_sample
|
||||||
case include_library_tracks
|
case include_library_tracks
|
||||||
@ -85,6 +157,13 @@ class Playlist: Identifiable, Equatable, Codable {
|
|||||||
case lastfm_stat_artist_percent
|
case lastfm_stat_artist_percent
|
||||||
|
|
||||||
case lastfm_stat_last_refresh
|
case lastfm_stat_last_refresh
|
||||||
|
|
||||||
|
case add_last_month
|
||||||
|
case add_this_month
|
||||||
|
case day_boundary
|
||||||
|
|
||||||
|
case chart_range
|
||||||
|
case chart_limit
|
||||||
}
|
}
|
||||||
|
|
||||||
//MARK: Initialization
|
//MARK: Initialization
|
||||||
@ -93,6 +172,8 @@ class Playlist: Identifiable, Equatable, Codable {
|
|||||||
uri: String = "spotify::",
|
uri: String = "spotify::",
|
||||||
username: String = "NO USER",
|
username: String = "NO USER",
|
||||||
|
|
||||||
|
type: String = "default",
|
||||||
|
|
||||||
include_recommendations: Bool = false,
|
include_recommendations: Bool = false,
|
||||||
recommendation_sample: Int = 0,
|
recommendation_sample: Int = 0,
|
||||||
include_library_tracks: Bool = false,
|
include_library_tracks: Bool = false,
|
||||||
@ -115,12 +196,21 @@ class Playlist: Identifiable, Equatable, Codable {
|
|||||||
lastfm_stat_album_percent: Float = 0,
|
lastfm_stat_album_percent: Float = 0,
|
||||||
lastfm_stat_artist_percent: Float = 0,
|
lastfm_stat_artist_percent: Float = 0,
|
||||||
|
|
||||||
lastfm_stat_last_refresh: String? = ""){
|
lastfm_stat_last_refresh: String? = "",
|
||||||
|
|
||||||
|
add_last_month: Bool = false,
|
||||||
|
add_this_month: Bool = false,
|
||||||
|
day_boundary: Int = 14,
|
||||||
|
|
||||||
|
chart_range: LastFmRange = .overall,
|
||||||
|
chart_limit: Int = 10){
|
||||||
|
|
||||||
self.name = name
|
self.name = name
|
||||||
self.uri = uri
|
self.uri = uri
|
||||||
self.username = username
|
self.username = username
|
||||||
|
|
||||||
|
self.type = type
|
||||||
|
|
||||||
self.last_updated = last_updated
|
self.last_updated = last_updated
|
||||||
|
|
||||||
self.lastfm_stat_count = lastfm_stat_count
|
self.lastfm_stat_count = lastfm_stat_count
|
||||||
@ -144,6 +234,13 @@ class Playlist: Identifiable, Equatable, Codable {
|
|||||||
self.sort = sort
|
self.sort = sort
|
||||||
self.description_overwrite = description_overwrite
|
self.description_overwrite = description_overwrite
|
||||||
self.description_suffix = description_suffix
|
self.description_suffix = description_suffix
|
||||||
|
|
||||||
|
self.add_last_month = add_last_month
|
||||||
|
self.add_this_month = add_this_month
|
||||||
|
self.day_boundary = day_boundary
|
||||||
|
|
||||||
|
self.chart_range = chart_range
|
||||||
|
self.chart_limit = chart_limit
|
||||||
}
|
}
|
||||||
|
|
||||||
var link: String {
|
var link: String {
|
||||||
@ -160,11 +257,26 @@ class Playlist: Identifiable, Equatable, Codable {
|
|||||||
|
|
||||||
name = try container.decode(String.self, forKey: .name)
|
name = try container.decode(String.self, forKey: .name)
|
||||||
uri = try container.decode(String.self, forKey: .uri)
|
uri = try container.decode(String.self, forKey: .uri)
|
||||||
// username = try container.decode(String.self, forKey: .username)
|
do{
|
||||||
|
username = try container.decode(String.self, forKey: .username)
|
||||||
|
}catch {
|
||||||
|
username = "NO USER"
|
||||||
|
debugPrint("failed to parse username")
|
||||||
|
}
|
||||||
|
|
||||||
// description_overwrite = try container.decode(String.self, forKey: .description_overwrite)
|
type = try container.decode(String.self, forKey: .type)
|
||||||
// description_suffix = try container.decode(String.self, forKey: .description_suffix)
|
|
||||||
|
|
||||||
|
do{
|
||||||
|
description_overwrite = try container.decode(String.self, forKey: .description_overwrite)
|
||||||
|
}catch {
|
||||||
|
debugPrint("no description overwrite")
|
||||||
|
}
|
||||||
|
|
||||||
|
do{
|
||||||
|
description_suffix = try container.decode(String.self, forKey: .description_suffix)
|
||||||
|
}catch {
|
||||||
|
debugPrint("no description suffix")
|
||||||
|
}
|
||||||
last_updated = try container.decode(String.self, forKey: .last_updated)
|
last_updated = try container.decode(String.self, forKey: .last_updated)
|
||||||
|
|
||||||
lastfm_stat_count = try container.decode(Int.self, forKey: .lastfm_stat_count)
|
lastfm_stat_count = try container.decode(Int.self, forKey: .lastfm_stat_count)
|
||||||
@ -179,55 +291,105 @@ class Playlist: Identifiable, Equatable, Codable {
|
|||||||
|
|
||||||
include_recommendations = try container.decode(Bool.self, forKey: .include_recommendations)
|
include_recommendations = try container.decode(Bool.self, forKey: .include_recommendations)
|
||||||
recommendation_sample = try container.decode(Int.self, forKey: .recommendation_sample)
|
recommendation_sample = try container.decode(Int.self, forKey: .recommendation_sample)
|
||||||
|
|
||||||
|
do{
|
||||||
include_library_tracks = try container.decode(Bool.self, forKey: .include_library_tracks)
|
include_library_tracks = try container.decode(Bool.self, forKey: .include_library_tracks)
|
||||||
|
}catch {
|
||||||
|
include_library_tracks = false
|
||||||
|
// debugPrint("failed to parse include_library_tracks")
|
||||||
|
}
|
||||||
|
|
||||||
parts = try container.decode([String].self, forKey: .parts)
|
parts = try container.decode([String].self, forKey: .parts)
|
||||||
playlist_references = try container.decode([String].self, forKey: .playlist_references)
|
playlist_references = try container.decode([String].self, forKey: .playlist_references)
|
||||||
shuffle = try container.decode(Bool.self, forKey: .shuffle)
|
shuffle = try container.decode(Bool.self, forKey: .shuffle)
|
||||||
|
|
||||||
|
do{
|
||||||
sort = try container.decode(String.self, forKey: .sort)
|
sort = try container.decode(String.self, forKey: .sort)
|
||||||
|
}catch {
|
||||||
|
sort = "release_date"
|
||||||
|
// debugPrint("failed to parse sort value")
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
do {
|
||||||
|
|
||||||
class RecentsPlaylist: Playlist {
|
|
||||||
|
|
||||||
//MARK: Properties
|
|
||||||
|
|
||||||
var add_last_month: Bool
|
|
||||||
var add_this_month: Bool
|
|
||||||
var day_boundary: Int
|
|
||||||
|
|
||||||
private enum CodingKeys: String, CodingKey { case add_last_month; case add_this_month; case day_boundary }
|
|
||||||
|
|
||||||
//MARK: Initialization
|
|
||||||
|
|
||||||
init(name: String,
|
|
||||||
username: String = "NO USER",
|
|
||||||
|
|
||||||
add_last_month: Bool = false,
|
|
||||||
add_this_month: Bool = false,
|
|
||||||
day_boundary: Int = 14){
|
|
||||||
|
|
||||||
self.add_last_month = add_last_month
|
|
||||||
self.add_this_month = add_this_month
|
|
||||||
self.day_boundary = day_boundary
|
|
||||||
|
|
||||||
super.init(name: name, username: username)
|
|
||||||
}
|
|
||||||
|
|
||||||
required init(from decoder: Decoder) throws {
|
|
||||||
let container = try decoder.container(keyedBy: CodingKeys.self)
|
|
||||||
|
|
||||||
add_last_month = try container.decode(Bool.self, forKey: .add_last_month)
|
add_last_month = try container.decode(Bool.self, forKey: .add_last_month)
|
||||||
|
}catch {
|
||||||
|
add_last_month = false
|
||||||
|
// debugPrint("failed to parse add last month")
|
||||||
|
}
|
||||||
|
|
||||||
|
do {
|
||||||
add_this_month = try container.decode(Bool.self, forKey: .add_this_month)
|
add_this_month = try container.decode(Bool.self, forKey: .add_this_month)
|
||||||
|
}catch {
|
||||||
|
add_this_month = false
|
||||||
|
// debugPrint("failed to parse add this month")
|
||||||
|
}
|
||||||
|
|
||||||
|
do {
|
||||||
day_boundary = try container.decode(Int.self, forKey: .day_boundary)
|
day_boundary = try container.decode(Int.self, forKey: .day_boundary)
|
||||||
|
}catch {
|
||||||
|
day_boundary = 21
|
||||||
|
// debugPrint("failed to parse day boundary")
|
||||||
|
}
|
||||||
|
|
||||||
try super.init(from: decoder)
|
do{
|
||||||
|
chart_range = try LastFmRange(rawValue: container.decode(String.self, forKey: .chart_range)) ?? LastFmRange.month
|
||||||
|
}catch {
|
||||||
|
chart_range = .halfyear
|
||||||
|
// debugPrint("failed to parse chart_range")
|
||||||
|
}
|
||||||
|
|
||||||
|
do{
|
||||||
|
chart_limit = try container.decode(Int.self, forKey: .chart_limit)
|
||||||
|
}catch {
|
||||||
|
chart_limit = 50
|
||||||
|
// debugPrint("failed to parse chart_limit")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
enum LastFmRange: String, Decodable {
|
func encode(to encoder: Encoder) throws {
|
||||||
|
var container = encoder.container(keyedBy: CodingKeys.self)
|
||||||
|
|
||||||
|
try container.encode(self.name, forKey: .name)
|
||||||
|
try container.encode(self.uri, forKey: .uri)
|
||||||
|
try container.encode(self.username, forKey: .username)
|
||||||
|
|
||||||
|
try container.encode(self.type, forKey: .type)
|
||||||
|
|
||||||
|
try container.encode(self.include_recommendations, forKey: .include_recommendations)
|
||||||
|
try container.encode(self.recommendation_sample, forKey: .recommendation_sample)
|
||||||
|
try container.encode(self.include_library_tracks, forKey: .include_library_tracks)
|
||||||
|
|
||||||
|
try container.encode(self.parts, forKey: .parts)
|
||||||
|
try container.encode(self.playlist_references, forKey: .playlist_references)
|
||||||
|
try container.encode(self.shuffle, forKey: .shuffle)
|
||||||
|
|
||||||
|
try container.encode(self.sort, forKey: .sort)
|
||||||
|
try container.encode(self.description_overwrite, forKey: .description_overwrite)
|
||||||
|
try container.encode(self.description_suffix, forKey: .description_suffix)
|
||||||
|
|
||||||
|
try container.encode(self.last_updated, forKey: .last_updated)
|
||||||
|
|
||||||
|
try container.encode(self.lastfm_stat_count, forKey: .lastfm_stat_count)
|
||||||
|
try container.encode(self.lastfm_stat_album_count, forKey: .lastfm_stat_album_count)
|
||||||
|
try container.encode(self.lastfm_stat_artist_count, forKey: .lastfm_stat_artist_count)
|
||||||
|
|
||||||
|
try container.encode(self.lastfm_stat_percent, forKey: .lastfm_stat_percent)
|
||||||
|
try container.encode(self.lastfm_stat_album_percent, forKey: .lastfm_stat_album_percent)
|
||||||
|
try container.encode(self.lastfm_stat_artist_percent, forKey: .lastfm_stat_artist_percent)
|
||||||
|
|
||||||
|
try container.encode(self.lastfm_stat_last_refresh, forKey: .lastfm_stat_last_refresh)
|
||||||
|
|
||||||
|
try container.encode(self.add_last_month, forKey: .add_last_month)
|
||||||
|
try container.encode(self.add_this_month, forKey: .add_this_month)
|
||||||
|
try container.encode(self.day_boundary, forKey: .day_boundary)
|
||||||
|
|
||||||
|
try container.encode(self.chart_range, forKey: .chart_range)
|
||||||
|
try container.encode(self.chart_limit, forKey: .chart_limit)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
enum LastFmRange: String, Codable {
|
||||||
case overall = "OVERALL"
|
case overall = "OVERALL"
|
||||||
case week = "WEEK"
|
case week = "WEEK"
|
||||||
case month = "MONTH"
|
case month = "MONTH"
|
||||||
@ -235,36 +397,3 @@ enum LastFmRange: String, Decodable {
|
|||||||
case halfyear = "HALFYEAR"
|
case halfyear = "HALFYEAR"
|
||||||
case year = "YEAR"
|
case year = "YEAR"
|
||||||
}
|
}
|
||||||
|
|
||||||
class LastFMChartPlaylist: Playlist {
|
|
||||||
|
|
||||||
//MARK: Properties
|
|
||||||
|
|
||||||
var chart_range: LastFmRange
|
|
||||||
var chart_limit: Int
|
|
||||||
|
|
||||||
private enum CodingKeys: String, CodingKey { case chart_range; case chart_limit }
|
|
||||||
|
|
||||||
//MARK: Initialization
|
|
||||||
|
|
||||||
init(name: String,
|
|
||||||
username: String = "NO USER",
|
|
||||||
|
|
||||||
chart_range: LastFmRange = .overall,
|
|
||||||
chart_limit: Int = 10){
|
|
||||||
|
|
||||||
self.chart_range = chart_range
|
|
||||||
self.chart_limit = chart_limit
|
|
||||||
|
|
||||||
super.init(name: name, username: username)
|
|
||||||
}
|
|
||||||
|
|
||||||
required init(from decoder: Decoder) throws {
|
|
||||||
let container = try decoder.container(keyedBy: CodingKeys.self)
|
|
||||||
|
|
||||||
chart_range = try LastFmRange(rawValue: container.decode(String.self, forKey: .chart_range))!
|
|
||||||
chart_limit = try container.decode(Int.self, forKey: .chart_limit)
|
|
||||||
|
|
||||||
try super.init(from: decoder)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -15,7 +15,7 @@ class Tag: Identifiable, Equatable, Codable {
|
|||||||
|
|
||||||
var tag_id: String
|
var tag_id: String
|
||||||
var name: String
|
var name: String
|
||||||
var username: String
|
// var username: String
|
||||||
|
|
||||||
var tracks: [JSON]
|
var tracks: [JSON]
|
||||||
var albums: [JSON]
|
var albums: [JSON]
|
||||||
@ -51,7 +51,7 @@ class Tag: Identifiable, Equatable, Codable {
|
|||||||
|
|
||||||
self.tag_id = tag_id
|
self.tag_id = tag_id
|
||||||
self.name = name
|
self.name = name
|
||||||
self.username = username
|
// self.username = username
|
||||||
|
|
||||||
self.tracks = tracks
|
self.tracks = tracks
|
||||||
self.albums = albums
|
self.albums = albums
|
||||||
|
@ -123,20 +123,8 @@ extension PlaylistApi: ApiRequest {
|
|||||||
|
|
||||||
let decoder = JSONDecoder()
|
let decoder = JSONDecoder()
|
||||||
do {
|
do {
|
||||||
let json = try JSON(data: playlist)
|
|
||||||
switch json["type"].string {
|
|
||||||
case "default":
|
|
||||||
let playlist = try decoder.decode(Playlist.self, from: playlist)
|
let playlist = try decoder.decode(Playlist.self, from: playlist)
|
||||||
return playlist
|
return playlist
|
||||||
case "recents":
|
|
||||||
let playlist = try decoder.decode(RecentsPlaylist.self, from: playlist)
|
|
||||||
return playlist
|
|
||||||
case "fmchart":
|
|
||||||
let playlist = try decoder.decode(LastFMChartPlaylist.self, from: playlist)
|
|
||||||
return playlist
|
|
||||||
default:
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
} catch {
|
} catch {
|
||||||
print(error)
|
print(error)
|
||||||
}
|
}
|
||||||
@ -150,19 +138,8 @@ extension PlaylistApi: ApiRequest {
|
|||||||
if let data = _json {
|
if let data = _json {
|
||||||
let decoder = JSONDecoder()
|
let decoder = JSONDecoder()
|
||||||
do {
|
do {
|
||||||
switch playlist["type"].string {
|
|
||||||
case "default":
|
|
||||||
let playlist = try decoder.decode(Playlist.self, from: data)
|
let playlist = try decoder.decode(Playlist.self, from: data)
|
||||||
return playlist
|
return playlist
|
||||||
case "recents":
|
|
||||||
let playlist = try decoder.decode(RecentsPlaylist.self, from: data)
|
|
||||||
return playlist
|
|
||||||
case "fmchart":
|
|
||||||
let playlist = try decoder.decode(LastFMChartPlaylist.self, from: data)
|
|
||||||
return playlist
|
|
||||||
default:
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
} catch {
|
} catch {
|
||||||
print(error)
|
print(error)
|
||||||
}
|
}
|
||||||
|
@ -75,16 +75,16 @@ struct AddPlaylistSheet: View {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
var playlist: Playlist? = nil
|
let playlist: Playlist = Playlist(name: name, username: username)
|
||||||
switch PlaylistType(rawValue: selectedType) ?? .defaultPlaylist {
|
switch PlaylistType(rawValue: selectedType) ?? .defaultPlaylist {
|
||||||
case .defaultPlaylist:
|
case .defaultPlaylist:
|
||||||
playlist = Playlist(name: name, username: username)
|
playlist.type = "default"
|
||||||
break
|
break
|
||||||
case .recents:
|
case .recents:
|
||||||
playlist = RecentsPlaylist(name: name, username: username)
|
playlist.type = "recents"
|
||||||
break
|
break
|
||||||
case .fmchart:
|
case .fmchart:
|
||||||
playlist = LastFMChartPlaylist(name: name, username: username)
|
playlist.type = "fmchart"
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -92,7 +92,7 @@ struct AddPlaylistSheet: View {
|
|||||||
let api = PlaylistApi.newPlaylist(name: self.name,
|
let api = PlaylistApi.newPlaylist(name: self.name,
|
||||||
type: PlaylistType(rawValue: selectedType) ?? .defaultPlaylist)
|
type: PlaylistType(rawValue: selectedType) ?? .defaultPlaylist)
|
||||||
RequestBuilder.buildRequest(apiRequest: api).responseJSON{ response in
|
RequestBuilder.buildRequest(apiRequest: api).responseJSON{ response in
|
||||||
self.playlists.append(playlist!)
|
self.playlists.append(playlist)
|
||||||
self.playlists = self.playlists.sorted(by: { $0.name.lowercased() < $1.name.lowercased() })
|
self.playlists = self.playlists.sorted(by: { $0.name.lowercased() < $1.name.lowercased() })
|
||||||
|
|
||||||
self.isLoading = false
|
self.isLoading = false
|
||||||
|
@ -15,11 +15,6 @@ struct PlaylistView: View {
|
|||||||
|
|
||||||
@Binding var playlist: Playlist
|
@Binding var playlist: Playlist
|
||||||
|
|
||||||
@State private var this_month: Bool = false
|
|
||||||
@State private var last_month: Bool = false
|
|
||||||
@State private var chart_range: LastFmRange = .overall
|
|
||||||
@State private var chart_limit: Int = 0
|
|
||||||
|
|
||||||
@State private var showingSheet = false
|
@State private var showingSheet = false
|
||||||
@State private var isRefreshing = false
|
@State private var isRefreshing = false
|
||||||
@State private var showingNetworkError = false
|
@State private var showingNetworkError = false
|
||||||
@ -32,7 +27,7 @@ struct PlaylistView: View {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
List {
|
Form {
|
||||||
Section(header: Text("Stats")){
|
Section(header: Text("Stats")){
|
||||||
HStack {
|
HStack {
|
||||||
Text("Track Count")
|
Text("Track Count")
|
||||||
@ -104,11 +99,12 @@ struct PlaylistView: View {
|
|||||||
if self.playlist.include_recommendations {
|
if self.playlist.include_recommendations {
|
||||||
Stepper(onIncrement: {
|
Stepper(onIncrement: {
|
||||||
self.$playlist.recommendation_sample.wrappedValue += 1
|
self.$playlist.recommendation_sample.wrappedValue += 1
|
||||||
self.updatePlaylist(updates: JSON(["recommendation_sample": self.playlist.recommendation_sample]))
|
|
||||||
},
|
},
|
||||||
onDecrement: {
|
onDecrement: {
|
||||||
|
if self.playlist.recommendation_sample > 0 {
|
||||||
self.$playlist.recommendation_sample.wrappedValue -= 1
|
self.$playlist.recommendation_sample.wrappedValue -= 1
|
||||||
self.updatePlaylist(updates: JSON(["recommendation_sample": self.playlist.recommendation_sample]))
|
|
||||||
|
}
|
||||||
}){
|
}){
|
||||||
Text("#:")
|
Text("#:")
|
||||||
.foregroundColor(Color.gray)
|
.foregroundColor(Color.gray)
|
||||||
@ -127,24 +123,24 @@ struct PlaylistView: View {
|
|||||||
Text("Shuffle")
|
Text("Shuffle")
|
||||||
}
|
}
|
||||||
|
|
||||||
if playlist is RecentsPlaylist {
|
if playlist.type == "recents" {
|
||||||
Toggle(isOn: $this_month) {
|
Toggle(isOn: self.$playlist.add_this_month) {
|
||||||
Text("This Month")
|
Text("This Month")
|
||||||
}
|
}
|
||||||
|
|
||||||
Toggle(isOn: $last_month) {
|
Toggle(isOn: self.$playlist.add_last_month) {
|
||||||
Text("Last Month")
|
Text("Last Month")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if playlist is LastFMChartPlaylist {
|
if playlist.type == "fmchart" {
|
||||||
HStack {
|
HStack {
|
||||||
Text("Chart Range")
|
Text("Chart Range")
|
||||||
Spacer()
|
Spacer()
|
||||||
Button(action: {
|
Button(action: {
|
||||||
self.showingSheet = true
|
self.showingSheet = true
|
||||||
}) {
|
}) {
|
||||||
Text("\(self.chart_range.rawValue)")
|
Text("\(self.playlist.chart_range.rawValue)")
|
||||||
.foregroundColor(Color.gray)
|
.foregroundColor(Color.gray)
|
||||||
}.actionSheet(isPresented: $showingSheet) {
|
}.actionSheet(isPresented: $showingSheet) {
|
||||||
ActionSheet(title: Text("Chart range"),
|
ActionSheet(title: Text("Chart range"),
|
||||||
@ -193,6 +189,7 @@ struct PlaylistView: View {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// alert seems to need to be within list root element
|
// alert seems to need to be within list root element
|
||||||
// else weird crash on half drag back
|
// else weird crash on half drag back
|
||||||
.alert(isPresented: $showingNetworkError) {
|
.alert(isPresented: $showingNetworkError) {
|
||||||
@ -200,31 +197,11 @@ struct PlaylistView: View {
|
|||||||
message: Text("Could not refresh playlist"))
|
message: Text("Could not refresh playlist"))
|
||||||
}
|
}
|
||||||
|
|
||||||
}.listStyle(GroupedListStyle())
|
}
|
||||||
|
.navigationBarTitle(Text(playlist.name), displayMode: .inline)
|
||||||
.pullToRefresh(isShowing: $isRefreshing) {
|
.pullToRefresh(isShowing: $isRefreshing) {
|
||||||
self.refreshPlaylist()
|
self.refreshPlaylist()
|
||||||
}
|
}
|
||||||
.navigationBarTitle(Text(playlist.name))
|
|
||||||
.onAppear {
|
|
||||||
|
|
||||||
// TODO are these binding properly?
|
|
||||||
if let playlist = self.playlist as? RecentsPlaylist {
|
|
||||||
self.$this_month.wrappedValue = playlist.add_this_month
|
|
||||||
self.$last_month.wrappedValue = playlist.add_last_month
|
|
||||||
}
|
|
||||||
|
|
||||||
if let playlist = self.playlist as? LastFMChartPlaylist {
|
|
||||||
self.$chart_range.wrappedValue = playlist.chart_range
|
|
||||||
self.$chart_limit.wrappedValue = playlist.chart_limit
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func changeChartRange(newRange: LastFmRange) {
|
|
||||||
self.chart_range = newRange
|
|
||||||
// self.updatePlaylist(["chart_range": newRange.rawValue])
|
|
||||||
//TODO: are enums wrong by the time they're here? not sure api will accept it now
|
|
||||||
//TODO: fix downcasting local playlist object to change state
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func runPlaylist() {
|
func runPlaylist() {
|
||||||
|
@ -114,9 +114,6 @@ struct TagView: View {
|
|||||||
self.refreshTag()
|
self.refreshTag()
|
||||||
}
|
}
|
||||||
.navigationBarTitle(Text(tag.name))
|
.navigationBarTitle(Text(tag.name))
|
||||||
.onAppear {
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func runTag() {
|
func runTag() {
|
||||||
|
Loading…
Reference in New Issue
Block a user