// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved. import UIKit import SessionUIKit import SessionMessagingKit import SessionUtilitiesKit final class ConversationTitleView: UIView { private static let leftInset: CGFloat = 8 private static let leftInsetWithCallButton: CGFloat = 54 private var oldSize: CGSize = .zero override var intrinsicContentSize: CGSize { return UIView.layoutFittingExpandedSize } // MARK: - UI Components private lazy var stackViewLeadingConstraint: NSLayoutConstraint = stackView.pin(.leading, to: .leading, of: self) private lazy var stackViewTrailingConstraint: NSLayoutConstraint = stackView.pin(.trailing, to: .trailing, of: self) private lazy var titleLabel: UILabel = { let result: UILabel = UILabel() result.accessibilityIdentifier = "Username" result.isAccessibilityElement = true result.font = .boldSystemFont(ofSize: Values.mediumFontSize) result.themeTextColor = .textPrimary result.lineBreakMode = .byTruncatingTail return result }() private lazy var subtitleLabel: UILabel = { let result: UILabel = UILabel() result.font = .systemFont(ofSize: 13) result.themeTextColor = .textPrimary result.lineBreakMode = .byTruncatingTail return result }() private lazy var stackView: UIStackView = { let result = UIStackView(arrangedSubviews: [ titleLabel, subtitleLabel ]) result.axis = .vertical result.alignment = .center return result }() // MARK: - Initialization init() { super.init(frame: .zero) addSubview(stackView) stackView.pin(.top, to: .top, of: self) stackViewLeadingConstraint.isActive = true stackViewTrailingConstraint.isActive = true stackView.pin(.bottom, to: .bottom, of: self) } deinit { NotificationCenter.default.removeObserver(self) } required init?(coder: NSCoder) { preconditionFailure("Use init() instead.") } // MARK: - Content public func initialSetup( with threadVariant: SessionThread.Variant, isNoteToSelf: Bool ) { self.update( with: " ", isNoteToSelf: isNoteToSelf, threadVariant: threadVariant, mutedUntilTimestamp: nil, onlyNotifyForMentions: false, userCount: (threadVariant != .contact ? 0 : nil) ) } override func layoutSubviews() { super.layoutSubviews() // There is an annoying issue where pushing seems to update the width of this // view resulting in the content shifting to the right during guard self.oldSize != .zero, self.oldSize != bounds.size else { self.oldSize = bounds.size return } let diff: CGFloat = (bounds.size.width - oldSize.width) self.stackViewTrailingConstraint.constant = -max(0, diff) self.oldSize = bounds.size } public func update( with name: String, isNoteToSelf: Bool, threadVariant: SessionThread.Variant, mutedUntilTimestamp: TimeInterval?, onlyNotifyForMentions: Bool, userCount: Int? ) { guard Thread.isMainThread else { DispatchQueue.main.async { [weak self] in self?.update( with: name, isNoteToSelf: isNoteToSelf, threadVariant: threadVariant, mutedUntilTimestamp: mutedUntilTimestamp, onlyNotifyForMentions: onlyNotifyForMentions, userCount: userCount ) } return } let shouldHaveSubtitle: Bool = ( Date().timeIntervalSince1970 <= (mutedUntilTimestamp ?? 0) || onlyNotifyForMentions || userCount != nil ) self.titleLabel.text = name self.titleLabel.accessibilityLabel = name self.titleLabel.font = .boldSystemFont( ofSize: (shouldHaveSubtitle ? Values.mediumFontSize : Values.veryLargeFontSize ) ) ThemeManager.onThemeChange(observer: self.subtitleLabel) { [weak subtitleLabel] theme, _ in guard let textPrimary: UIColor = theme.color(for: .textPrimary) else { return } guard Date().timeIntervalSince1970 > (mutedUntilTimestamp ?? 0) else { subtitleLabel?.attributedText = NSAttributedString( string: FullConversationCell.mutePrefix, attributes: [ .font: UIFont(name: "ElegantIcons", size: 10) as Any, .foregroundColor: textPrimary ] ) .appending(string: "Muted") return } guard !onlyNotifyForMentions else { let imageAttachment = NSTextAttachment() imageAttachment.image = UIImage(named: "NotifyMentions.png")?.withTint(textPrimary) imageAttachment.bounds = CGRect( x: 0, y: -2, width: Values.smallFontSize, height: Values.smallFontSize ) subtitleLabel?.attributedText = NSAttributedString(attachment: imageAttachment) .appending(string: " ") .appending(string: "view_conversation_title_notify_for_mentions_only".localized()) return } guard let userCount: Int = userCount else { return } switch threadVariant { case .contact: break case .legacyGroup, .group: subtitleLabel?.attributedText = NSAttributedString( string: "\(userCount) member\(userCount == 1 ? "" : "s")" ) case .community: subtitleLabel?.attributedText = NSAttributedString( string: "\(userCount) active member\(userCount == 1 ? "" : "s")" ) } } // Contact threads also have the call button to compensate for let shouldShowCallButton: Bool = ( SessionCall.isEnabled && !isNoteToSelf && threadVariant == .contact ) self.stackViewLeadingConstraint.constant = (shouldShowCallButton ? ConversationTitleView.leftInsetWithCallButton : ConversationTitleView.leftInset ) self.stackViewTrailingConstraint.constant = 0 } }