2022-04-17 09:34:26 +00:00
|
|
|
import CoreMedia
|
2021-07-27 22:40:04 +00:00
|
|
|
import Defaults
|
2021-10-21 23:29:10 +00:00
|
|
|
import SDWebImageSwiftUI
|
2021-07-27 21:26:52 +00:00
|
|
|
import SwiftUI
|
|
|
|
|
2021-10-21 23:29:10 +00:00
|
|
|
struct VideoCell: View {
|
2021-12-26 21:14:46 +00:00
|
|
|
private var video: Video
|
2021-10-05 20:20:09 +00:00
|
|
|
|
2022-08-25 23:38:04 +00:00
|
|
|
@Environment(\.horizontalCells) private var horizontalCells
|
2022-05-29 18:26:56 +00:00
|
|
|
@Environment(\.inChannelView) private var inChannelView
|
2022-08-25 23:38:04 +00:00
|
|
|
@Environment(\.navigationStyle) private var navigationStyle
|
2021-07-27 22:40:04 +00:00
|
|
|
|
|
|
|
#if os(iOS)
|
|
|
|
@Environment(\.verticalSizeClass) private var verticalSizeClass
|
|
|
|
#endif
|
|
|
|
|
2022-11-24 20:36:05 +00:00
|
|
|
@ObservedObject var thumbnails = ThumbnailsModel.shared
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2021-11-04 23:25:51 +00:00
|
|
|
@Default(.channelOnThumbnail) private var channelOnThumbnail
|
|
|
|
@Default(.timeOnThumbnail) private var timeOnThumbnail
|
2022-01-06 14:56:59 +00:00
|
|
|
@Default(.roundedThumbnails) private var roundedThumbnails
|
2021-12-26 21:14:46 +00:00
|
|
|
@Default(.saveHistory) private var saveHistory
|
|
|
|
@Default(.showWatchingProgress) private var showWatchingProgress
|
|
|
|
@Default(.watchedVideoStyle) private var watchedVideoStyle
|
2022-01-02 19:39:19 +00:00
|
|
|
@Default(.watchedVideoBadgeColor) private var watchedVideoBadgeColor
|
2021-12-26 21:14:46 +00:00
|
|
|
@Default(.watchedVideoPlayNowBehavior) private var watchedVideoPlayNowBehavior
|
2021-11-04 23:25:51 +00:00
|
|
|
|
2022-08-31 22:00:28 +00:00
|
|
|
private var navigation: NavigationModel { .shared }
|
|
|
|
private var player: PlayerModel { .shared }
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
@FetchRequest private var watchRequest: FetchedResults<Watch>
|
2021-10-05 20:20:09 +00:00
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
init(video: Video) {
|
|
|
|
self.video = video
|
|
|
|
_watchRequest = video.watchFetchRequest
|
|
|
|
}
|
2021-10-28 17:14:55 +00:00
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
var body: some View {
|
2022-08-13 14:46:45 +00:00
|
|
|
Button(action: playAction) {
|
|
|
|
content
|
2022-08-25 23:38:04 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.frame(width: 580, height: 470)
|
|
|
|
#endif
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
2021-12-26 21:14:46 +00:00
|
|
|
.opacity(contentOpacity)
|
2022-08-25 23:38:04 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.buttonStyle(.card)
|
|
|
|
#else
|
|
|
|
.buttonStyle(.plain)
|
|
|
|
#endif
|
|
|
|
.contentShape(RoundedRectangle(cornerRadius: thumbnailRoundingCornerRadius))
|
|
|
|
.contextMenu {
|
|
|
|
VideoContextMenuView(video: video)
|
|
|
|
}
|
2021-08-01 23:01:24 +00:00
|
|
|
}
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2022-01-06 14:56:59 +00:00
|
|
|
private var thumbnailRoundingCornerRadius: Double {
|
|
|
|
#if os(tvOS)
|
|
|
|
return Double(12)
|
|
|
|
#else
|
|
|
|
return Double(roundedThumbnails ? 12 : 0)
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
private func playAction() {
|
2022-04-17 09:34:26 +00:00
|
|
|
DispatchQueue.main.async {
|
|
|
|
guard video.videoID != Video.fixtureID else {
|
|
|
|
return
|
2021-12-26 21:14:46 +00:00
|
|
|
}
|
|
|
|
|
2022-06-07 21:27:48 +00:00
|
|
|
if player.musicMode {
|
|
|
|
player.toggleMusicMode()
|
|
|
|
}
|
|
|
|
|
2022-04-17 09:34:26 +00:00
|
|
|
if watchingNow {
|
|
|
|
if !player.playingInPictureInPicture {
|
|
|
|
player.show()
|
|
|
|
}
|
2021-12-26 21:14:46 +00:00
|
|
|
|
2022-04-17 09:34:26 +00:00
|
|
|
if !playNowContinues {
|
2022-08-28 17:18:49 +00:00
|
|
|
player.backend.seek(to: .zero, seekType: .userInteracted)
|
2022-04-17 09:34:26 +00:00
|
|
|
}
|
2021-12-26 21:14:46 +00:00
|
|
|
|
2022-04-17 09:34:26 +00:00
|
|
|
player.play()
|
2021-12-26 21:14:46 +00:00
|
|
|
|
2022-04-17 09:34:26 +00:00
|
|
|
return
|
|
|
|
}
|
2021-12-26 21:14:46 +00:00
|
|
|
|
2022-04-17 09:34:26 +00:00
|
|
|
var playAt: CMTime?
|
2021-12-26 21:14:46 +00:00
|
|
|
|
2022-05-29 13:38:48 +00:00
|
|
|
if saveHistory,
|
|
|
|
playNowContinues,
|
2022-04-17 09:34:26 +00:00
|
|
|
!watch.isNil,
|
|
|
|
!watch!.finished
|
|
|
|
{
|
|
|
|
playAt = .secondsInDefaultTimescale(watch!.stoppedAt)
|
|
|
|
}
|
|
|
|
|
2022-05-21 20:58:11 +00:00
|
|
|
player.avPlayerBackend.startPictureInPictureOnPlay = player.playingInPictureInPicture
|
|
|
|
|
2022-05-28 21:41:23 +00:00
|
|
|
player.play(video, at: playAt)
|
2022-04-17 09:34:26 +00:00
|
|
|
}
|
2021-12-26 21:14:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private var playNowContinues: Bool {
|
|
|
|
watchedVideoPlayNowBehavior == .continue
|
|
|
|
}
|
|
|
|
|
|
|
|
private var watch: Watch? {
|
|
|
|
watchRequest.first
|
|
|
|
}
|
|
|
|
|
|
|
|
private var finished: Bool {
|
|
|
|
watch?.finished ?? false
|
|
|
|
}
|
|
|
|
|
|
|
|
private var watchingNow: Bool {
|
|
|
|
player.currentVideo == video
|
|
|
|
}
|
|
|
|
|
|
|
|
private var content: some View {
|
2021-08-31 21:17:50 +00:00
|
|
|
VStack {
|
2021-09-26 22:28:42 +00:00
|
|
|
#if os(iOS)
|
|
|
|
if verticalSizeClass == .compact, !horizontalCells {
|
|
|
|
horizontalRow
|
|
|
|
.padding(.vertical, 4)
|
|
|
|
} else {
|
2021-08-31 21:17:50 +00:00
|
|
|
verticalRow
|
2021-09-26 22:28:42 +00:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
verticalRow
|
|
|
|
#endif
|
2021-08-31 21:17:50 +00:00
|
|
|
}
|
|
|
|
#if os(macOS)
|
2021-12-19 23:36:12 +00:00
|
|
|
.background(Color.secondaryBackground)
|
2021-08-31 21:17:50 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
private var contentOpacity: Double {
|
|
|
|
guard saveHistory,
|
|
|
|
!watch.isNil,
|
2022-01-02 19:39:19 +00:00
|
|
|
watchedVideoStyle == .decreasedOpacity || watchedVideoStyle == .both
|
2021-12-26 21:14:46 +00:00
|
|
|
else {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
return watch!.finished ? 0.5 : 1
|
|
|
|
}
|
|
|
|
|
2021-09-29 14:29:17 +00:00
|
|
|
#if os(iOS)
|
2021-12-26 21:14:46 +00:00
|
|
|
private var horizontalRow: some View {
|
2021-09-29 14:29:17 +00:00
|
|
|
HStack(alignment: .top, spacing: 2) {
|
|
|
|
Section {
|
|
|
|
#if os(tvOS)
|
2021-11-05 20:53:43 +00:00
|
|
|
thumbnailImage
|
2021-09-29 14:29:17 +00:00
|
|
|
#else
|
|
|
|
thumbnail
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
.frame(maxWidth: 320)
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2021-09-29 14:29:17 +00:00
|
|
|
VStack(alignment: .leading, spacing: 0) {
|
2022-11-11 18:19:48 +00:00
|
|
|
videoDetail(video.displayTitle, lineLimit: 5)
|
2021-09-29 14:29:17 +00:00
|
|
|
.frame(minWidth: 0, maxWidth: .infinity, alignment: .leading)
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2022-08-08 17:26:51 +00:00
|
|
|
if !channelOnThumbnail, !inChannelView {
|
2022-03-27 10:50:36 +00:00
|
|
|
channelButton(badge: false)
|
2021-11-04 23:25:51 +00:00
|
|
|
}
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2021-09-29 14:29:17 +00:00
|
|
|
if additionalDetailsAvailable {
|
|
|
|
Spacer()
|
2021-09-18 20:36:42 +00:00
|
|
|
|
2021-11-04 23:25:51 +00:00
|
|
|
HStack(spacing: 15) {
|
2021-09-29 14:29:17 +00:00
|
|
|
if let date = video.publishedDate {
|
|
|
|
VStack {
|
|
|
|
Image(systemName: "calendar")
|
2021-11-04 23:25:51 +00:00
|
|
|
.frame(height: 15)
|
2021-09-29 14:29:17 +00:00
|
|
|
Text(date)
|
|
|
|
}
|
2021-09-18 20:36:42 +00:00
|
|
|
}
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2021-10-22 23:04:03 +00:00
|
|
|
if video.views > 0 {
|
2021-09-29 14:29:17 +00:00
|
|
|
VStack {
|
|
|
|
Image(systemName: "eye")
|
2021-11-04 23:25:51 +00:00
|
|
|
.frame(height: 15)
|
2021-09-29 14:29:17 +00:00
|
|
|
Text(video.viewsCount!)
|
|
|
|
}
|
2021-09-18 20:36:42 +00:00
|
|
|
}
|
2021-11-04 23:25:51 +00:00
|
|
|
|
|
|
|
if !timeOnThumbnail, let time = video.length.formattedAsPlaybackTime() {
|
|
|
|
VStack {
|
|
|
|
Image(systemName: "clock")
|
|
|
|
.frame(height: 15)
|
|
|
|
Text(time)
|
|
|
|
}
|
|
|
|
}
|
2021-09-18 20:36:42 +00:00
|
|
|
}
|
2021-09-29 14:29:17 +00:00
|
|
|
.foregroundColor(.secondary)
|
2021-09-18 20:36:42 +00:00
|
|
|
}
|
|
|
|
}
|
2021-09-29 14:29:17 +00:00
|
|
|
.padding()
|
|
|
|
.frame(minHeight: 180)
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2021-09-29 14:29:17 +00:00
|
|
|
#if os(tvOS)
|
2021-10-22 20:49:31 +00:00
|
|
|
if let time = video.length.formattedAsPlaybackTime() || video.live || video.upcoming {
|
2021-08-02 21:10:22 +00:00
|
|
|
Spacer()
|
|
|
|
|
2022-11-25 18:31:48 +00:00
|
|
|
VStack {
|
2021-09-29 14:29:17 +00:00
|
|
|
Spacer()
|
|
|
|
|
2021-10-22 20:49:31 +00:00
|
|
|
if let time = video.length.formattedAsPlaybackTime() {
|
2021-09-29 14:29:17 +00:00
|
|
|
HStack(spacing: 4) {
|
|
|
|
Image(systemName: "clock")
|
|
|
|
Text(time)
|
|
|
|
.fontWeight(.bold)
|
|
|
|
}
|
|
|
|
.foregroundColor(.secondary)
|
|
|
|
} else if video.live {
|
|
|
|
DetailBadge(text: "Live", style: .outstanding)
|
|
|
|
} else if video.upcoming {
|
|
|
|
DetailBadge(text: "Upcoming", style: .informational)
|
2021-08-02 21:10:22 +00:00
|
|
|
}
|
2021-08-01 23:01:24 +00:00
|
|
|
|
2021-09-29 14:29:17 +00:00
|
|
|
Spacer()
|
|
|
|
}
|
2021-10-22 23:04:03 +00:00
|
|
|
.lineLimit(1)
|
2021-08-02 21:10:22 +00:00
|
|
|
}
|
2021-09-29 14:29:17 +00:00
|
|
|
#endif
|
|
|
|
}
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
2021-09-29 14:29:17 +00:00
|
|
|
#endif
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
private var verticalRow: some View {
|
2021-09-18 20:36:42 +00:00
|
|
|
VStack(alignment: .leading, spacing: 0) {
|
2021-08-01 23:01:24 +00:00
|
|
|
thumbnail
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2021-09-25 08:18:22 +00:00
|
|
|
VStack(alignment: .leading, spacing: 0) {
|
2021-11-04 23:25:51 +00:00
|
|
|
Group {
|
|
|
|
VStack(alignment: .leading, spacing: 0) {
|
2022-11-11 18:19:48 +00:00
|
|
|
videoDetail(video.displayTitle, lineLimit: 2)
|
2021-11-04 23:25:51 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.frame(minHeight: 60, alignment: .top)
|
|
|
|
#elseif os(macOS)
|
|
|
|
.frame(minHeight: 32, alignment: .top)
|
|
|
|
#else
|
|
|
|
.frame(minHeight: 40, alignment: .top)
|
|
|
|
#endif
|
2022-08-08 17:26:51 +00:00
|
|
|
if !channelOnThumbnail, !inChannelView {
|
2022-03-27 10:50:36 +00:00
|
|
|
channelButton(badge: false)
|
2021-11-04 23:25:51 +00:00
|
|
|
.padding(.top, 4)
|
|
|
|
.padding(.bottom, 6)
|
|
|
|
}
|
|
|
|
}
|
2021-11-07 22:27:09 +00:00
|
|
|
.frame(minWidth: 0, maxWidth: .infinity, alignment: .leading)
|
2021-11-04 23:25:51 +00:00
|
|
|
}
|
2021-08-01 23:01:24 +00:00
|
|
|
#if os(tvOS)
|
2021-11-08 16:29:35 +00:00
|
|
|
.frame(minHeight: channelOnThumbnail ? 80 : 120, alignment: .top)
|
2021-08-02 21:10:22 +00:00
|
|
|
#elseif os(macOS)
|
2021-11-08 16:29:35 +00:00
|
|
|
.frame(minHeight: 35, alignment: .top)
|
2021-08-02 21:10:22 +00:00
|
|
|
#else
|
2021-11-08 16:29:35 +00:00
|
|
|
.frame(minHeight: 50, alignment: .top)
|
2021-08-01 23:01:24 +00:00
|
|
|
#endif
|
2021-10-05 20:20:09 +00:00
|
|
|
.padding(.bottom, 4)
|
2021-07-27 22:40:04 +00:00
|
|
|
|
2021-11-01 21:56:18 +00:00
|
|
|
HStack(spacing: 8) {
|
|
|
|
if let date = video.publishedDate {
|
2021-11-04 23:25:51 +00:00
|
|
|
HStack(spacing: 2) {
|
|
|
|
Text(date)
|
|
|
|
.allowsTightening(true)
|
|
|
|
}
|
2021-11-01 21:56:18 +00:00
|
|
|
}
|
2021-09-18 20:36:42 +00:00
|
|
|
|
2021-11-01 21:56:18 +00:00
|
|
|
if video.views > 0 {
|
2021-11-04 23:25:51 +00:00
|
|
|
HStack(spacing: 2) {
|
|
|
|
Image(systemName: "eye")
|
|
|
|
Text(video.viewsCount!)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-28 14:27:01 +00:00
|
|
|
if let time, !timeOnThumbnail {
|
2021-11-04 23:25:51 +00:00
|
|
|
Spacer()
|
|
|
|
|
|
|
|
HStack(spacing: 2) {
|
|
|
|
Text(time)
|
|
|
|
}
|
2021-09-13 20:41:16 +00:00
|
|
|
}
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
2021-11-04 23:25:51 +00:00
|
|
|
.lineLimit(1)
|
2021-11-01 21:56:18 +00:00
|
|
|
.foregroundColor(.secondary)
|
2022-06-15 21:48:14 +00:00
|
|
|
.frame(maxWidth: .infinity, minHeight: 30, alignment: .topLeading)
|
2021-09-25 08:18:22 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.padding(.bottom, 10)
|
|
|
|
#endif
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
2021-09-18 20:36:42 +00:00
|
|
|
.padding(.top, 4)
|
|
|
|
.frame(minWidth: 0, maxWidth: .infinity, alignment: .topLeading)
|
2021-08-01 23:01:24 +00:00
|
|
|
#if os(tvOS)
|
2022-08-25 23:38:04 +00:00
|
|
|
.padding(.horizontal, horizontalCells ? 10 : 20)
|
2021-08-01 23:01:24 +00:00
|
|
|
#endif
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-18 21:46:00 +00:00
|
|
|
@ViewBuilder private func channelButton(badge: Bool = true) -> some View {
|
|
|
|
if !video.channel.name.isEmpty {
|
|
|
|
Button {
|
|
|
|
guard !inChannelView else {
|
|
|
|
return
|
|
|
|
}
|
2022-05-29 18:26:56 +00:00
|
|
|
|
2022-11-24 20:36:05 +00:00
|
|
|
NavigationModel.shared.openChannel(
|
2022-11-18 21:46:00 +00:00
|
|
|
video.channel,
|
|
|
|
navigationStyle: navigationStyle
|
|
|
|
)
|
|
|
|
} label: {
|
|
|
|
if badge {
|
|
|
|
DetailBadge(text: video.author, style: .prominent)
|
|
|
|
.foregroundColor(.primary)
|
|
|
|
} else {
|
|
|
|
Text(video.channel.name)
|
|
|
|
.fontWeight(.semibold)
|
|
|
|
.foregroundColor(.secondary)
|
|
|
|
}
|
2022-03-27 10:50:36 +00:00
|
|
|
}
|
2022-11-18 21:46:00 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.buttonStyle(.card)
|
|
|
|
#else
|
|
|
|
.buttonStyle(.plain)
|
|
|
|
#endif
|
|
|
|
.help("\(video.channel.name) Channel")
|
2022-03-27 10:50:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
private var additionalDetailsAvailable: Bool {
|
|
|
|
video.publishedDate != nil || video.views != 0 ||
|
|
|
|
(!timeOnThumbnail && !video.length.formattedAsPlaybackTime().isNil)
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
private var thumbnail: some View {
|
2021-07-31 22:10:56 +00:00
|
|
|
ZStack(alignment: .leading) {
|
2021-12-26 21:14:46 +00:00
|
|
|
ZStack(alignment: .bottomLeading) {
|
|
|
|
thumbnailImage
|
|
|
|
if saveHistory, showWatchingProgress, watch?.progress ?? 0 > 0 {
|
|
|
|
ProgressView(value: watch!.progress, total: 100)
|
2022-01-02 19:39:19 +00:00
|
|
|
.progressViewStyle(LinearProgressViewStyle(tint: Color("AppRedColor")))
|
2021-12-26 21:14:46 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.padding(.horizontal, 16)
|
|
|
|
#else
|
|
|
|
.padding(.horizontal, 10)
|
|
|
|
#endif
|
|
|
|
#if os(macOS)
|
|
|
|
.offset(x: 0, y: 4)
|
|
|
|
#else
|
|
|
|
.offset(x: 0, y: -3)
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2021-07-27 22:40:04 +00:00
|
|
|
|
|
|
|
VStack {
|
|
|
|
HStack(alignment: .top) {
|
|
|
|
if video.live {
|
|
|
|
DetailBadge(text: "Live", style: .outstanding)
|
|
|
|
} else if video.upcoming {
|
|
|
|
DetailBadge(text: "Upcoming", style: .informational)
|
|
|
|
}
|
|
|
|
|
|
|
|
Spacer()
|
|
|
|
|
2022-08-08 17:26:51 +00:00
|
|
|
if channelOnThumbnail, !inChannelView {
|
2022-03-27 10:50:36 +00:00
|
|
|
channelButton()
|
2021-11-04 23:25:51 +00:00
|
|
|
}
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
2021-12-26 21:14:46 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.padding(16)
|
|
|
|
#else
|
2021-07-27 22:40:04 +00:00
|
|
|
.padding(10)
|
2021-12-26 21:14:46 +00:00
|
|
|
#endif
|
2021-07-27 22:40:04 +00:00
|
|
|
|
|
|
|
Spacer()
|
|
|
|
|
2022-11-25 18:31:48 +00:00
|
|
|
HStack {
|
2021-12-26 21:14:46 +00:00
|
|
|
if saveHistory,
|
2022-08-31 19:24:46 +00:00
|
|
|
watchedVideoStyle.isShowingBadge,
|
2021-12-26 21:14:46 +00:00
|
|
|
watch?.finished ?? false
|
|
|
|
{
|
|
|
|
Image(systemName: "checkmark.circle.fill")
|
2022-01-02 19:39:19 +00:00
|
|
|
.foregroundColor(Color(
|
|
|
|
watchedVideoBadgeColor == .colorSchemeBased ? "WatchProgressBarColor" :
|
|
|
|
watchedVideoBadgeColor == .red ? "AppRedColor" : "AppBlueColor"
|
|
|
|
))
|
2021-12-26 21:14:46 +00:00
|
|
|
.background(Color.white)
|
|
|
|
.clipShape(Circle())
|
|
|
|
#if os(tvOS)
|
|
|
|
.font(.system(size: 40))
|
|
|
|
#else
|
|
|
|
.font(.system(size: 30))
|
|
|
|
#endif
|
|
|
|
}
|
2021-07-27 22:40:04 +00:00
|
|
|
Spacer()
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
if timeOnThumbnail,
|
|
|
|
!video.live,
|
2022-09-28 14:27:01 +00:00
|
|
|
let time
|
2021-12-26 21:14:46 +00:00
|
|
|
{
|
2021-07-27 22:40:04 +00:00
|
|
|
DetailBadge(text: time, style: .prominent)
|
|
|
|
}
|
|
|
|
}
|
2021-12-26 21:14:46 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.padding(16)
|
|
|
|
#else
|
2021-07-27 22:40:04 +00:00
|
|
|
.padding(10)
|
2021-12-26 21:14:46 +00:00
|
|
|
#endif
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
2021-10-22 23:04:03 +00:00
|
|
|
.lineLimit(1)
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
private var thumbnailImage: some View {
|
2021-10-24 22:26:25 +00:00
|
|
|
Group {
|
2022-08-31 19:24:46 +00:00
|
|
|
let url = thumbnails.best(video)
|
2022-09-11 19:33:08 +00:00
|
|
|
|
2022-11-10 18:11:19 +00:00
|
|
|
WebImage(url: url, options: [.lowPriority])
|
2022-09-11 19:33:08 +00:00
|
|
|
.resizable()
|
|
|
|
.placeholder {
|
2022-08-31 19:24:46 +00:00
|
|
|
Rectangle().foregroundColor(Color("PlaceholderColor"))
|
|
|
|
}
|
2022-09-11 19:33:08 +00:00
|
|
|
.retryOnAppear(true)
|
|
|
|
.onFailure { _ in
|
2022-09-28 14:27:01 +00:00
|
|
|
guard let url else { return }
|
2022-09-11 19:33:08 +00:00
|
|
|
thumbnails.insertUnloadable(url)
|
|
|
|
}
|
2022-09-11 19:21:50 +00:00
|
|
|
|
2022-09-11 19:33:08 +00:00
|
|
|
#if os(tvOS)
|
|
|
|
.frame(minHeight: 320)
|
|
|
|
#endif
|
2021-10-24 22:26:25 +00:00
|
|
|
}
|
2022-01-06 14:56:59 +00:00
|
|
|
.mask(RoundedRectangle(cornerRadius: thumbnailRoundingCornerRadius))
|
2021-10-24 22:26:25 +00:00
|
|
|
.modifier(AspectRatioModifier())
|
2021-07-27 22:40:04 +00:00
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
private var time: String? {
|
|
|
|
guard var videoTime = video.length.formattedAsPlaybackTime() else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if !saveHistory || !showWatchingProgress || watch?.finished ?? false {
|
|
|
|
return videoTime
|
|
|
|
}
|
|
|
|
|
|
|
|
if let stoppedAt = watch?.stoppedAt,
|
|
|
|
stoppedAt.isFinite,
|
|
|
|
let stoppedAtFormatted = stoppedAt.formattedAsPlaybackTime()
|
|
|
|
{
|
2022-03-27 10:50:36 +00:00
|
|
|
if (watch?.videoDuration ?? 0) > 0 {
|
2021-12-26 21:14:46 +00:00
|
|
|
videoTime = watch!.videoDuration.formattedAsPlaybackTime() ?? "?"
|
|
|
|
}
|
|
|
|
return "\(stoppedAtFormatted) / \(videoTime)"
|
|
|
|
}
|
|
|
|
|
|
|
|
return videoTime
|
|
|
|
}
|
|
|
|
|
|
|
|
private func videoDetail(_ text: String, lineLimit: Int = 1) -> some View {
|
2021-07-27 22:40:04 +00:00
|
|
|
Text(text)
|
2021-08-01 23:01:24 +00:00
|
|
|
.fontWeight(.bold)
|
2021-07-27 22:40:04 +00:00
|
|
|
.lineLimit(lineLimit)
|
|
|
|
.truncationMode(.middle)
|
2021-07-27 21:26:52 +00:00
|
|
|
}
|
2021-08-02 21:10:22 +00:00
|
|
|
|
2021-09-18 20:36:42 +00:00
|
|
|
struct AspectRatioModifier: ViewModifier {
|
|
|
|
@Environment(\.horizontalCells) private var horizontalCells
|
|
|
|
|
|
|
|
func body(content: Content) -> some View {
|
|
|
|
Group {
|
|
|
|
if horizontalCells {
|
|
|
|
content
|
|
|
|
} else {
|
|
|
|
content
|
2021-12-26 21:14:46 +00:00
|
|
|
.aspectRatio(
|
|
|
|
VideoPlayerView.defaultAspectRatio,
|
|
|
|
contentMode: .fill
|
|
|
|
)
|
2021-09-18 20:36:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-07-27 21:26:52 +00:00
|
|
|
}
|
2021-10-21 23:29:10 +00:00
|
|
|
|
2021-10-22 23:04:03 +00:00
|
|
|
struct VideoCell_Preview: PreviewProvider {
|
2021-10-21 23:29:10 +00:00
|
|
|
static var previews: some View {
|
|
|
|
Group {
|
|
|
|
VideoCell(video: Video.fixture)
|
|
|
|
}
|
2021-11-04 23:25:51 +00:00
|
|
|
#if os(macOS)
|
2021-11-08 16:29:35 +00:00
|
|
|
.frame(maxWidth: 300, maxHeight: 250)
|
2021-11-04 23:25:51 +00:00
|
|
|
#elseif os(iOS)
|
2021-11-08 16:29:35 +00:00
|
|
|
.frame(maxWidth: 300, maxHeight: 200)
|
2021-11-04 23:25:51 +00:00
|
|
|
#endif
|
2021-10-21 23:29:10 +00:00
|
|
|
.injectFixtureEnvironmentObjects()
|
|
|
|
}
|
|
|
|
}
|