// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved. import UIKit import SessionUIKit import SessionMessagingKit import SignalUtilitiesKit import SessionUtilitiesKit public class MediaView: UIView { static let contentMode: UIView.ContentMode = .scaleAspectFill private enum MediaError { case missing case invalid case failed } // MARK: - private let dependencies: Dependencies private let mediaCache: NSCache? public let attachment: Attachment private let isOutgoing: Bool private let shouldSupressControls: Bool private var loadBlock: (() -> Void)? private var unloadBlock: (() -> Void)? // MARK: - LoadState // The loadState property allows us to: // // * Make sure we only have one load attempt // enqueued at a time for a given piece of media. // * We never retry media that can't be loaded. // * We skip media loads which are no longer // necessary by the time they reach the front // of the queue. enum LoadState: ThreadSafeType { case unloaded case loading case loaded case failed } @ThreadSafe private var loadState: LoadState = .unloaded // MARK: - Initializers public required init( mediaCache: NSCache? = nil, attachment: Attachment, isOutgoing: Bool, shouldSupressControls: Bool, cornerRadius: CGFloat, using dependencies: Dependencies ) { self.dependencies = dependencies self.mediaCache = mediaCache self.attachment = attachment self.isOutgoing = isOutgoing self.shouldSupressControls = shouldSupressControls super.init(frame: .zero) themeBackgroundColor = .backgroundSecondary clipsToBounds = true layer.masksToBounds = true layer.cornerRadius = cornerRadius createContents() } @available(*, unavailable, message: "use other init() instead.") required public init?(coder aDecoder: NSCoder) { fatalError("init(coder:) has not been implemented") } deinit { loadState = .unloaded } // MARK: - private func createContents() { Log.assertOnMainThread() guard attachment.state != .pendingDownload && attachment.state != .downloading else { addDownloadProgressIfNecessary() return } guard attachment.state != .failedDownload else { configure(forError: .failed) return } guard attachment.isValid else { configure(forError: .invalid) return } if attachment.isAnimated { configureForAnimatedImage(attachment: attachment) } else if attachment.isImage { configureForStillImage(attachment: attachment) } else if attachment.isVideo { configureForVideo(attachment: attachment) } else { Log.error("[MediaView] Attachment has unexpected type.") configure(forError: .invalid) } } private func addDownloadProgressIfNecessary() { guard attachment.state != .failedDownload else { configure(forError: .failed) return } guard attachment.state != .uploading && attachment.state != .uploaded else { // TODO: Show "restoring" indicator and possibly progress. configure(forError: .missing) return } themeBackgroundColor = .backgroundSecondary let loader = MediaLoaderView() addSubview(loader) loader.pin([ UIView.HorizontalEdge.left, UIView.VerticalEdge.bottom, UIView.HorizontalEdge.right ], to: self) } private func addUploadProgressIfNecessary(_ subview: UIView) -> Bool { guard isOutgoing else { return false } guard attachment.state != .failedUpload else { configure(forError: .failed) return false } // If this message was uploaded on a different device it'll now be seen as 'downloaded' (but // will still be outgoing - we don't want to show a loading indicator in this case) guard attachment.state != .uploaded && attachment.state != .downloaded else { return false } let loader = MediaLoaderView() addSubview(loader) loader.pin([ UIView.HorizontalEdge.left, UIView.VerticalEdge.bottom, UIView.HorizontalEdge.right ], to: self) return true } private func configureForAnimatedImage(attachment: Attachment) { let animatedImageView: AnimatedImageView = AnimatedImageView() // We need to specify a contentMode since the size of the image // might not match the aspect ratio of the view. animatedImageView.contentMode = MediaView.contentMode // Use trilinear filters for better scaling quality at // some performance cost. animatedImageView.layer.minificationFilter = .trilinear animatedImageView.layer.magnificationFilter = .trilinear animatedImageView.themeBackgroundColor = .backgroundSecondary animatedImageView.isHidden = !attachment.isValid addSubview(animatedImageView) animatedImageView.pin(to: self) _ = addUploadProgressIfNecessary(animatedImageView) loadBlock = { [weak self, dependencies] in Log.assertOnMainThread() if animatedImageView.image != nil { Log.error("[MediaView] Unexpectedly already loaded.") return } self?.tryToLoadMedia( loadMediaBlock: { applyMediaBlock in guard attachment.isValid else { self?.configure(forError: .invalid) return } guard let filePath: String = attachment.originalFilePath(using: dependencies) else { Log.error("[MediaView] Attachment stream missing original file path.") self?.configure(forError: .invalid) return } applyMediaBlock(filePath as AnyObject) }, applyMediaBlock: { filePath in Log.assertOnMainThread() guard let filePath: String = filePath as? String else { Log.error("[MediaView] Media has unexpected type: \(type(of: filePath))") self?.configure(forError: .invalid) return } // FIXME: Animated images flicker when reloading the cells (even though they are in the cache) animatedImageView.loadAnimatedImage(from: filePath) }, cacheKey: attachment.id ) } unloadBlock = { Log.assertOnMainThread() animatedImageView.image = nil } } private func configureForStillImage(attachment: Attachment) { let stillImageView = UIImageView() // We need to specify a contentMode since the size of the image // might not match the aspect ratio of the view. stillImageView.contentMode = MediaView.contentMode // Use trilinear filters for better scaling quality at // some performance cost. stillImageView.layer.minificationFilter = .trilinear stillImageView.layer.magnificationFilter = .trilinear stillImageView.themeBackgroundColor = .backgroundSecondary stillImageView.isHidden = !attachment.isValid addSubview(stillImageView) stillImageView.pin(to: self) _ = addUploadProgressIfNecessary(stillImageView) loadBlock = { [weak self, dependencies] in Log.assertOnMainThread() if stillImageView.image != nil { Log.error("[MediaView] Unexpectedly already loaded.") return } self?.tryToLoadMedia( loadMediaBlock: { applyMediaBlock in guard attachment.isValid else { self?.configure(forError: .invalid) return } attachment.thumbnail( size: .large, using: dependencies, success: { image, _ in applyMediaBlock(image) }, failure: { Log.error("[MediaView] Could not load thumbnail") self?.configure(forError: .invalid) } ) }, applyMediaBlock: { media in Log.assertOnMainThread() guard let image: UIImage = media as? UIImage else { Log.error("[MediaView] Media has unexpected type: \(type(of: media))") self?.configure(forError: .invalid) return } stillImageView.image = image }, cacheKey: attachment.id ) } unloadBlock = { Log.assertOnMainThread() stillImageView.image = nil } } private func configureForVideo(attachment: Attachment) { let stillImageView = UIImageView() // We need to specify a contentMode since the size of the image // might not match the aspect ratio of the view. stillImageView.contentMode = MediaView.contentMode // Use trilinear filters for better scaling quality at // some performance cost. stillImageView.layer.minificationFilter = .trilinear stillImageView.layer.magnificationFilter = .trilinear stillImageView.themeBackgroundColor = .backgroundSecondary stillImageView.isHidden = !attachment.isValid addSubview(stillImageView) stillImageView.pin(to: self) if !addUploadProgressIfNecessary(stillImageView) && !shouldSupressControls { if let duration: TimeInterval = attachment.duration { let fadeView: GradientView = GradientView() fadeView.themeBackgroundGradient = [ .value(.black, alpha: 0), .value(.black, alpha: 0.4) ] stillImageView.addSubview(fadeView) fadeView.set(.height, to: 40) fadeView.pin(.leading, to: .leading, of: stillImageView) fadeView.pin(.trailing, to: .trailing, of: stillImageView) fadeView.pin(.bottom, to: .bottom, of: stillImageView) let durationLabel: UILabel = UILabel() durationLabel.font = .systemFont(ofSize: Values.smallFontSize) durationLabel.text = Format.duration(duration) durationLabel.themeTextColor = .white stillImageView.addSubview(durationLabel) durationLabel.pin(.trailing, to: .trailing, of: stillImageView, withInset: -Values.smallSpacing) durationLabel.pin(.bottom, to: .bottom, of: stillImageView, withInset: -Values.smallSpacing) } // Add the play button above the duration label and fade let videoPlayIcon = UIImage(named: "CirclePlay") let videoPlayButton = UIImageView(image: videoPlayIcon) videoPlayButton.set(.width, to: 72) videoPlayButton.set(.height, to: 72) stillImageView.addSubview(videoPlayButton) videoPlayButton.center(in: stillImageView) } loadBlock = { [weak self, dependencies] in Log.assertOnMainThread() if stillImageView.image != nil { Log.error("[MediaView] Unexpectedly already loaded.") return } self?.tryToLoadMedia( loadMediaBlock: { applyMediaBlock in guard attachment.isValid else { self?.configure(forError: .invalid) return } attachment.thumbnail( size: .medium, using: dependencies, success: { image, _ in applyMediaBlock(image) }, failure: { Log.error("[MediaView] Could not load thumbnail") self?.configure(forError: .invalid) } ) }, applyMediaBlock: { media in Log.assertOnMainThread() guard let image: UIImage = media as? UIImage else { Log.error("[MediaView] Media has unexpected type: \(type(of: media))") self?.configure(forError: .invalid) return } stillImageView.image = image }, cacheKey: attachment.id ) } unloadBlock = { Log.assertOnMainThread() stillImageView.image = nil } } private func configure(forError error: MediaError) { // When there is a failure in the 'loadMediaBlock' closure this can be called // on a background thread - rather than dispatching in every 'loadMediaBlock' // usage we just do so here guard Thread.isMainThread else { DispatchQueue.main.async { self.configure(forError: error) } return } let icon: UIImage switch error { case .failed: guard let asset = UIImage(named: "media_retry") else { Log.error("[MediaView] Missing image") return } icon = asset self.isAccessibilityElement = true self.accessibilityIdentifier = "Media retry" case .invalid: guard let asset = UIImage(named: "media_invalid") else { Log.error("[MediaView] Missing image") return } icon = asset self.isAccessibilityElement = true self.accessibilityIdentifier = "Media invalid" case .missing: return } themeBackgroundColor = .backgroundSecondary // For failed ougoing messages add an overlay to make the icon more visible if isOutgoing { let attachmentOverlayView: UIView = UIView() attachmentOverlayView.themeBackgroundColor = .messageBubble_overlay addSubview(attachmentOverlayView) attachmentOverlayView.pin(to: self) } let iconView = UIImageView(image: icon.withRenderingMode(.alwaysTemplate)) iconView.themeTintColor = .textPrimary iconView.alpha = Values.mediumOpacity addSubview(iconView) iconView.center(in: self) } private func tryToLoadMedia( loadMediaBlock: @escaping (@escaping (AnyObject?) -> Void) -> Void, applyMediaBlock: @escaping (AnyObject) -> Void, cacheKey: String ) { // It's critical that we update loadState once // our load attempt is complete. let loadCompletion: (AnyObject?) -> Void = { [weak self] possibleMedia in guard self?.loadState == .loading else { Log.verbose("[MediaView] Skipping obsolete load.") return } guard let media: AnyObject = possibleMedia else { self?.loadState = .failed // TODO: // [self showAttachmentErrorViewWithMediaView:mediaView]; return } applyMediaBlock(media) self?.mediaCache?.setObject(media, forKey: cacheKey as NSString) self?.loadState = .loaded } guard loadState == .loading else { Log.error("[MediaView] Unexpected load state: \(loadState)") return } if let media: AnyObject = self.mediaCache?.object(forKey: cacheKey as NSString) { Log.verbose("[MediaView] media cache hit") guard Thread.isMainThread else { DispatchQueue.main.async { loadCompletion(media) } return } loadCompletion(media) return } Log.verbose("[MediaView] media cache miss") MediaView.loadQueue.async { [weak self] in guard self?.loadState == .loading else { Log.verbose("[MediaView] Skipping obsolete load.") return } loadMediaBlock { media in guard Thread.isMainThread else { DispatchQueue.main.async { loadCompletion(media) } return } loadCompletion(media) } } } // We use this queue to perform the media loads. // These loads are expensive, so we want to: // // * Do them off the main thread. // * Only do one at a time. // * Avoid this work if possible (obsolete loads for // views that are no longer visible, redundant loads // of media already being loaded, don't retry media // that can't be loaded, etc.). // * Do them in _reverse_ order. More recently enqueued // loads more closely reflect the current view state. // By processing in reverse order, we improve our // "skip rate" of obsolete loads. private static let loadQueue = ReverseDispatchQueue(label: "org.signal.asyncMediaLoadQueue") public func loadMedia() { switch loadState { case .unloaded: loadState = .loading loadBlock?() case .loading, .loaded, .failed: break } } public func unloadMedia() { loadState = .unloaded unloadBlock?() } } // MARK: - SwiftUI import SwiftUI struct MediaView_SwiftUI: UIViewRepresentable { public typealias UIViewType = MediaView private let dependencies: Dependencies private let mediaCache: NSCache? public let attachment: Attachment private let isOutgoing: Bool private let shouldSupressControls: Bool private let cornerRadius: CGFloat public init( mediaCache: NSCache? = nil, attachment: Attachment, isOutgoing: Bool, shouldSupressControls: Bool, cornerRadius: CGFloat, using dependencies: Dependencies ) { self.dependencies = dependencies self.mediaCache = mediaCache self.attachment = attachment self.isOutgoing = isOutgoing self.shouldSupressControls = shouldSupressControls self.cornerRadius = cornerRadius } func makeUIView(context: Context) -> MediaView { let mediaView = MediaView( mediaCache: mediaCache, attachment: attachment, isOutgoing: isOutgoing, shouldSupressControls: shouldSupressControls, cornerRadius: cornerRadius, using: dependencies ) return mediaView } func updateUIView(_ mediaView: MediaView, context: Context) { mediaView.loadMedia() } }