Signal-iOS/SignalUI/ViewControllers/Stickers/StickerView.swift
Evan Hahn 370ff654e7
Change license to AGPL
Change license to AGPL

This commit:

- Updates the `LICENSE` file

- Start every file with something like:

      // Copyright YEAR_FIRST_PUBLISHED Signal Messenger, LLC
      // SPDX-License-Identifier: AGPL-3.0-only

---

First, I removed existing license headers with this Ruby 3.1.2 script:

    require 'set'

    EXTENSIONS_TO_CHECK = Set['.h', '.hpp', '.cpp', '.m', '.mm', '.pch', '.swift']

    same = 0
    different = 0

    all_files = `git ls-files`.lines.map { |line| line.strip }
    all_files.each do |relative_path|
      if relative_path == 'Pods'
        next
      end

      unless EXTENSIONS_TO_CHECK.include? File.extname(relative_path)
        next
      end

      path = File.expand_path(relative_path)

      contents = File.read(path)
      new_contents = contents.sub(/\/\/\n\/\/  Copyright .*\n\/\/\n\n/, '')

      if contents == new_contents
        same += 1
      else
        different += 1
      end

      File.write(path, new_contents)
    end

    puts "updated #{different} file(s), left #{same} untouched"

I'm sure this script could be improved, but it worked well enough.

Then, I created `Scripts/lint/lint-license-headers` and ran it to auto-
fix a lot of files. This changed the mode of some files, but I think
that's actually desirable. For example,
`SignalServiceKit/src/Util/AppContext.m` previously had a mode of
`0755/-rwxr-xr-x`, and it's now `0644/-rw-r--r--`.

Then I fixed some stragglers and updated the precommit script.

See [a similar change in the Desktop app][0].

[0]: 8bfaf598af
2022-10-13 08:25:37 -05:00

163 lines
5.6 KiB
Swift

//
// Copyright 2019 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
//
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()
}
}
}
}