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.
		
		
		
		
		
			
		
			
				
	
	
		
			612 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			612 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Swift
		
	
| // Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | |
| 
 | |
| import UIKit
 | |
| import Combine
 | |
| import GRDB
 | |
| import DifferenceKit
 | |
| import SessionUIKit
 | |
| import SessionUtilitiesKit
 | |
| 
 | |
| public class SessionCell: UITableViewCell {
 | |
|     public static let cornerRadius: CGFloat = 17
 | |
|     
 | |
|     private var isEditingTitle = false
 | |
|     public private(set) var interactionMode: SessionCell.TextInfo.Interaction = .none
 | |
|     private var shouldHighlightTitle: Bool = true
 | |
|     private var originalInputValue: String?
 | |
|     private var titleExtraView: UIView?
 | |
|     private var subtitleExtraView: UIView?
 | |
|     var disposables: Set<AnyCancellable> = Set()
 | |
|     
 | |
|     // MARK: - UI
 | |
|     
 | |
|     private var backgroundLeftConstraint: NSLayoutConstraint = NSLayoutConstraint()
 | |
|     private var backgroundRightConstraint: NSLayoutConstraint = NSLayoutConstraint()
 | |
|     private var topSeparatorLeftConstraint: NSLayoutConstraint = NSLayoutConstraint()
 | |
|     private var topSeparatorRightConstraint: NSLayoutConstraint = NSLayoutConstraint()
 | |
|     private var botSeparatorLeftConstraint: NSLayoutConstraint = NSLayoutConstraint()
 | |
|     private var botSeparatorRightConstraint: NSLayoutConstraint = NSLayoutConstraint()
 | |
|     private lazy var contentStackViewTopConstraint: NSLayoutConstraint = contentStackView.pin(.top, to: .top, of: cellBackgroundView)
 | |
|     private lazy var contentStackViewLeadingConstraint: NSLayoutConstraint = contentStackView.pin(.leading, to: .leading, of: cellBackgroundView)
 | |
|     private lazy var contentStackViewTrailingConstraint: NSLayoutConstraint = contentStackView.pin(.trailing, to: .trailing, of: cellBackgroundView)
 | |
|     private lazy var contentStackViewBottomConstraint: NSLayoutConstraint = contentStackView.pin(.bottom, to: .bottom, of: cellBackgroundView)
 | |
|     private lazy var contentStackViewHorizontalCenterConstraint: NSLayoutConstraint = contentStackView.center(.horizontal, in: cellBackgroundView)
 | |
|     private lazy var leftAccessoryFillConstraint: NSLayoutConstraint = contentStackView.set(.height, to: .height, of: leftAccessoryView)
 | |
|     private lazy var titleTextFieldLeadingConstraint: NSLayoutConstraint = titleTextField.pin(.leading, to: .leading, of: cellBackgroundView)
 | |
|     private lazy var titleTextFieldTrailingConstraint: NSLayoutConstraint = titleTextField.pin(.trailing, to: .trailing, of: cellBackgroundView)
 | |
|     private lazy var titleMinHeightConstraint: NSLayoutConstraint = titleStackView.heightAnchor
 | |
|         .constraint(greaterThanOrEqualTo: titleTextField.heightAnchor)
 | |
|     private lazy var rightAccessoryFillConstraint: NSLayoutConstraint = contentStackView.set(.height, to: .height, of: rightAccessoryView)
 | |
|     private lazy var accessoryWidthMatchConstraint: NSLayoutConstraint = leftAccessoryView.set(.width, to: .width, of: rightAccessoryView)
 | |
|     
 | |
|     private let cellBackgroundView: UIView = {
 | |
|         let result: UIView = UIView()
 | |
|         result.translatesAutoresizingMaskIntoConstraints = false
 | |
|         result.clipsToBounds = true
 | |
|         result.themeBackgroundColor = .settings_tabBackground
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     private let cellSelectedBackgroundView: UIView = {
 | |
|         let result: UIView = UIView()
 | |
|         result.translatesAutoresizingMaskIntoConstraints = false
 | |
|         result.themeBackgroundColor = .highlighted(.settings_tabBackground)
 | |
|         result.alpha = 0
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     private let topSeparator: UIView = {
 | |
|         let result: UIView = UIView.separator()
 | |
|         result.translatesAutoresizingMaskIntoConstraints = false
 | |
|         result.isHidden = true
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     private let contentStackView: UIStackView = {
 | |
|         let result: UIStackView = UIStackView()
 | |
|         result.translatesAutoresizingMaskIntoConstraints = false
 | |
|         result.axis = .horizontal
 | |
|         result.distribution = .fill
 | |
|         result.alignment = .center
 | |
|         result.spacing = Values.mediumSpacing
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     public let leftAccessoryView: AccessoryView = {
 | |
|         let result: AccessoryView = AccessoryView()
 | |
|         result.isHidden = true
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     private let titleStackView: UIStackView = {
 | |
|         let result: UIStackView = UIStackView()
 | |
|         result.translatesAutoresizingMaskIntoConstraints = false
 | |
|         result.axis = .vertical
 | |
|         result.distribution = .equalSpacing
 | |
|         result.alignment = .fill
 | |
|         result.setContentHugging(to: .defaultLow)
 | |
|         result.setCompressionResistance(to: .defaultLow)
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     fileprivate let titleLabel: SRCopyableLabel = {
 | |
|         let result: SRCopyableLabel = SRCopyableLabel()
 | |
|         result.translatesAutoresizingMaskIntoConstraints = false
 | |
|         result.isUserInteractionEnabled = false
 | |
|         result.themeTextColor = .textPrimary
 | |
|         result.numberOfLines = 0
 | |
|         result.setContentHugging(to: .defaultLow)
 | |
|         result.setCompressionResistance(to: .defaultLow)
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     fileprivate let titleTextField: UITextField = {
 | |
|         let textField: TextField = TextField(placeholder: "", usesDefaultHeight: false)
 | |
|         textField.translatesAutoresizingMaskIntoConstraints = false
 | |
|         textField.textAlignment = .center
 | |
|         textField.alpha = 0
 | |
|         textField.isHidden = true
 | |
|         textField.set(.height, to: Values.largeButtonHeight)
 | |
|         
 | |
|         return textField
 | |
|     }()
 | |
|     
 | |
|     private let subtitleLabel: SRCopyableLabel = {
 | |
|         let result: SRCopyableLabel = SRCopyableLabel()
 | |
|         result.translatesAutoresizingMaskIntoConstraints = false
 | |
|         result.isUserInteractionEnabled = false
 | |
|         result.font = .systemFont(ofSize: 12)
 | |
|         result.themeTextColor = .textPrimary
 | |
|         result.numberOfLines = 0
 | |
|         result.isHidden = true
 | |
|         result.setContentHugging(to: .defaultLow)
 | |
|         result.setCompressionResistance(to: .defaultLow)
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     public let rightAccessoryView: AccessoryView = {
 | |
|         let result: AccessoryView = AccessoryView()
 | |
|         result.isHidden = true
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     private let botSeparator: UIView = {
 | |
|         let result: UIView = UIView.separator()
 | |
|         result.translatesAutoresizingMaskIntoConstraints = false
 | |
|         result.isHidden = true
 | |
|         
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     // MARK: - Initialization
 | |
|     
 | |
|     override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
 | |
|         super.init(style: style, reuseIdentifier: reuseIdentifier)
 | |
|         
 | |
|         setupViewHierarchy()
 | |
|     }
 | |
| 
 | |
|     required init?(coder: NSCoder) {
 | |
|         super.init(coder: coder)
 | |
|         
 | |
|         setupViewHierarchy()
 | |
|     }
 | |
| 
 | |
|     private func setupViewHierarchy() {
 | |
|         self.themeBackgroundColor = .clear
 | |
|         self.selectedBackgroundView = UIView()
 | |
|         
 | |
|         contentView.addSubview(cellBackgroundView)
 | |
|         cellBackgroundView.addSubview(cellSelectedBackgroundView)
 | |
|         cellBackgroundView.addSubview(topSeparator)
 | |
|         cellBackgroundView.addSubview(contentStackView)
 | |
|         cellBackgroundView.addSubview(botSeparator)
 | |
|         
 | |
|         contentStackView.addArrangedSubview(leftAccessoryView)
 | |
|         contentStackView.addArrangedSubview(titleStackView)
 | |
|         contentStackView.addArrangedSubview(rightAccessoryView)
 | |
|         
 | |
|         titleStackView.addArrangedSubview(titleLabel)
 | |
|         titleStackView.addArrangedSubview(subtitleLabel)
 | |
|         
 | |
|         cellBackgroundView.addSubview(titleTextField)
 | |
|         
 | |
|         setupLayout()
 | |
|     }
 | |
|     
 | |
|     private func setupLayout() {
 | |
|         cellBackgroundView.pin(.top, to: .top, of: contentView)
 | |
|         backgroundLeftConstraint = cellBackgroundView.pin(.leading, to: .leading, of: contentView)
 | |
|         backgroundRightConstraint = cellBackgroundView.pin(.trailing, to: .trailing, of: contentView)
 | |
|         cellBackgroundView.pin(.bottom, to: .bottom, of: contentView)
 | |
|         
 | |
|         cellSelectedBackgroundView.pin(to: cellBackgroundView)
 | |
|         
 | |
|         topSeparator.pin(.top, to: .top, of: cellBackgroundView)
 | |
|         topSeparatorLeftConstraint = topSeparator.pin(.left, to: .left, of: cellBackgroundView)
 | |
|         topSeparatorRightConstraint = topSeparator.pin(.right, to: .right, of: cellBackgroundView)
 | |
|         
 | |
|         contentStackViewTopConstraint.isActive = true
 | |
|         contentStackViewBottomConstraint.isActive = true
 | |
|         
 | |
|         titleTextField.center(.vertical, in: titleLabel)
 | |
|         
 | |
|         botSeparatorLeftConstraint = botSeparator.pin(.left, to: .left, of: cellBackgroundView)
 | |
|         botSeparatorRightConstraint = botSeparator.pin(.right, to: .right, of: cellBackgroundView)
 | |
|         botSeparator.pin(.bottom, to: .bottom, of: cellBackgroundView)
 | |
|         
 | |
|         // Explicitly call this to ensure we have initialised the constraints before we initially
 | |
|         // layout (if we don't do this then some constraints get created for the first time when
 | |
|         // updating the cell before the `isActive` value gets set, resulting in breaking constriants)
 | |
|         prepareForReuse()
 | |
|     }
 | |
|     
 | |
|     public override func layoutSubviews() {
 | |
|         super.layoutSubviews()
 | |
|         
 | |
|         // Need to force the contentStackView to layout if needed as it might not have updated it's
 | |
|         // sizing yet
 | |
|         self.contentStackView.layoutIfNeeded()
 | |
|         repositionExtraView(titleExtraView, for: titleLabel)
 | |
|         repositionExtraView(subtitleExtraView, for: subtitleLabel)
 | |
|     }
 | |
|     
 | |
|     private func repositionExtraView(_ targetView: UIView?, for label: UILabel) {
 | |
|         guard
 | |
|             let targetView: UIView = targetView,
 | |
|             let content: String = label.text,
 | |
|             let font: UIFont = label.font
 | |
|         else { return }
 | |
|         
 | |
|         // Position the 'targetView' at the end of the last line of text
 | |
|         let layoutManager: NSLayoutManager = NSLayoutManager()
 | |
|         let textStorage = NSTextStorage(
 | |
|             attributedString: NSAttributedString(
 | |
|                 string: content,
 | |
|                 attributes: [ .font: font ]
 | |
|             )
 | |
|         )
 | |
|         textStorage.addLayoutManager(layoutManager)
 | |
|         
 | |
|         let textContainer: NSTextContainer = NSTextContainer(
 | |
|             size: CGSize(
 | |
|                 width: label.bounds.size.width,
 | |
|                 height: 999
 | |
|             )
 | |
|         )
 | |
|         textContainer.lineFragmentPadding = 0
 | |
|         layoutManager.addTextContainer(textContainer)
 | |
|         
 | |
|         var glyphRange: NSRange = NSRange()
 | |
|         layoutManager.characterRange(
 | |
|             forGlyphRange: NSRange(location: content.glyphCount - 1, length: 1),
 | |
|             actualGlyphRange: &glyphRange
 | |
|         )
 | |
|         let lastGlyphRect: CGRect = layoutManager.boundingRect(forGlyphRange: glyphRange, in: textContainer)
 | |
|         
 | |
|         // Remove and re-add the 'subtitleExtraView' to clear any old constraints
 | |
|         targetView.removeFromSuperview()
 | |
|         contentView.addSubview(targetView)
 | |
|         
 | |
|         targetView.pin(
 | |
|             .top,
 | |
|             to: .top,
 | |
|             of: label,
 | |
|             withInset: (lastGlyphRect.minY + ((lastGlyphRect.height / 2) - (targetView.bounds.height / 2)))
 | |
|         )
 | |
|         targetView.pin(
 | |
|             .leading,
 | |
|             to: .leading,
 | |
|             of: label,
 | |
|             withInset: lastGlyphRect.maxX
 | |
|         )
 | |
|     }
 | |
|     
 | |
|     // MARK: - Content
 | |
|     
 | |
|     public override func prepareForReuse() {
 | |
|         super.prepareForReuse()
 | |
|         
 | |
|         isEditingTitle = false
 | |
|         interactionMode = .none
 | |
|         shouldHighlightTitle = true
 | |
|         accessibilityIdentifier = nil
 | |
|         accessibilityLabel = nil
 | |
|         isAccessibilityElement = false
 | |
|         originalInputValue = nil
 | |
|         titleExtraView?.removeFromSuperview()
 | |
|         titleExtraView = nil
 | |
|         subtitleExtraView?.removeFromSuperview()
 | |
|         subtitleExtraView = nil
 | |
|         disposables = Set()
 | |
|         
 | |
|         contentStackView.spacing = Values.mediumSpacing
 | |
|         contentStackViewLeadingConstraint.isActive = false
 | |
|         contentStackViewTrailingConstraint.isActive = false
 | |
|         contentStackViewHorizontalCenterConstraint.isActive = false
 | |
|         titleMinHeightConstraint.isActive = false
 | |
|         leftAccessoryView.prepareForReuse()
 | |
|         leftAccessoryView.alpha = 1
 | |
|         leftAccessoryFillConstraint.isActive = false
 | |
|         titleLabel.text = ""
 | |
|         titleLabel.themeTextColor = .textPrimary
 | |
|         titleLabel.alpha = 1
 | |
|         titleTextField.text = ""
 | |
|         titleTextField.textAlignment = .center
 | |
|         titleTextField.themeTextColor = .textPrimary
 | |
|         titleTextField.isHidden = true
 | |
|         titleTextField.alpha = 0
 | |
|         subtitleLabel.isUserInteractionEnabled = false
 | |
|         subtitleLabel.text = ""
 | |
|         subtitleLabel.themeTextColor = .textPrimary
 | |
|         rightAccessoryView.prepareForReuse()
 | |
|         rightAccessoryView.alpha = 1
 | |
|         rightAccessoryFillConstraint.isActive = false
 | |
|         accessoryWidthMatchConstraint.isActive = false
 | |
|         
 | |
|         topSeparator.isHidden = true
 | |
|         subtitleLabel.isHidden = true
 | |
|         botSeparator.isHidden = true
 | |
|     }
 | |
|     
 | |
|     public func update<ID: Hashable & Differentiable>(with info: Info<ID>, isManualReload: Bool = false) {
 | |
|         interactionMode = (info.title?.interaction ?? .none)
 | |
|         shouldHighlightTitle = (info.title?.interaction != .copy)
 | |
|         titleExtraView = info.title?.extraViewGenerator?()
 | |
|         subtitleExtraView = info.subtitle?.extraViewGenerator?()
 | |
|         accessibilityIdentifier = info.accessibility?.identifier
 | |
|         accessibilityLabel = info.accessibility?.label
 | |
|         isAccessibilityElement = true
 | |
|         originalInputValue = info.title?.text
 | |
|         
 | |
|         // Convenience Flags
 | |
|         let leftFitToEdge: Bool = (info.leftAccessory?.shouldFitToEdge == true)
 | |
|         let rightFitToEdge: Bool = (!leftFitToEdge && info.rightAccessory?.shouldFitToEdge == true)
 | |
|         
 | |
|         // Content
 | |
|         contentStackView.spacing = (info.styling.customPadding?.interItem ?? Values.mediumSpacing)
 | |
|         leftAccessoryView.update(
 | |
|             with: info.leftAccessory,
 | |
|             tintColor: info.styling.tintColor,
 | |
|             isEnabled: info.isEnabled,
 | |
|             isManualReload: isManualReload
 | |
|         )
 | |
|         titleStackView.isHidden = (info.title == nil && info.subtitle == nil)
 | |
|         titleLabel.isUserInteractionEnabled = (info.title?.interaction == .copy)
 | |
|         titleLabel.font = info.title?.font
 | |
|         titleLabel.text = info.title?.text
 | |
|         titleLabel.themeTextColor = info.styling.tintColor
 | |
|         titleLabel.textAlignment = (info.title?.textAlignment ?? .left)
 | |
|         titleLabel.isHidden = (info.title == nil)
 | |
|         titleTextField.text = info.title?.text
 | |
|         titleTextField.textAlignment = (info.title?.textAlignment ?? .left)
 | |
|         titleTextField.placeholder = info.title?.editingPlaceholder
 | |
|         titleTextField.isHidden = (info.title == nil)
 | |
|         titleTextField.accessibilityIdentifier = info.accessibility?.identifier
 | |
|         titleTextField.accessibilityLabel = info.accessibility?.label
 | |
|         subtitleLabel.isUserInteractionEnabled = (info.subtitle?.interaction == .copy)
 | |
|         subtitleLabel.font = info.subtitle?.font
 | |
|         subtitleLabel.text = info.subtitle?.text
 | |
|         subtitleLabel.themeTextColor = info.styling.tintColor
 | |
|         subtitleLabel.textAlignment = (info.subtitle?.textAlignment ?? .left)
 | |
|         subtitleLabel.isHidden = (info.subtitle == nil)
 | |
|         rightAccessoryView.update(
 | |
|             with: info.rightAccessory,
 | |
|             tintColor: info.styling.tintColor,
 | |
|             isEnabled: info.isEnabled,
 | |
|             isManualReload: isManualReload
 | |
|         )
 | |
|         
 | |
|         contentStackViewLeadingConstraint.isActive = (info.styling.alignment == .leading)
 | |
|         contentStackViewTrailingConstraint.isActive = (info.styling.alignment == .leading)
 | |
|         contentStackViewHorizontalCenterConstraint.constant = ((info.styling.customPadding?.leading ?? 0) + (info.styling.customPadding?.trailing ?? 0))
 | |
|         contentStackViewHorizontalCenterConstraint.isActive = (info.styling.alignment == .centerHugging)
 | |
|         leftAccessoryFillConstraint.isActive = leftFitToEdge
 | |
|         rightAccessoryFillConstraint.isActive = rightFitToEdge
 | |
|         accessoryWidthMatchConstraint.isActive = {
 | |
|             switch (info.leftAccessory, info.rightAccessory) {
 | |
|                 case (.button, .button): return true
 | |
|                 default: return false
 | |
|             }
 | |
|         }()
 | |
|         titleLabel.setContentHuggingPriority(
 | |
|             (info.rightAccessory != nil ? .defaultLow : .required),
 | |
|             for: .horizontal
 | |
|         )
 | |
|         titleLabel.setContentCompressionResistancePriority(
 | |
|             (info.rightAccessory != nil ? .defaultLow : .required),
 | |
|             for: .horizontal
 | |
|         )
 | |
|         contentStackViewTopConstraint.constant = {
 | |
|             if let customPadding: CGFloat = info.styling.customPadding?.top {
 | |
|                 return customPadding
 | |
|             }
 | |
|             
 | |
|             return (leftFitToEdge || rightFitToEdge ? 0 : Values.mediumSpacing)
 | |
|         }()
 | |
|         contentStackViewLeadingConstraint.constant = {
 | |
|             if let customPadding: CGFloat = info.styling.customPadding?.leading {
 | |
|                 return customPadding
 | |
|             }
 | |
|             
 | |
|             return (leftFitToEdge ? 0 : Values.mediumSpacing)
 | |
|         }()
 | |
|         contentStackViewTrailingConstraint.constant = {
 | |
|             if let customPadding: CGFloat = info.styling.customPadding?.trailing {
 | |
|                 return -customPadding
 | |
|             }
 | |
|             
 | |
|             return -(rightFitToEdge ? 0 : Values.mediumSpacing)
 | |
|         }()
 | |
|         contentStackViewBottomConstraint.constant = {
 | |
|             if let customPadding: CGFloat = info.styling.customPadding?.bottom {
 | |
|                 return -customPadding
 | |
|             }
 | |
|             
 | |
|             return -(leftFitToEdge || rightFitToEdge ? 0 : Values.mediumSpacing)
 | |
|         }()
 | |
|         titleTextFieldLeadingConstraint.constant = {
 | |
|             guard info.styling.backgroundStyle != .noBackground else { return 0 }
 | |
|             
 | |
|             return (leftFitToEdge ? 0 : Values.mediumSpacing)
 | |
|         }()
 | |
|         titleTextFieldTrailingConstraint.constant = {
 | |
|             guard info.styling.backgroundStyle != .noBackground else { return 0 }
 | |
|             
 | |
|             return -(rightFitToEdge ? 0 : Values.mediumSpacing)
 | |
|         }()
 | |
|         
 | |
|         // Styling and positioning
 | |
|         let defaultEdgePadding: CGFloat
 | |
|         
 | |
|         switch info.styling.backgroundStyle {
 | |
|             case .rounded:
 | |
|                 cellBackgroundView.themeBackgroundColor = .settings_tabBackground
 | |
|                 cellSelectedBackgroundView.isHidden = !info.isEnabled
 | |
|                 
 | |
|                 defaultEdgePadding = Values.mediumSpacing
 | |
|                 backgroundLeftConstraint.constant = Values.largeSpacing
 | |
|                 backgroundRightConstraint.constant = -Values.largeSpacing
 | |
|                 cellBackgroundView.layer.cornerRadius = SessionCell.cornerRadius
 | |
|                 
 | |
|             case .edgeToEdge:
 | |
|                 cellBackgroundView.themeBackgroundColor = .settings_tabBackground
 | |
|                 cellSelectedBackgroundView.isHidden = !info.isEnabled
 | |
|                 
 | |
|                 defaultEdgePadding = 0
 | |
|                 backgroundLeftConstraint.constant = 0
 | |
|                 backgroundRightConstraint.constant = 0
 | |
|                 cellBackgroundView.layer.cornerRadius = 0
 | |
|                 
 | |
|             case .noBackground:
 | |
|                 defaultEdgePadding = Values.mediumSpacing
 | |
|                 backgroundLeftConstraint.constant = Values.largeSpacing
 | |
|                 backgroundRightConstraint.constant = -Values.largeSpacing
 | |
|                 cellBackgroundView.themeBackgroundColor = nil
 | |
|                 cellBackgroundView.layer.cornerRadius = 0
 | |
|                 cellSelectedBackgroundView.isHidden = true
 | |
|         }
 | |
|         
 | |
|         let fittedEdgePadding: CGFloat = {
 | |
|             func targetSize(accessory: Accessory?) -> CGFloat {
 | |
|                 switch accessory {
 | |
|                     case .icon(_, let iconSize, _, _, _), .iconAsync(let iconSize, _, _, _, _):
 | |
|                         return iconSize.size
 | |
|                         
 | |
|                     default: return defaultEdgePadding
 | |
|                 }
 | |
|             }
 | |
|             
 | |
|             guard leftFitToEdge else {
 | |
|                 guard rightFitToEdge else { return defaultEdgePadding }
 | |
|                 
 | |
|                 return targetSize(accessory: info.rightAccessory)
 | |
|             }
 | |
|             
 | |
|             return targetSize(accessory: info.leftAccessory)
 | |
|         }()
 | |
|         topSeparatorLeftConstraint.constant = (leftFitToEdge ? fittedEdgePadding : defaultEdgePadding)
 | |
|         topSeparatorRightConstraint.constant = (rightFitToEdge ? -fittedEdgePadding : -defaultEdgePadding)
 | |
|         botSeparatorLeftConstraint.constant = (leftFitToEdge ? fittedEdgePadding : defaultEdgePadding)
 | |
|         botSeparatorRightConstraint.constant = (rightFitToEdge ? -fittedEdgePadding : -defaultEdgePadding)
 | |
|         
 | |
|         switch info.position {
 | |
|             case .top:
 | |
|                 cellBackgroundView.layer.maskedCorners = [.layerMinXMinYCorner, .layerMaxXMinYCorner]
 | |
|                 topSeparator.isHidden = (
 | |
|                     !info.styling.allowedSeparators.contains(.top) ||
 | |
|                     info.styling.backgroundStyle != .edgeToEdge
 | |
|                 )
 | |
|                 botSeparator.isHidden = (
 | |
|                     !info.styling.allowedSeparators.contains(.bottom) ||
 | |
|                     info.styling.backgroundStyle == .noBackground
 | |
|                 )
 | |
|                 
 | |
|             case .middle:
 | |
|                 cellBackgroundView.layer.maskedCorners = []
 | |
|                 topSeparator.isHidden = true
 | |
|                 botSeparator.isHidden = (
 | |
|                     !info.styling.allowedSeparators.contains(.bottom) ||
 | |
|                     info.styling.backgroundStyle == .noBackground
 | |
|                 )
 | |
|                 
 | |
|             case .bottom:
 | |
|                 cellBackgroundView.layer.maskedCorners = [.layerMinXMaxYCorner, .layerMaxXMaxYCorner]
 | |
|                 topSeparator.isHidden = true
 | |
|                 botSeparator.isHidden = (
 | |
|                     !info.styling.allowedSeparators.contains(.bottom) ||
 | |
|                     info.styling.backgroundStyle != .edgeToEdge
 | |
|                 )
 | |
|                 
 | |
|             case .individual:
 | |
|                 cellBackgroundView.layer.maskedCorners = [
 | |
|                     .layerMinXMinYCorner, .layerMaxXMinYCorner,
 | |
|                     .layerMinXMaxYCorner, .layerMaxXMaxYCorner
 | |
|                 ]
 | |
|                 topSeparator.isHidden = (
 | |
|                     !info.styling.allowedSeparators.contains(.top) ||
 | |
|                     info.styling.backgroundStyle != .edgeToEdge
 | |
|                 )
 | |
|                 botSeparator.isHidden = (
 | |
|                     !info.styling.allowedSeparators.contains(.bottom) ||
 | |
|                     info.styling.backgroundStyle != .edgeToEdge
 | |
|                 )
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     public func update(isEditing: Bool, becomeFirstResponder: Bool, animated: Bool) {
 | |
|         // Note: We set 'isUserInteractionEnabled' based on the 'info.isEditable' flag
 | |
|         // so can use that to determine whether this element can become editable
 | |
|         guard interactionMode == .editable || interactionMode == .alwaysEditing else { return }
 | |
|         
 | |
|         self.isEditingTitle = isEditing
 | |
|         
 | |
|         let changes = { [weak self] in
 | |
|             self?.titleLabel.alpha = (isEditing ? 0 : 1)
 | |
|             self?.titleTextField.alpha = (isEditing ? 1 : 0)
 | |
|             self?.leftAccessoryView.alpha = (isEditing ? 0 : 1)
 | |
|             self?.rightAccessoryView.alpha = (isEditing ? 0 : 1)
 | |
|             self?.titleMinHeightConstraint.isActive = isEditing
 | |
|         }
 | |
|         let completion: (Bool) -> Void = { [weak self] complete in
 | |
|             self?.titleTextField.text = self?.originalInputValue
 | |
|         }
 | |
| 
 | |
|         if animated {
 | |
|             UIView.animate(withDuration: 0.25, animations: changes, completion: completion)
 | |
|         }
 | |
|         else {
 | |
|             changes()
 | |
|             completion(true)
 | |
|         }
 | |
| 
 | |
|         if isEditing && becomeFirstResponder {
 | |
|             titleTextField.becomeFirstResponder()
 | |
|         }
 | |
|         else if !isEditing {
 | |
|             titleTextField.resignFirstResponder()
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     // MARK: - Interaction
 | |
|     
 | |
|     public override func setHighlighted(_ highlighted: Bool, animated: Bool) {
 | |
|         super.setHighlighted(highlighted, animated: animated)
 | |
|         
 | |
|         // When editing disable the highlighted state changes (would result in UI elements
 | |
|         // reappearing otherwise)
 | |
|         guard !self.isEditingTitle else { return }
 | |
|         
 | |
|         // If the 'cellSelectedBackgroundView' is hidden then there is no background so we
 | |
|         // should update the titleLabel to indicate the highlighted state
 | |
|         if cellSelectedBackgroundView.isHidden && shouldHighlightTitle {
 | |
|             // Note: We delay the "unhighlight" of the titleLabel so that the transition doesn't
 | |
|             // conflict with the transition into edit mode
 | |
|             DispatchQueue.main.asyncAfter(deadline: .now() + .milliseconds(10)) { [weak self] in
 | |
|                 guard self?.isEditingTitle == false else { return }
 | |
|                 
 | |
|                 self?.titleLabel.alpha = (highlighted ? 0.8 : 1)
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         cellSelectedBackgroundView.alpha = (highlighted ? 1 : 0)
 | |
|         leftAccessoryView.setHighlighted(highlighted, animated: animated)
 | |
|         rightAccessoryView.setHighlighted(highlighted, animated: animated)
 | |
|     }
 | |
|     
 | |
|     public override func setSelected(_ selected: Bool, animated: Bool) {
 | |
|         super.setSelected(selected, animated: animated)
 | |
| 
 | |
|         leftAccessoryView.setSelected(selected, animated: animated)
 | |
|         rightAccessoryView.setSelected(selected, animated: animated)
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: - Compose
 | |
| 
 | |
| extension CombineCompatible where Self: SessionCell {
 | |
|     var textPublisher: AnyPublisher<String, Never> {
 | |
|         return self.titleTextField.publisher(for: [.editingChanged, .editingDidEnd])
 | |
|             .handleEvents(
 | |
|                 receiveOutput: { [weak self] textField in
 | |
|                     // When editing the text update the 'accessibilityLabel' of the cell to match
 | |
|                     // the text
 | |
|                     let targetText: String? = (textField.isEditing ? textField.text : self?.titleLabel.text)
 | |
|                     self?.accessibilityLabel = (targetText ?? self?.accessibilityLabel)
 | |
|                 }
 | |
|             )
 | |
|             .filter { $0.isEditing }    // Don't bother sending events for 'editingDidEnd'
 | |
|             .map { textField -> String in (textField.text ?? "") }
 | |
|             .eraseToAnyPublisher()
 | |
|     }
 | |
| }
 |