Swiftfin/Shared/Components/ImageView.swift
2024-05-22 13:45:48 +09:00

143 lines
4.0 KiB
Swift

//
// Swiftfin is subject to the terms of the Mozilla Public
// License, v2.0. If a copy of the MPL was not distributed with this
// file, you can obtain one at https://mozilla.org/MPL/2.0/.
//
// Copyright (c) 2024 Jellyfin & Jellyfin Contributors
//
import BlurHashKit
import Nuke
import NukeUI
import SwiftUI
// TODO: currently SVGs are only supported for logos, which are only used in a few places.
// make it so when displaying an SVG there is a unified `image` caller modifier
// TODO: `LazyImage` uses a transaction for view swapping, which will fade out old views
// and fade in new views, causing a black "flash" between the placeholder and final image.
// Since we use blur hashes, we actually just want the final image to fade in on top while
// the blur hash view is at full opacity.
// - refactor for option
// - take a look at `RotateContentView`
struct ImageView: View {
@State
private var sources: [ImageSource]
private var image: (Image) -> any View
private var pipeline: ImagePipeline
private var placeholder: ((ImageSource) -> any View)?
private var failure: () -> any View
@ViewBuilder
private func _placeholder(_ currentSource: ImageSource) -> some View {
if let placeholder = placeholder {
placeholder(currentSource)
.eraseToAnyView()
} else {
DefaultPlaceholderView(blurHash: currentSource.blurHash)
}
}
var body: some View {
if let currentSource = sources.first {
LazyImage(url: currentSource.url, transaction: .init(animation: .linear)) { state in
if state.isLoading {
_placeholder(currentSource)
} else if let _image = state.image {
if let data = state.imageContainer?.data {
FastSVGView(data: data)
} else {
image(_image.resizable())
.eraseToAnyView()
}
} else if state.error != nil {
failure()
.eraseToAnyView()
.onAppear {
sources.removeFirstSafe()
}
}
}
.pipeline(pipeline)
} else {
failure()
.eraseToAnyView()
}
}
}
extension ImageView {
init(_ source: ImageSource) {
self.init([source].compacted(using: \.url))
}
init(_ sources: [ImageSource]) {
self.init(
sources: sources.compacted(using: \.url),
image: { $0 },
pipeline: .shared,
placeholder: nil,
failure: { EmptyView() }
)
}
init(_ source: URL?) {
self.init([ImageSource(url: source)])
}
init(_ sources: [URL?]) {
let imageSources = sources
.compacted()
.map { ImageSource(url: $0) }
self.init(imageSources)
}
}
// MARK: Modifiers
extension ImageView {
func image(@ViewBuilder _ content: @escaping (Image) -> any View) -> Self {
copy(modifying: \.image, with: content)
}
func pipeline(_ pipeline: ImagePipeline) -> Self {
copy(modifying: \.pipeline, with: pipeline)
}
func placeholder(@ViewBuilder _ content: @escaping (ImageSource) -> any View) -> Self {
copy(modifying: \.placeholder, with: content)
}
func failure(@ViewBuilder _ content: @escaping () -> any View) -> Self {
copy(modifying: \.failure, with: content)
}
}
// MARK: Defaults
extension ImageView {
struct DefaultFailureView: View {
var body: some View {
Color.secondarySystemFill
.opacity(0.75)
}
}
struct DefaultPlaceholderView: View {
let blurHash: String?
var body: some View {
if let blurHash {
BlurHashView(blurHash: blurHash, size: .Square(length: 8))
}
}
}
}