get thumb image and fallback imageview

This commit is contained in:
Ethan Pippin 2022-01-17 15:51:22 -07:00
parent 9e9baed0ff
commit 70506a276e
5 changed files with 115 additions and 26 deletions

View File

@ -92,6 +92,19 @@ public extension BaseItemDto {
return URL(string: urlString)! return URL(string: urlString)!
} }
func getThumbImage(maxWidth: Int) -> URL {
let imageType = ImageType.thumb
let imageItemId = id ?? ""
let x = UIScreen.main.nativeScale * CGFloat(maxWidth)
let urlString = ImageAPI.getItemImageWithRequestBuilder(itemId: imageItemId,
imageType: imageType,
maxWidth: Int(x),
quality: 96).URLString
return URL(string: urlString)!
}
func getEpisodeLocator() -> String? { func getEpisodeLocator() -> String? {
if let seasonNo = parentIndexNumber, let episodeNo = indexNumber { if let seasonNo = parentIndexNumber, let episodeNo = indexNumber {
return L10n.seasonAndEpisode(String(seasonNo), String(episodeNo)) return L10n.seasonAndEpisode(String(seasonNo), String(episodeNo))

View File

@ -10,26 +10,49 @@ import NukeUI
import SwiftUI import SwiftUI
struct ImageView: View { struct ImageView: View {
private let source: URL
private let sources: [URL]
private let blurhash: String private let blurhash: String
private let failureInitials: String private let failureInitials: String
init(src: URL, bh: String = "001fC^", failureInitials: String = "") { init(src: URL, bh: String = "001fC^", failureInitials: String = "") {
self.source = src self.sources = [src]
self.blurhash = bh self.blurhash = bh
self.failureInitials = failureInitials self.failureInitials = failureInitials
} }
// TODO: fix placeholder hash image init(sources: [URL], bh: String = "001fC^", failureInitials: String = "") {
assert(!sources.isEmpty, "Must supply at least one source")
self.sources = sources
self.blurhash = bh
self.failureInitials = failureInitials
}
// TODO: fix placeholder hash view
@ViewBuilder @ViewBuilder
private var placeholderImage: some View { private func placeholderView() -> some View {
Image(uiImage: UIImage(blurHash: blurhash, size: CGSize(width: 8, height: 8)) ?? // Image(uiImage: UIImage(blurHash: blurhash, size: CGSize(width: 8, height: 8)) ??
UIImage(blurHash: "001fC^", size: CGSize(width: 8, height: 8))!) // UIImage(blurHash: "001fC^", size: CGSize(width: 8, height: 8))!)
.resizable() // .resizable()
#if os(tvOS)
ZStack {
Color.black.ignoresSafeArea()
ProgressView()
}
#else
ZStack {
Color.gray.ignoresSafeArea()
ProgressView()
}
#endif
} }
@ViewBuilder @ViewBuilder
private var failureImage: some View { private func failureImage() -> some View {
ZStack { ZStack {
Rectangle() Rectangle()
.foregroundColor(Color(UIColor.darkGray)) .foregroundColor(Color(UIColor.darkGray))
@ -42,25 +65,66 @@ struct ImageView: View {
} }
var body: some View { var body: some View {
LazyImage(source: source) { state in ImageViewBackgroundA(index: 0,
sources: sources,
placeholderView: placeholderView,
failureView: failureImage)
}
}
// Two image view are necessary to switch between one another to appease the type system
// as a recursive view with itself isn't valid
fileprivate struct ImageViewBackgroundA<PlaceholderView: View, FailureView: View>: View {
let index: Int
let sources: [URL]
let placeholderView: () -> PlaceholderView
let failureView: () -> FailureView
var body: some View {
LazyImage(source: sources[index]) { state in
if let image = state.image { if let image = state.image {
image image
} else if state.error != nil { } else if state.error != nil {
failureImage if index + 1 == sources.count {
failureView()
} else {
ImageViewBackgroundB(index: index + 1,
sources: sources,
placeholderView: placeholderView,
failureView: failureView)
}
} else { } else {
#if os(tvOS) placeholderView()
ZStack { }
Color.black.ignoresSafeArea() }
.pipeline(ImagePipeline(configuration: .withDataCache))
ProgressView() }
} }
#else
ZStack { fileprivate struct ImageViewBackgroundB<PlaceholderView: View, FailureView: View>: View {
Color.gray.ignoresSafeArea()
let index: Int
ProgressView() let sources: [URL]
} let placeholderView: () -> PlaceholderView
#endif let failureView: () -> FailureView
var body: some View {
LazyImage(source: sources[index]) { state in
if let image = state.image {
image
} else if state.error != nil {
if index + 1 == sources.count {
failureView()
} else {
ImageViewBackgroundA(index: index + 1,
sources: sources,
placeholderView: placeholderView,
failureView: failureView)
}
} else {
placeholderView()
} }
} }
.pipeline(ImagePipeline(configuration: .withDataCache)) .pipeline(ImagePipeline(configuration: .withDataCache))

View File

@ -27,7 +27,11 @@ struct CinematicNextUpCardView: View {
ImageView(src: item.getSeriesBackdropImage(maxWidth: 350)) ImageView(src: item.getSeriesBackdropImage(maxWidth: 350))
.frame(width: 350, height: 210) .frame(width: 350, height: 210)
} else { } else {
ImageView(src: item.getBackdropImage(maxWidth: 350)) ImageView(sources: [
item.getThumbImage(maxWidth: 320),
item.getBackdropImage(maxWidth: 320),
],
bh: item.getBackdropImageBlurHash())
.frame(width: 350, height: 210) .frame(width: 350, height: 210)
} }

View File

@ -28,7 +28,11 @@ struct CinematicResumeCardView: View {
ImageView(src: item.getSeriesBackdropImage(maxWidth: 350)) ImageView(src: item.getSeriesBackdropImage(maxWidth: 350))
.frame(width: 350, height: 210) .frame(width: 350, height: 210)
} else { } else {
ImageView(src: item.getBackdropImage(maxWidth: 350)) ImageView(sources: [
item.getThumbImage(maxWidth: 320),
item.getBackdropImage(maxWidth: 320),
],
bh: item.getBackdropImageBlurHash())
.frame(width: 350, height: 210) .frame(width: 350, height: 210)
} }

View File

@ -27,7 +27,11 @@ struct ContinueWatchingView: View {
VStack(alignment: .leading) { VStack(alignment: .leading) {
ZStack { ZStack {
ImageView(src: item.getBackdropImage(maxWidth: 320), bh: item.getBackdropImageBlurHash()) ImageView(sources: [
item.getThumbImage(maxWidth: 320),
item.getBackdropImage(maxWidth: 320),
],
bh: item.getBackdropImageBlurHash())
.frame(width: 320, height: 180) .frame(width: 320, height: 180)
.accessibilityIgnoresInvertColors() .accessibilityIgnoresInvertColors()