2022-12-13 11:09:20 +00:00
|
|
|
import Defaults
|
2022-12-11 11:38:57 +00:00
|
|
|
import SDWebImageSwiftUI
|
|
|
|
import SwiftUI
|
|
|
|
|
|
|
|
struct ChannelsView: View {
|
2022-12-13 12:14:20 +00:00
|
|
|
@ObservedObject private var feed = FeedModel.shared
|
2022-12-11 15:15:42 +00:00
|
|
|
@ObservedObject private var subscriptions = SubscribedChannelsModel.shared
|
2022-12-11 11:38:57 +00:00
|
|
|
@ObservedObject private var accounts = AccountsModel.shared
|
2022-12-16 21:26:14 +00:00
|
|
|
@ObservedObject private var feedCount = UnwatchedFeedCountModel.shared
|
2023-04-22 19:39:27 +00:00
|
|
|
private var navigation = NavigationModel.shared
|
2022-12-11 11:38:57 +00:00
|
|
|
|
2022-12-13 11:09:20 +00:00
|
|
|
@Default(.showCacheStatus) private var showCacheStatus
|
2023-02-28 20:17:12 +00:00
|
|
|
@Default(.showUnwatchedFeedBadges) private var showUnwatchedFeedBadges
|
2023-05-26 20:38:32 +00:00
|
|
|
@Default(.keepChannelsWithUnwatchedFeedOnTop) private var keepChannelsWithUnwatchedFeedOnTop
|
2023-07-22 17:02:59 +00:00
|
|
|
@Default(.showChannelAvatarInChannelsLists) private var showChannelAvatarInChannelsLists
|
2023-05-26 20:38:32 +00:00
|
|
|
|
|
|
|
@State private var channelLinkActive = false
|
|
|
|
@State private var channelForLink: Channel?
|
2022-12-13 11:09:20 +00:00
|
|
|
|
2022-12-11 11:38:57 +00:00
|
|
|
var body: some View {
|
|
|
|
List {
|
|
|
|
Section(header: header) {
|
2023-05-26 20:38:32 +00:00
|
|
|
ForEach(channels) { channel in
|
2023-04-22 19:39:27 +00:00
|
|
|
let label = HStack {
|
2023-07-22 17:02:59 +00:00
|
|
|
if showChannelAvatarInChannelsLists {
|
|
|
|
ChannelAvatarView(channel: channel, subscribedBadge: false)
|
2023-04-22 19:39:27 +00:00
|
|
|
.frame(width: 35, height: 35)
|
|
|
|
} else {
|
2023-05-26 20:38:32 +00:00
|
|
|
Image(systemName: RecentsModel.symbolSystemImage(channel.name))
|
|
|
|
.imageScale(.large)
|
|
|
|
.foregroundColor(.accentColor)
|
|
|
|
.frame(width: 35, height: 35)
|
2022-12-11 11:38:57 +00:00
|
|
|
}
|
2023-05-26 20:38:32 +00:00
|
|
|
Text(channel.name)
|
|
|
|
.lineLimit(1)
|
2023-04-22 19:39:27 +00:00
|
|
|
}
|
2023-10-15 11:35:23 +00:00
|
|
|
.backport
|
2023-04-22 19:39:27 +00:00
|
|
|
.badge(showUnwatchedFeedBadges ? feedCount.unwatchedByChannelText(channel) : nil)
|
|
|
|
|
|
|
|
Group {
|
|
|
|
#if os(tvOS)
|
|
|
|
Button {
|
|
|
|
navigation.openChannel(channel, navigationStyle: .tab)
|
|
|
|
} label: {
|
|
|
|
label
|
|
|
|
}
|
|
|
|
#else
|
2023-05-26 20:38:32 +00:00
|
|
|
Button {
|
|
|
|
channelForLink = channel
|
|
|
|
channelLinkActive = channelForLink != nil
|
|
|
|
} label: {
|
2023-04-22 19:39:27 +00:00
|
|
|
label
|
2023-05-26 20:38:32 +00:00
|
|
|
.contentShape(Rectangle())
|
|
|
|
.foregroundColor(.primary)
|
2023-04-22 19:39:27 +00:00
|
|
|
}
|
2023-05-26 20:38:32 +00:00
|
|
|
.buttonStyle(.plain)
|
2023-04-22 19:39:27 +00:00
|
|
|
#endif
|
2022-12-11 11:38:57 +00:00
|
|
|
}
|
2022-12-11 16:06:02 +00:00
|
|
|
.contextMenu {
|
2022-12-14 17:10:01 +00:00
|
|
|
if subscriptions.isSubscribing(channel.id) {
|
|
|
|
toggleWatchedButton(channel)
|
|
|
|
}
|
2022-12-11 16:06:02 +00:00
|
|
|
Button {
|
|
|
|
subscriptions.unsubscribe(channel.id)
|
|
|
|
} label: {
|
|
|
|
Label("Unsubscribe", systemImage: "xmark.circle")
|
|
|
|
}
|
|
|
|
}
|
2022-12-11 11:38:57 +00:00
|
|
|
}
|
|
|
|
#if os(tvOS)
|
|
|
|
.padding(.horizontal, 50)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Color.clear.padding(.bottom, 50)
|
|
|
|
.listRowBackground(Color.clear)
|
2023-10-15 11:35:23 +00:00
|
|
|
.backport
|
|
|
|
.listRowSeparator(false)
|
2022-12-11 11:38:57 +00:00
|
|
|
}
|
|
|
|
}
|
2023-06-08 10:17:16 +00:00
|
|
|
#if !os(tvOS)
|
2023-05-26 20:38:32 +00:00
|
|
|
.background(
|
2023-05-29 14:31:01 +00:00
|
|
|
NavigationLink(destination: ChannelVideosView(channel: channelForLink ?? Video.fixture.channel), isActive: $channelLinkActive, label: EmptyView.init)
|
2023-05-26 20:38:32 +00:00
|
|
|
)
|
2023-06-08 10:17:16 +00:00
|
|
|
#endif
|
2022-12-11 11:38:57 +00:00
|
|
|
.onAppear {
|
|
|
|
subscriptions.load()
|
|
|
|
}
|
|
|
|
.onChange(of: accounts.current) { _ in
|
|
|
|
subscriptions.load(force: true)
|
|
|
|
}
|
|
|
|
#if os(iOS)
|
2023-10-15 11:35:23 +00:00
|
|
|
.refreshControl { refreshControl in
|
|
|
|
subscriptions.load(force: true) {
|
|
|
|
refreshControl.endRefreshing()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.backport
|
2022-12-11 11:38:57 +00:00
|
|
|
.refreshable {
|
2023-10-15 11:46:30 +00:00
|
|
|
await subscriptions.load(force: true)
|
2022-12-11 11:38:57 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if !os(tvOS)
|
|
|
|
.background(
|
|
|
|
Button("Refresh") {
|
|
|
|
subscriptions.load(force: true)
|
|
|
|
}
|
|
|
|
.keyboardShortcut("r")
|
|
|
|
.opacity(0)
|
|
|
|
)
|
|
|
|
#endif
|
|
|
|
#if !os(macOS)
|
|
|
|
.onReceive(NotificationCenter.default.publisher(for: UIApplication.willEnterForegroundNotification)) { _ in
|
|
|
|
subscriptions.load()
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if os(tvOS)
|
|
|
|
.padding(.horizontal, 30)
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2023-05-26 20:38:32 +00:00
|
|
|
var channels: [Channel] {
|
|
|
|
keepChannelsWithUnwatchedFeedOnTop ? subscriptions.allByUnwatchedCount : subscriptions.all
|
|
|
|
}
|
|
|
|
|
2022-12-11 11:38:57 +00:00
|
|
|
var header: some View {
|
|
|
|
HStack {
|
|
|
|
#if os(tvOS)
|
|
|
|
SubscriptionsPageButton()
|
|
|
|
#endif
|
|
|
|
|
2022-12-13 11:09:20 +00:00
|
|
|
if showCacheStatus {
|
|
|
|
Spacer()
|
2022-12-11 11:38:57 +00:00
|
|
|
|
2022-12-13 11:09:20 +00:00
|
|
|
CacheStatusHeader(
|
|
|
|
refreshTime: subscriptions.formattedCacheTime,
|
|
|
|
isLoading: subscriptions.isLoading
|
|
|
|
)
|
|
|
|
}
|
2022-12-11 11:38:57 +00:00
|
|
|
|
|
|
|
#if os(tvOS)
|
2022-12-13 11:09:20 +00:00
|
|
|
if !showCacheStatus {
|
|
|
|
Spacer()
|
|
|
|
}
|
2022-12-11 11:38:57 +00:00
|
|
|
Button {
|
|
|
|
subscriptions.load(force: true)
|
|
|
|
} label: {
|
|
|
|
Label("Refresh", systemImage: "arrow.clockwise")
|
|
|
|
.labelStyle(.iconOnly)
|
|
|
|
.imageScale(.small)
|
2023-04-22 19:07:30 +00:00
|
|
|
.font(.caption)
|
2022-12-11 11:38:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#if os(tvOS)
|
|
|
|
.padding(.bottom, 15)
|
|
|
|
.padding(.top, 15)
|
|
|
|
#endif
|
|
|
|
}
|
2022-12-14 17:10:01 +00:00
|
|
|
|
|
|
|
@ViewBuilder func toggleWatchedButton(_ channel: Channel) -> some View {
|
|
|
|
if feed.canMarkChannelAsWatched(channel.id) {
|
|
|
|
markChannelAsWatchedButton(channel)
|
|
|
|
} else {
|
|
|
|
markChannelAsUnwatchedButton(channel)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func markChannelAsWatchedButton(_ channel: Channel) -> some View {
|
|
|
|
Button {
|
|
|
|
feed.markChannelAsWatched(channel.id)
|
|
|
|
} label: {
|
|
|
|
Label("Mark channel feed as watched", systemImage: "checkmark.circle.fill")
|
|
|
|
}
|
|
|
|
.disabled(!feed.canMarkAllFeedAsWatched)
|
|
|
|
}
|
|
|
|
|
|
|
|
func markChannelAsUnwatchedButton(_ channel: Channel) -> some View {
|
|
|
|
Button {
|
|
|
|
feed.markChannelAsUnwatched(channel.id)
|
|
|
|
} label: {
|
|
|
|
Label("Mark channel feed as unwatched", systemImage: "checkmark.circle")
|
|
|
|
}
|
|
|
|
}
|
2022-12-11 11:38:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct ChannelsView_Previews: PreviewProvider {
|
|
|
|
static var previews: some View {
|
|
|
|
NavigationView {
|
|
|
|
ChannelsView()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|