Signal-iOS/SignalUI/ViewControllers/Stickers/StickerView.swift
2022-04-13 20:21:47 -07:00

162 lines
5.5 KiB
Swift

//
// Copyright (c) 2022 Open Whisper Systems. All rights reserved.
//
import Foundation
import YYImage
import Lottie
@objc
public class StickerView: NSObject {
// Never instantiate this class.
private override init() {}
public static func stickerView(forStickerInfo stickerInfo: StickerInfo,
dataSource: StickerPackDataSource,
size: CGFloat? = nil) -> UIView? {
guard let stickerMetadata = dataSource.metadata(forSticker: stickerInfo) else {
Logger.warn("Missing sticker metadata.")
return nil
}
return stickerView(stickerInfo: stickerInfo, stickerMetadata: stickerMetadata, size: size)
}
public static func stickerView(forInstalledStickerInfo stickerInfo: StickerInfo,
size: CGFloat? = nil) -> UIView? {
let metadata = StickerManager.installedStickerMetadataWithSneakyTransaction(stickerInfo: stickerInfo)
guard let stickerMetadata = metadata else {
Logger.warn("Missing sticker metadata.")
return nil
}
return stickerView(stickerInfo: stickerInfo, stickerMetadata: stickerMetadata, size: size)
}
private static func stickerView(stickerInfo: StickerInfo,
stickerMetadata: StickerMetadata,
size: CGFloat? = nil) -> UIView? {
let stickerDataUrl = stickerMetadata.stickerDataUrl
guard let stickerView = self.stickerView(stickerInfo: stickerInfo,
stickerType: stickerMetadata.stickerType,
stickerDataUrl: stickerDataUrl) else {
Logger.warn("Could not load sticker for display.")
return nil
}
if let size = size {
stickerView.autoSetDimensions(to: CGSize(square: size))
}
return stickerView
}
static func stickerView(stickerInfo: StickerInfo,
stickerType: StickerType,
stickerDataUrl: URL) -> UIView? {
guard OWSFileSystem.fileOrFolderExists(url: stickerDataUrl) else {
Logger.warn("Sticker path does not exist: \(stickerDataUrl).")
return nil
}
guard NSData.ows_isValidImage(at: stickerDataUrl, mimeType: stickerType.contentType) else {
owsFailDebug("Invalid sticker.")
return nil
}
let stickerView: UIView
switch stickerType {
case .webp, .apng, .gif:
guard let stickerImage = YYImage(contentsOfFile: stickerDataUrl.path) else {
owsFailDebug("Sticker could not be parsed.")
return nil
}
let yyView = YYAnimatedImageView()
yyView.alwaysInfiniteLoop = true
yyView.contentMode = .scaleAspectFit
yyView.image = stickerImage
stickerView = yyView
case .signalLottie:
let lottieView = Lottie.AnimationView(filePath: stickerDataUrl.path)
lottieView.contentMode = .scaleAspectFit
lottieView.animationSpeed = 1
lottieView.loopMode = .loop
lottieView.backgroundBehavior = .pause
lottieView.play()
stickerView = lottieView
}
return stickerView
}
}
public class StickerPlaceholderView: UIView {
let placeholderView = UIView()
public init(color: UIColor) {
super.init(frame: .zero)
placeholderView.backgroundColor = color
addSubview(placeholderView)
placeholderView.autoPinEdgesToSuperviewMargins()
if #available(iOS 13, *) { placeholderView.layer.cornerCurve = .continuous }
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
public override func layoutSubviews() {
super.layoutSubviews()
layoutMargins = UIEdgeInsets(hMargin: width / 8, vMargin: height / 8)
placeholderView.layer.cornerRadius = placeholderView.width / 3
}
}
@objc
public class StickerReusableView: UIView {
public var hasStickerView: Bool { stickerView != nil }
private weak var stickerView: UIView?
public func configure(with stickerView: UIView) {
guard stickerView != self.stickerView else { return }
self.stickerView = stickerView
addSubview(stickerView)
stickerView.autoPinEdgesToSuperviewEdges()
if let placeholderView = placeholderView {
self.placeholderView = nil
stickerView.alpha = 0
UIView.animate(withDuration: 0.2) {
stickerView.alpha = 1
placeholderView.alpha = 0
} completion: { _ in
placeholderView.removeFromSuperview()
}
}
}
private weak var placeholderView: StickerPlaceholderView?
public func showPlaceholder(color: UIColor = Theme.secondaryBackgroundColor) {
guard placeholderView == nil else { return }
let placeholderView = StickerPlaceholderView(color: color)
self.placeholderView = placeholderView
addSubview(placeholderView)
placeholderView.autoPinEdgesToSuperviewEdges()
if let stickerView = stickerView {
self.stickerView = nil
placeholderView.alpha = 0
UIView.animate(withDuration: 0.2) {
placeholderView.alpha = 1
stickerView.alpha = 0
} completion: { _ in
stickerView.removeFromSuperview()
}
}
}
}