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()
|
|
}
|
|
}
|