TM-SGNL-iOS/SignalUI/RecipientPickers/ContactCellView.swift
TeleMessage developers dde0620daf initial commit
2025-05-03 12:28:28 -07:00

368 lines
14 KiB
Swift

//
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
//
import Foundation
public import SignalServiceKit
public class ContactCellAccessoryView: NSObject {
let accessoryView: UIView
let size: CGSize
public init(accessoryView: UIView, size: CGSize) {
self.accessoryView = accessoryView
self.size = size
}
}
// MARK: -
public class ContactCellConfiguration: NSObject {
fileprivate enum CellDataSource {
case address(SignalServiceAddress)
case groupThread(TSGroupThread)
case `static`(name: String, avatar: UIImage)
}
fileprivate let dataSource: CellDataSource
public let localUserDisplayMode: LocalUserDisplayMode
public var forceDarkAppearance = false
public var accessoryMessage: String?
public var customName: String?
public var accessoryView: ContactCellAccessoryView?
public var attributedSubtitle: NSAttributedString?
public var shouldShowContactIcon = false
public var allowUserInteraction = false
public var badged = true // TODO: Badges Default false? Configure each use-case?
public var storyState: StoryContextViewState?
public var hasAccessoryText: Bool {
accessoryMessage?.nilIfEmpty != nil
}
public init(address: SignalServiceAddress, localUserDisplayMode: LocalUserDisplayMode) {
self.dataSource = .address(address)
self.localUserDisplayMode = localUserDisplayMode
super.init()
}
public init(groupThread: TSGroupThread, localUserDisplayMode: LocalUserDisplayMode) {
self.dataSource = .groupThread(groupThread)
self.localUserDisplayMode = localUserDisplayMode
super.init()
}
public init(name: String, avatar: UIImage) {
self.dataSource = .static(name: name, avatar: avatar)
self.localUserDisplayMode = .asUser
super.init()
}
public func useVerifiedSubtitle() {
let text = NSMutableAttributedString()
text.append(SignalSymbol.safetyNumber.attributedString(for: .caption1, clamped: true))
text.append(" ", attributes: [:])
text.append(SafetyNumberStrings.verified, attributes: [:])
self.attributedSubtitle = text
}
}
// MARK: -
public class ContactCellView: ManualStackView {
private var configuration: ContactCellConfiguration? {
didSet {
ensureObservers()
}
}
public static var avatarSizeClass: ConversationAvatarView.Configuration.SizeClass { .thirtySix }
private var avatarDataSource: ConversationAvatarDataSource? {
switch configuration?.dataSource {
case .groupThread(let thread): return .thread(thread)
case .address(let address): return .address(address)
case .static(_, let avatar): return .asset(avatar: avatar, badge: nil)
case nil: return nil
}
}
// TODO: Update localUserDisplayMode.
private let avatarView = ConversationAvatarView(
sizeClass: avatarSizeClass,
localUserDisplayMode: .asUser,
useAutolayout: false)
public var tooltipTailReferenceView: UIView { return avatarView }
public static let avatarTextHSpacing: CGFloat = 12
private lazy var groupStoryBadgeView: UIView = {
let backgroundView = UIView.container()
let symbolView = UIImageView(image: UIImage(named: "stories-fill-compact"))
backgroundView.addSubview(symbolView)
symbolView.tintColor = .ows_white
symbolView.autoSetDimensions(to: .square(12))
symbolView.autoCenterInSuperview()
return backgroundView
}()
private let nameLabel = CVLabel()
private let subtitleLabel = CVLabel()
private let accessoryLabel = CVLabel()
private let textStack = ManualStackView(name: "textStack")
public init() {
super.init(name: "ContactCellView")
nameLabel.lineBreakMode = .byTruncatingTail
accessoryLabel.textAlignment = .right
self.shouldDeactivateConstraints = false
}
private var nameLabelFont: UIFont { OWSTableItem.primaryLabelFont }
fileprivate static var subtitleFont: UIFont { .dynamicTypeCaption1Clamped }
private func nameLabelColor(forceDarkAppearance: Bool) -> UIColor {
forceDarkAppearance ? Theme.darkThemePrimaryColor : Theme.primaryTextColor
}
private func configureFontsAndColors(forceDarkAppearance: Bool) {
nameLabel.font = nameLabelFont
subtitleLabel.font = Self.subtitleFont
accessoryLabel.font = .dynamicTypeSubheadlineClamped
nameLabel.textColor = nameLabelColor(forceDarkAppearance: forceDarkAppearance)
subtitleLabel.textColor = (forceDarkAppearance ? Theme.darkThemeSecondaryTextAndIconColor : Theme.secondaryTextAndIconColor)
accessoryLabel.textColor = Theme.isDarkThemeEnabled ? .ows_gray25 : .ows_gray45
if let nameLabelText = nameLabel.attributedText?.string.nilIfEmpty,
let nameLabelColor = nameLabel.textColor {
nameLabel.attributedText = nameLabelText.asAttributedString(attributes: [
.foregroundColor: nameLabelColor
])
}
}
public func configure(configuration: ContactCellConfiguration,
transaction: SDSAnyReadTransaction) {
AssertIsOnMainThread()
owsAssertDebug(!shouldDeactivateConstraints)
self.configuration = configuration
self.isUserInteractionEnabled = configuration.allowUserInteraction
avatarView.update(transaction) { config in
config.dataSource = avatarDataSource
config.addBadgeIfApplicable = configuration.badged
config.localUserDisplayMode = configuration.localUserDisplayMode
if let storyState = configuration.storyState {
config.storyConfiguration = .fixed(storyState)
} else {
config.storyConfiguration = .disabled
}
}
if avatarDataSource?.isGroupAvatar ?? false,
let storyState = configuration.storyState {
// Group story. Add badge
avatarView.addSubview(groupStoryBadgeView)
let badgeColor: UIColor
switch storyState {
case .unviewed:
badgeColor = .ows_accentBlue
case .viewed, .noStories:
badgeColor = Theme.isDarkThemeEnabled ? .ows_gray65 : .ows_gray25
}
let size: CGFloat = 20
groupStoryBadgeView.backgroundColor = badgeColor
groupStoryBadgeView.layer.cornerRadius = size/2
groupStoryBadgeView.layer.masksToBounds = true
groupStoryBadgeView.autoSetDimensions(to: .square(size))
groupStoryBadgeView.autoPinEdge(toSuperviewEdge: .bottom, withInset: -2)
groupStoryBadgeView.autoPinEdge(toSuperviewEdge: .trailing, withInset: -5)
} else {
// Not a group or not a story. Remove badge
groupStoryBadgeView.removeFromSuperview()
}
// Update fonts to reflect changes to dynamic type.
configureFontsAndColors(forceDarkAppearance: configuration.forceDarkAppearance)
updateNameLabels(configuration: configuration, transaction: transaction)
// Configure self.
do {
var rootStackSubviews: [UIView] = [ avatarView ]
let avatarSize = Self.avatarSizeClass.size
var rootStackSubviewInfos = [ avatarSize.asManualSubviewInfo(hasFixedSize: true) ]
// Configure textStack.
do {
var textStackSubviews = [ nameLabel ]
let nameSize = nameLabel.sizeThatFits(.square(.greatestFiniteMagnitude))
var textStackSubviewInfos = [ nameSize.asManualSubviewInfo ]
if let attributedSubtitle = configuration.attributedSubtitle?.nilIfEmpty {
subtitleLabel.attributedText = attributedSubtitle
textStackSubviews.append(subtitleLabel)
let subtitleSize = subtitleLabel.sizeThatFits(.square(.greatestFiniteMagnitude))
textStackSubviewInfos.append(subtitleSize.asManualSubviewInfo)
}
let textStackConfig = ManualStackView.Config(axis: .vertical,
alignment: .leading,
spacing: 0,
layoutMargins: .zero)
let textStackMeasurement = textStack.configure(config: textStackConfig,
subviews: textStackSubviews,
subviewInfos: textStackSubviewInfos)
rootStackSubviews.append(textStack)
rootStackSubviewInfos.append(textStackMeasurement.measuredSize.asManualSubviewInfo)
}
if let accessoryMessage = configuration.accessoryMessage {
accessoryLabel.text = accessoryMessage
let labelSize = accessoryLabel.sizeThatFits(.square(.greatestFiniteMagnitude))
configuration.accessoryView = ContactCellAccessoryView(accessoryView: accessoryLabel,
size: labelSize)
}
if let accessoryView = configuration.accessoryView {
rootStackSubviews.append(accessoryView.accessoryView)
rootStackSubviewInfos.append(accessoryView.size.asManualSubviewInfo(hasFixedSize: true))
}
let rootStackConfig = ManualStackView.Config(axis: .horizontal,
alignment: .center,
spacing: Self.avatarTextHSpacing,
layoutMargins: .zero)
let rootStackMeasurement = ManualStackView.measure(config: rootStackConfig,
subviewInfos: rootStackSubviewInfos)
self.configure(config: rootStackConfig,
measurement: rootStackMeasurement,
subviews: rootStackSubviews)
}
}
// MARK: - Notifications
private func ensureObservers() {
NotificationCenter.default.removeObserver(self)
if case .address = configuration?.dataSource {
NotificationCenter.default.addObserver(self,
selector: #selector(otherUsersProfileChanged(notification:)),
name: UserProfileNotifications.otherUsersProfileDidChange,
object: nil)
}
}
// MARK: -
private func updateNameLabelsWithSneakyTransaction(configuration: ContactCellConfiguration) {
SSKEnvironment.shared.databaseStorageRef.read { transaction in
updateNameLabels(configuration: configuration, transaction: transaction)
}
}
private func updateNameLabels(configuration: ContactCellConfiguration,
transaction: SDSAnyReadTransaction) {
AssertIsOnMainThread()
let textColor = self.nameLabelColor(forceDarkAppearance: configuration.forceDarkAppearance)
let nameString = { () -> NSAttributedString in
if let customName = configuration.customName?.nilIfEmpty {
return customName.asAttributedString
}
switch configuration.dataSource {
case .address(let address):
let name = SSKEnvironment.shared.contactManagerRef.nameForAddress(
address,
localUserDisplayMode: configuration.localUserDisplayMode,
short: false,
transaction: transaction
)
switch (address.isLocalAddress, configuration.localUserDisplayMode) {
case (false, _), (true, .asLocalUser), (true, .asUser):
return name
case (true, .noteToSelf):
let verifiedIcon = NSAttributedString.with(
image: Theme.iconImage(.official),
font: .dynamicTypeSubheadline,
centerVerticallyRelativeTo: .dynamicTypeBody
)
return name.stringByAppendingString(" ").stringByAppendingString(verifiedIcon)
}
case .groupThread(let thread):
let threadName = SSKEnvironment.shared.contactManagerRef.displayName(for: thread, transaction: transaction)
return threadName.asAttributedString(attributes: [
.foregroundColor: textColor,
])
case .static(let name, _):
return name.asAttributedString
}
}()
if configuration.shouldShowContactIcon {
let contactIcon = SignalSymbol.personCircle.attributedString(
dynamicTypeBaseSize: 14,
weight: .bold,
leadingCharacter: .space,
attributes: [.foregroundColor: textColor]
)
nameLabel.attributedText = nameString.stringByAppendingString(contactIcon)
} else {
nameLabel.attributedText = nameString
}
}
public override func reset() {
super.reset()
NotificationCenter.default.removeObserver(self)
configuration = nil
avatarView.reset()
textStack.reset()
nameLabel.text = nil
subtitleLabel.text = nil
accessoryLabel.text = nil
}
@objc
private func otherUsersProfileChanged(notification: Notification) {
AssertIsOnMainThread()
guard let configuration = self.configuration else {
return
}
guard let changedAddress = notification.userInfo?[UserProfileNotifications.profileAddressKey] as? SignalServiceAddress,
changedAddress.isValid else {
owsFailDebug("changedAddress was unexpectedly nil")
return
}
if case .address(changedAddress) = configuration.dataSource {
updateNameLabelsWithSneakyTransaction(configuration: configuration)
}
}
}