mirror of https://github.com/oxen-io/session-ios
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
130 lines
4.4 KiB
Swift
130 lines
4.4 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import UIKit
|
|
import CryptoKit
|
|
import SessionUtilitiesKit
|
|
|
|
public class PlaceholderIcon {
|
|
private static let placeholderCache: Atomic<NSCache<NSString, UIImage>> = {
|
|
let result = NSCache<NSString, UIImage>()
|
|
result.countLimit = 50
|
|
|
|
return Atomic(result)
|
|
}()
|
|
|
|
private let seed: Int
|
|
|
|
// Colour palette
|
|
private var colors: [UIColor] = Theme.PrimaryColor.allCases.map { $0.color }
|
|
|
|
// MARK: - Initialization
|
|
|
|
init(seed: Int, colors: [UIColor]? = nil) {
|
|
self.seed = seed
|
|
if let colors = colors { self.colors = colors }
|
|
}
|
|
|
|
convenience init(seed: String, colors: [UIColor]? = nil) {
|
|
// Ensure we have a correct hash
|
|
var hash = seed
|
|
|
|
if (hash.matches("^[0-9A-Fa-f]+$") && hash.count >= 12) {
|
|
hash = Data(SHA512.hash(data: Data(Array(seed.utf8))).makeIterator()).toHexString()
|
|
}
|
|
|
|
guard let number = Int(hash.substring(to: 12), radix: 16) else {
|
|
SNLog("Failed to generate number from seed string: \(seed).")
|
|
self.init(seed: 0, colors: colors)
|
|
return
|
|
}
|
|
|
|
self.init(seed: number, colors: colors)
|
|
}
|
|
|
|
// MARK: - Convenience
|
|
|
|
public static func generate(seed: String, text: String, size: CGFloat) -> UIImage {
|
|
let icon = PlaceholderIcon(seed: seed)
|
|
|
|
var content: String = (text.hasSuffix("\(String(seed.suffix(4))))") ?
|
|
(text.split(separator: "(")
|
|
.first
|
|
.map { String($0) })
|
|
.defaulting(to: text) :
|
|
text
|
|
)
|
|
|
|
if content.count > 2 && (try? SessionId.Prefix(from: content)) != nil {
|
|
content.removeFirst(2)
|
|
}
|
|
|
|
let initials: String = content
|
|
.split(separator: " ")
|
|
.compactMap { word in word.first.map { String($0) } }
|
|
.joined()
|
|
let cacheKey: String = "\(content)-\(Int(floor(size)))"
|
|
|
|
if let cachedIcon: UIImage = placeholderCache.wrappedValue.object(forKey: cacheKey as NSString) {
|
|
return cachedIcon
|
|
}
|
|
|
|
let layer = icon.generateLayer(
|
|
with: size,
|
|
text: (initials.count >= 2 ?
|
|
initials.substring(to: 2).uppercased() :
|
|
content.substring(to: 2).uppercased()
|
|
)
|
|
)
|
|
|
|
let rect = CGRect(origin: CGPoint.zero, size: layer.frame.size)
|
|
let renderer = UIGraphicsImageRenderer(size: rect.size)
|
|
let result = renderer.image { layer.render(in: $0.cgContext) }
|
|
|
|
placeholderCache.mutate { $0.setObject(result, forKey: cacheKey as NSString) }
|
|
|
|
return result
|
|
}
|
|
|
|
// MARK: - Internal
|
|
|
|
private func generateLayer(with diameter: CGFloat, text: String) -> CALayer {
|
|
let color: UIColor = self.colors[seed % self.colors.count]
|
|
let base: CALayer = getTextLayer(with: diameter, color: color, text: text)
|
|
base.masksToBounds = true
|
|
|
|
return base
|
|
}
|
|
|
|
private func getTextLayer(with diameter: CGFloat, color: UIColor, text: String) -> CALayer {
|
|
let font = UIFont.boldSystemFont(ofSize: diameter / 2)
|
|
let height = NSString(string: text).boundingRect(with: CGSize(width: diameter, height: CGFloat.greatestFiniteMagnitude),
|
|
options: .usesLineFragmentOrigin, attributes: [ NSAttributedString.Key.font : font ], context: nil).height
|
|
let frame = CGRect(x: 0, y: (diameter - height) / 2, width: diameter, height: height)
|
|
|
|
let layer = CATextLayer()
|
|
layer.frame = frame
|
|
layer.themeForegroundColorForced = .color(.white)
|
|
layer.contentsScale = UIScreen.main.scale
|
|
|
|
let fontName = font.fontName
|
|
let fontRef = CGFont(fontName as CFString)
|
|
layer.font = fontRef
|
|
layer.fontSize = font.pointSize
|
|
layer.alignmentMode = .center
|
|
layer.string = text
|
|
|
|
let base = CALayer()
|
|
base.frame = CGRect(x: 0, y: 0, width: diameter, height: diameter)
|
|
base.themeBackgroundColorForced = .color(color)
|
|
base.addSublayer(layer)
|
|
|
|
return base
|
|
}
|
|
}
|
|
|
|
private extension String {
|
|
func matches(_ regex: String) -> Bool {
|
|
return self.range(of: regex, options: .regularExpression, range: nil, locale: nil) != nil
|
|
}
|
|
}
|