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.
		
		
		
		
		
			
		
			
				
	
	
		
			196 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			196 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Swift
		
	
// 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) {
 | 
						|
        self.update(
 | 
						|
            with: " ",
 | 
						|
            isNoteToSelf: false,
 | 
						|
            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: "\u{e067}  ",
 | 
						|
                    attributes: [
 | 
						|
                        .font: UIFont.ows_elegantIconsFont(10),
 | 
						|
                        .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 .closedGroup:
 | 
						|
                    subtitleLabel?.attributedText = NSAttributedString(
 | 
						|
                        string: "\(userCount) member\(userCount == 1 ? "" : "s")"
 | 
						|
                    )
 | 
						|
                    
 | 
						|
                case .openGroup:
 | 
						|
                    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
 | 
						|
    }
 | 
						|
}
 |