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.
134 lines
4.5 KiB
Swift
134 lines
4.5 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import UIKit
|
|
import CryptoKit
|
|
|
|
public class PlaceholderIcon {
|
|
private static let colors: [UIColor] = Theme.PrimaryColor.allCases.map { $0.color }
|
|
|
|
private let seed: Int
|
|
|
|
// MARK: - Initialization
|
|
|
|
init(seed: Int) {
|
|
self.seed = seed
|
|
}
|
|
|
|
// stringlint:ignore_contents
|
|
convenience init(seed: String) {
|
|
// Ensure we have a correct hash
|
|
var hash = seed
|
|
|
|
if (hash.matches("^[0-9A-Fa-f]+$") && hash.count >= 12) {
|
|
// This is the same as the `SessionUtilitiesKit` `toHexString` function
|
|
hash = Data(SHA512.hash(data: Data(Array(seed.utf8))).makeIterator())
|
|
.map { String(format: "%02x", $0) }.joined()
|
|
}
|
|
|
|
guard let number = Int(String(hash.prefix(12)), radix: 16) else {
|
|
self.init(seed: 0)
|
|
return
|
|
}
|
|
|
|
self.init(seed: number)
|
|
}
|
|
|
|
// MARK: - Convenience
|
|
|
|
// stringlint:ignore_contents
|
|
public static func generate(seed: String, text: String, size: CGFloat) -> UIImage {
|
|
let icon = PlaceholderIcon(seed: seed)
|
|
|
|
var content: String = {
|
|
guard text.hasSuffix("\(String(seed.suffix(4))))") else {
|
|
guard let result: String = text.split(separator: "(").first.map({ String($0) }) else {
|
|
return text
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
return text
|
|
}()
|
|
|
|
if ValidSessionIdPrefixes.hasValidPrefix(content) {
|
|
content.removeFirst(2)
|
|
}
|
|
|
|
let initials: String = content
|
|
.split(separator: " ")
|
|
.compactMap { word in word.first.map { String($0) } }
|
|
.joined()
|
|
|
|
return SNUIKit.placeholderIconCacher(cacheKey: "\(seed)-\(Int(floor(size)))") {
|
|
let layer = icon.generateLayer(
|
|
with: size,
|
|
text: (initials.count >= 2 ?
|
|
String(initials.prefix(2)).uppercased() :
|
|
String(content.prefix(2)).uppercased()
|
|
)
|
|
)
|
|
|
|
let rect = CGRect(origin: CGPoint.zero, size: layer.frame.size)
|
|
let renderer = UIGraphicsImageRenderer(size: rect.size)
|
|
|
|
return renderer.image { layer.render(in: $0.cgContext) }
|
|
}
|
|
}
|
|
|
|
// MARK: - Internal
|
|
|
|
private func generateLayer(with diameter: CGFloat, text: String) -> CALayer {
|
|
let color: UIColor = PlaceholderIcon.colors[seed % PlaceholderIcon.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
|
|
}
|
|
}
|
|
|
|
/// These enums should always match the `SessionUtilitiesKit.SessionId.Prefix` cases
|
|
private enum ValidSessionIdPrefixes: String, CaseIterable {
|
|
case standard = "05"
|
|
case blinded15 = "15"
|
|
case blinded25 = "25"
|
|
case unblinded = "00"
|
|
case group = "03"
|
|
|
|
static func hasValidPrefix(_ value: String) -> Bool {
|
|
return value.count >= 2 && allCases.map({ $0.rawValue }).contains(String(value.prefix(2)))
|
|
}
|
|
}
|