mirror of
https://github.com/yattee/yattee.git
synced 2025-01-25 14:17:03 +00:00
143 lines
4.3 KiB
Swift
143 lines
4.3 KiB
Swift
import Defaults
|
|
import Siesta
|
|
import SwiftUI
|
|
import UniformTypeIdentifiers
|
|
|
|
struct FavoriteItemView: View {
|
|
let item: FavoriteItem
|
|
|
|
@StateObject private var store = FavoriteResourceObserver()
|
|
|
|
@Default(.favorites) private var favorites
|
|
@Binding private var dragging: FavoriteItem?
|
|
|
|
@ObservedObject private var accounts = AccountsModel.shared
|
|
private var playlists = PlaylistsModel.shared
|
|
private var favoritesModel = FavoritesModel.shared
|
|
|
|
init(
|
|
item: FavoriteItem,
|
|
dragging: Binding<FavoriteItem?>
|
|
) {
|
|
self.item = item
|
|
_dragging = dragging
|
|
}
|
|
|
|
var body: some View {
|
|
Group {
|
|
if isVisible {
|
|
VStack(alignment: .leading, spacing: 2) {
|
|
Text(label)
|
|
.font(.title3.bold())
|
|
.foregroundColor(.secondary)
|
|
.contextMenu {
|
|
Button {
|
|
favoritesModel.remove(item)
|
|
} label: {
|
|
Label("Remove from Favorites", systemImage: "trash")
|
|
}
|
|
}
|
|
.contentShape(Rectangle())
|
|
#if os(tvOS)
|
|
.padding(.leading, 40)
|
|
#else
|
|
.padding(.leading, 15)
|
|
#endif
|
|
|
|
HorizontalCells(items: store.contentItems)
|
|
}
|
|
|
|
.contentShape(Rectangle())
|
|
#if os(macOS)
|
|
.opacity(dragging?.id == item.id ? 0.5 : 1)
|
|
#endif
|
|
.onAppear {
|
|
resource?.addObserver(store)
|
|
if item.section == .subscriptions {
|
|
cacheFeed(resource?.loadIfNeeded())
|
|
} else {
|
|
resource?.loadIfNeeded()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.onChange(of: accounts.current) { _ in
|
|
resource?.addObserver(store)
|
|
if item.section == .subscriptions {
|
|
cacheFeed(resource?.load())
|
|
} else {
|
|
resource?.load()
|
|
}
|
|
}
|
|
}
|
|
|
|
private func cacheFeed(_ request: Request?) {
|
|
request?.onSuccess { response in
|
|
if let videos: [Video] = response.typedContent() {
|
|
FeedCacheModel.shared.storeFeed(account: accounts.current, videos: videos)
|
|
}
|
|
}
|
|
}
|
|
|
|
private var isVisible: Bool {
|
|
switch item.section {
|
|
case .subscriptions:
|
|
return accounts.app.supportsSubscriptions && accounts.signedIn
|
|
case .popular:
|
|
return accounts.app.supportsPopular
|
|
default:
|
|
return true
|
|
}
|
|
}
|
|
|
|
private var resource: Resource? {
|
|
switch item.section {
|
|
case .subscriptions:
|
|
if accounts.app.supportsSubscriptions {
|
|
return accounts.api.feed(1)
|
|
}
|
|
|
|
case .popular:
|
|
if accounts.app.supportsPopular {
|
|
return accounts.api.popular
|
|
}
|
|
|
|
case let .trending(country, category):
|
|
let trendingCountry = Country(rawValue: country)!
|
|
let trendingCategory = category.isNil ? nil : TrendingCategory(rawValue: category!)
|
|
|
|
return accounts.api.trending(country: trendingCountry, category: trendingCategory)
|
|
|
|
case let .channel(id, _):
|
|
return accounts.api.channelVideos(id)
|
|
|
|
case let .channelPlaylist(id, _):
|
|
return accounts.api.channelPlaylist(id)
|
|
|
|
case let .playlist(id):
|
|
return accounts.api.playlist(id)
|
|
|
|
case let .searchQuery(text, date, duration, order):
|
|
return accounts.api.search(
|
|
.init(
|
|
query: text,
|
|
sortBy: SearchQuery.SortOrder(rawValue: order) ?? .uploadDate,
|
|
date: SearchQuery.Date(rawValue: date),
|
|
duration: SearchQuery.Duration(rawValue: duration)
|
|
),
|
|
page: nil
|
|
)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
private var label: String {
|
|
if case let .playlist(id) = item.section {
|
|
return playlists.find(id: id)?.title ?? "Playlist".localized()
|
|
}
|
|
|
|
return item.section.label.localized()
|
|
}
|
|
}
|