// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved. import UIKit import CryptoKit import SessionUtilitiesKit public class PlaceholderIcon { private static let placeholderCache: Atomic> = { let result = NSCache() 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 = SHA512.hash(data: Data(seed.bytes)).hexString } 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 && 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 } }