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.
		
		
		
		
		
			
		
			
				
	
	
		
			183 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			183 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Swift
		
	
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | 
						|
 | 
						|
import UIKit
 | 
						|
import SessionUIKit
 | 
						|
import SessionMessagingKit
 | 
						|
 | 
						|
final class InfoMessageCell: MessageCell {
 | 
						|
    private static let iconSize: CGFloat = 12
 | 
						|
    public static let inset = Values.mediumSpacing
 | 
						|
    
 | 
						|
    private var isHandlingLongPress: Bool = false
 | 
						|
    
 | 
						|
    override var contextSnapshotView: UIView? { return label }
 | 
						|
    
 | 
						|
    // MARK: - UI
 | 
						|
    
 | 
						|
    private lazy var iconContainerViewWidthConstraint = iconContainerView.set(.width, to: InfoMessageCell.iconSize)
 | 
						|
    private lazy var iconContainerViewHeightConstraint = iconContainerView.set(.height, to: InfoMessageCell.iconSize)
 | 
						|
    
 | 
						|
    private lazy var iconImageView: UIImageView = UIImageView()
 | 
						|
    private lazy var timerView: DisappearingMessageTimerView = DisappearingMessageTimerView()
 | 
						|
    
 | 
						|
    private lazy var iconContainerView: UIView = {
 | 
						|
        let result: UIView = UIView()
 | 
						|
        result.addSubview(iconImageView)
 | 
						|
        result.addSubview(timerView)
 | 
						|
        iconImageView.pin(to: result)
 | 
						|
        timerView.pin(to: result)
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
 | 
						|
    private lazy var label: UILabel = {
 | 
						|
        let result: UILabel = UILabel()
 | 
						|
        result.font = .systemFont(ofSize: Values.verySmallFontSize)
 | 
						|
        result.themeTextColor = .textSecondary
 | 
						|
        result.textAlignment = .center
 | 
						|
        result.lineBreakMode = .byWordWrapping
 | 
						|
        result.numberOfLines = 0
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var actionLabel: UILabel = {
 | 
						|
        let result: UILabel = UILabel()
 | 
						|
        result.font = .systemFont(ofSize: Values.verySmallFontSize)
 | 
						|
        result.themeTextColor = .primary
 | 
						|
        result.textAlignment = .center
 | 
						|
        result.numberOfLines = 1
 | 
						|
        result.isAccessibilityElement = true
 | 
						|
        result.accessibilityIdentifier = "Follow setting"
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
 | 
						|
    private lazy var stackView: UIStackView = {
 | 
						|
        let result: UIStackView = UIStackView(arrangedSubviews: [ iconContainerView, label, actionLabel ])
 | 
						|
        result.axis = .vertical
 | 
						|
        result.alignment = .center
 | 
						|
        result.spacing = Values.smallSpacing
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
 | 
						|
    // MARK: - Lifecycle
 | 
						|
    
 | 
						|
    override func setUpViewHierarchy() {
 | 
						|
        super.setUpViewHierarchy()
 | 
						|
        
 | 
						|
        iconContainerViewWidthConstraint.isActive = true
 | 
						|
        iconContainerViewHeightConstraint.isActive = true
 | 
						|
        addSubview(stackView)
 | 
						|
        
 | 
						|
        stackView.pin(.left, to: .left, of: self, withInset: Values.massiveSpacing)
 | 
						|
        stackView.pin(.top, to: .top, of: self, withInset: InfoMessageCell.inset)
 | 
						|
        stackView.pin(.right, to: .right, of: self, withInset: -Values.massiveSpacing)
 | 
						|
        stackView.pin(.bottom, to: .bottom, of: self, withInset: -InfoMessageCell.inset)
 | 
						|
    }
 | 
						|
    
 | 
						|
    override func setUpGestureRecognizers() {
 | 
						|
        let longPressRecognizer = UILongPressGestureRecognizer(target: self, action: #selector(handleLongPress))
 | 
						|
        addGestureRecognizer(longPressRecognizer)
 | 
						|
        
 | 
						|
        let tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(handleTap))
 | 
						|
        tapGestureRecognizer.numberOfTapsRequired = 1
 | 
						|
        addGestureRecognizer(tapGestureRecognizer)
 | 
						|
    }
 | 
						|
 | 
						|
    // MARK: - Updating
 | 
						|
    
 | 
						|
    override func update(
 | 
						|
        with cellViewModel: MessageViewModel,
 | 
						|
        mediaCache: NSCache<NSString, AnyObject>,
 | 
						|
        playbackInfo: ConversationViewModel.PlaybackInfo?,
 | 
						|
        showExpandedReactions: Bool,
 | 
						|
        lastSearchText: String?
 | 
						|
    ) {
 | 
						|
        guard cellViewModel.variant.isInfoMessage else { return }
 | 
						|
        
 | 
						|
        self.accessibilityIdentifier = "Control message"
 | 
						|
        self.isAccessibilityElement = true
 | 
						|
        self.viewModel = cellViewModel
 | 
						|
        
 | 
						|
        self.actionLabel.isHidden = true
 | 
						|
        self.actionLabel.text = nil
 | 
						|
        
 | 
						|
        let icon: UIImage? = {
 | 
						|
            switch cellViewModel.variant {
 | 
						|
                case .infoDisappearingMessagesUpdate:
 | 
						|
                    return UIImage(systemName: "timer")
 | 
						|
                    
 | 
						|
                case .infoMediaSavedNotification: return UIImage(named: "ic_download")
 | 
						|
                    
 | 
						|
                default: return nil
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        
 | 
						|
        if let icon = icon {
 | 
						|
            iconImageView.image = icon.withRenderingMode(.alwaysTemplate)
 | 
						|
            iconImageView.themeTintColor = .textSecondary
 | 
						|
        }
 | 
						|
        
 | 
						|
        self.label.attributedText = cellViewModel.body?.formatted(in: self.label)
 | 
						|
        
 | 
						|
        if cellViewModel.canDoFollowingSetting() {
 | 
						|
            self.actionLabel.isHidden = false
 | 
						|
            self.actionLabel.text = "FOLLOW_SETTING_TITLE".localized()
 | 
						|
        }
 | 
						|
        
 | 
						|
        self.label.themeTextColor = (cellViewModel.variant == .infoClosedGroupCurrentUserErrorLeaving) ? .danger : .textSecondary
 | 
						|
 | 
						|
        let shouldShowIcon: Bool = (icon != nil) || ((cellViewModel.expiresInSeconds ?? 0) > 0)
 | 
						|
        
 | 
						|
        iconContainerViewWidthConstraint.constant = shouldShowIcon ? InfoMessageCell.iconSize : 0
 | 
						|
        iconContainerViewHeightConstraint.constant = shouldShowIcon ? InfoMessageCell.iconSize : 0
 | 
						|
        
 | 
						|
        guard shouldShowIcon else { return }
 | 
						|
        // Timer
 | 
						|
        if
 | 
						|
            let expiresStartedAtMs: Double = cellViewModel.expiresStartedAtMs,
 | 
						|
            let expiresInSeconds: TimeInterval = cellViewModel.expiresInSeconds
 | 
						|
        {
 | 
						|
            let expirationTimestampMs: Double = (expiresStartedAtMs + (expiresInSeconds * 1000))
 | 
						|
            
 | 
						|
            timerView.configure(
 | 
						|
                expirationTimestampMs: expirationTimestampMs,
 | 
						|
                initialDurationSeconds: expiresInSeconds
 | 
						|
            )
 | 
						|
            timerView.themeTintColor = .textSecondary
 | 
						|
            timerView.isHidden = false
 | 
						|
            iconImageView.isHidden = true
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            timerView.isHidden = true
 | 
						|
            iconImageView.isHidden = false
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    override func dynamicUpdate(with cellViewModel: MessageViewModel, playbackInfo: ConversationViewModel.PlaybackInfo?) {
 | 
						|
    }
 | 
						|
    
 | 
						|
    // MARK: - Interaction
 | 
						|
    
 | 
						|
    @objc func handleLongPress(_ gestureRecognizer: UILongPressGestureRecognizer) {
 | 
						|
        if [ .ended, .cancelled, .failed ].contains(gestureRecognizer.state) {
 | 
						|
            isHandlingLongPress = false
 | 
						|
            return
 | 
						|
        }
 | 
						|
        guard !isHandlingLongPress, let cellViewModel: MessageViewModel = self.viewModel else { return }
 | 
						|
        
 | 
						|
        delegate?.handleItemLongPressed(cellViewModel)
 | 
						|
        isHandlingLongPress = true
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc func handleTap(_ gestureRecognizer: UITapGestureRecognizer) {
 | 
						|
        guard let cellViewModel: MessageViewModel = self.viewModel else { return }
 | 
						|
        
 | 
						|
        if cellViewModel.variant == .infoDisappearingMessagesUpdate && cellViewModel.canDoFollowingSetting() {
 | 
						|
            delegate?.handleItemTapped(cellViewModel, cell: self, cellLocation: gestureRecognizer.location(in: self))
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |