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.
899 lines
36 KiB
Swift
899 lines
36 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import UIKit
|
|
|
|
// FIXME: Refactor as part of the Groups Rebuild
|
|
public class ConfirmationModal: Modal, UITextFieldDelegate, UITextViewDelegate {
|
|
public static let explanationFont: UIFont = .systemFont(ofSize: Values.smallFontSize)
|
|
private static let closeSize: CGFloat = 24
|
|
|
|
public private(set) var info: Info
|
|
private var internalOnConfirm: ((ConfirmationModal) -> ())? = nil
|
|
private var internalOnCancel: ((ConfirmationModal) -> ())? = nil
|
|
private var internalOnBodyTap: ((@escaping (ValueUpdate) -> Void) -> Void)? = nil
|
|
private var internalOnTextChanged: ((String, String) -> ())? = nil
|
|
|
|
// MARK: - Components
|
|
|
|
private lazy var contentTapGestureRecognizer: UITapGestureRecognizer = {
|
|
let result: UITapGestureRecognizer = UITapGestureRecognizer(
|
|
target: self,
|
|
action: #selector(contentViewTapped)
|
|
)
|
|
contentView.addGestureRecognizer(result)
|
|
result.isEnabled = false
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var imageViewTapGestureRecognizer: UITapGestureRecognizer = {
|
|
let result: UITapGestureRecognizer = UITapGestureRecognizer(
|
|
target: self,
|
|
action: #selector(imageViewTapped)
|
|
)
|
|
imageViewContainer.addGestureRecognizer(result)
|
|
result.isEnabled = false
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var titleLabel: UILabel = {
|
|
let result: UILabel = UILabel()
|
|
result.font = .boldSystemFont(ofSize: Values.mediumFontSize)
|
|
result.themeTextColor = .alert_text
|
|
result.textAlignment = .center
|
|
result.lineBreakMode = .byWordWrapping
|
|
result.numberOfLines = 0
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var explanationLabel: ScrollableLabel = {
|
|
let result: ScrollableLabel = ScrollableLabel()
|
|
result.font = ConfirmationModal.explanationFont
|
|
result.themeTextColor = .alert_text
|
|
result.textAlignment = .center
|
|
result.lineBreakMode = .byWordWrapping
|
|
result.numberOfLines = 0
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var warningLabel: UILabel = {
|
|
let result: UILabel = UILabel()
|
|
result.font = .systemFont(ofSize: Values.verySmallFontSize)
|
|
result.themeTextColor = .warning
|
|
result.textAlignment = .center
|
|
result.lineBreakMode = .byWordWrapping
|
|
result.numberOfLines = 0
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var textFieldContainer: UIView = {
|
|
let result: UIView = UIView()
|
|
result.themeBorderColor = .borderSeparator
|
|
result.layer.cornerRadius = 11
|
|
result.layer.borderWidth = 1
|
|
result.isHidden = true
|
|
result.set(.height, to: 40)
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var textField: UITextField = {
|
|
let result: UITextField = UITextField()
|
|
result.font = .systemFont(ofSize: Values.smallFontSize)
|
|
result.themeTextColor = .textPrimary
|
|
result.delegate = self
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var textViewContainer: UIView = {
|
|
let result: UIView = UIView()
|
|
result.themeBorderColor = .borderSeparator
|
|
result.layer.cornerRadius = 11
|
|
result.layer.borderWidth = 1
|
|
result.isHidden = true
|
|
result.set(.height, to: 75)
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var textView: UITextView = {
|
|
let result: UITextView = UITextView()
|
|
result.font = .systemFont(ofSize: Values.smallFontSize)
|
|
result.themeTextColor = .textPrimary
|
|
result.themeBackgroundColor = .clear
|
|
result.textContainerInset = .zero
|
|
result.textContainer.lineFragmentPadding = 0
|
|
result.delegate = self
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var textViewPlaceholder: UILabel = {
|
|
let result: UILabel = UILabel()
|
|
result.font = .systemFont(ofSize: Values.smallFontSize)
|
|
result.themeTextColor = .textSecondary
|
|
result.alpha = 0.5
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var imageViewContainer: UIView = {
|
|
let result: UIView = UIView()
|
|
result.isHidden = true
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var profileView: ProfilePictureView = ProfilePictureView(size: .hero)
|
|
|
|
private lazy var confirmButton: UIButton = {
|
|
let result: UIButton = Modal.createButton(
|
|
title: "",
|
|
titleColor: .danger
|
|
)
|
|
result.addTarget(self, action: #selector(confirmationPressed), for: .touchUpInside)
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var buttonStackView: UIStackView = {
|
|
let result = UIStackView(arrangedSubviews: [ confirmButton, cancelButton ])
|
|
result.axis = .horizontal
|
|
result.distribution = .fillEqually
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var contentStackView: UIStackView = {
|
|
let result = UIStackView(arrangedSubviews: [ titleLabel, explanationLabel, warningLabel, textFieldContainer, textViewContainer, imageViewContainer ])
|
|
result.axis = .vertical
|
|
result.spacing = Values.smallSpacing
|
|
result.isLayoutMarginsRelativeArrangement = true
|
|
result.layoutMargins = UIEdgeInsets(
|
|
top: Values.largeSpacing,
|
|
left: Values.veryLargeSpacing,
|
|
bottom: Values.verySmallSpacing,
|
|
right: Values.veryLargeSpacing
|
|
)
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var mainStackView: UIStackView = {
|
|
let result = UIStackView(arrangedSubviews: [ contentStackView, buttonStackView ])
|
|
result.axis = .vertical
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var closeButton: UIButton = UIButton(primaryAction: UIAction { [weak self] _ in self?.close() })
|
|
.withConfiguration(
|
|
UIButton.Configuration
|
|
.plain()
|
|
.withImage(UIImage(named: "X")?.withRenderingMode(.alwaysTemplate))
|
|
.withContentInsets(NSDirectionalEdgeInsets(top: 6, leading: 6, bottom: 6, trailing: 6))
|
|
)
|
|
.withConfigurationUpdateHandler { button in
|
|
switch button.state {
|
|
case .highlighted: button.imageView?.tintAdjustmentMode = .dimmed
|
|
default: button.imageView?.tintAdjustmentMode = .normal
|
|
}
|
|
}
|
|
.withImageViewContentMode(.scaleAspectFit)
|
|
.withThemeTintColor(.textPrimary)
|
|
.withAccessibility(
|
|
identifier: "Close button",
|
|
label: "Close button"
|
|
)
|
|
.with(.width, of: ConfirmationModal.closeSize)
|
|
.with(.height, of: ConfirmationModal.closeSize)
|
|
.withHidden(true)
|
|
|
|
// MARK: - Lifecycle
|
|
|
|
public init(targetView: UIView? = nil, info: Info) {
|
|
self.info = info
|
|
|
|
super.init(targetView: targetView, dismissType: info.dismissType, afterClosed: info.afterClosed)
|
|
|
|
self.modalPresentationStyle = .overFullScreen
|
|
self.modalTransitionStyle = .crossDissolve
|
|
self.updateContent(with: info)
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
|
|
deinit {
|
|
NotificationCenter.default.removeObserver(self)
|
|
}
|
|
|
|
public override func populateContentView() {
|
|
let gestureRecogniser: UITapGestureRecognizer = UITapGestureRecognizer(
|
|
target: self,
|
|
action: #selector(contentViewTapped)
|
|
)
|
|
contentView.addGestureRecognizer(gestureRecogniser)
|
|
|
|
contentView.addSubview(mainStackView)
|
|
contentView.addSubview(closeButton)
|
|
|
|
textFieldContainer.addSubview(textField)
|
|
textField.pin(to: textFieldContainer, withInset: 12)
|
|
|
|
textViewContainer.addSubview(textView)
|
|
textViewContainer.addSubview(textViewPlaceholder)
|
|
textView.pin(to: textViewContainer, withInset: 12)
|
|
textViewPlaceholder.pin(.top, to: .top, of: textViewContainer, withInset: 12)
|
|
textViewPlaceholder.pin(.leading, to: .leading, of: textViewContainer, withInset: 12)
|
|
textViewPlaceholder.pin(.trailing, to: .trailing, of: textViewContainer, withInset: -12)
|
|
|
|
imageViewContainer.addSubview(profileView)
|
|
profileView.center(.horizontal, in: imageViewContainer)
|
|
profileView.pin(.top, to: .top, of: imageViewContainer)
|
|
profileView.pin(.bottom, to: .bottom, of: imageViewContainer)
|
|
|
|
mainStackView.pin(to: contentView)
|
|
closeButton.pin(.top, to: .top, of: contentView, withInset: 8)
|
|
closeButton.pin(.right, to: .right, of: contentView, withInset: -8)
|
|
|
|
// Observe keyboard notifications
|
|
let keyboardNotifications: [Notification.Name] = [
|
|
UIResponder.keyboardWillShowNotification,
|
|
UIResponder.keyboardDidShowNotification,
|
|
UIResponder.keyboardWillChangeFrameNotification,
|
|
UIResponder.keyboardDidChangeFrameNotification,
|
|
UIResponder.keyboardWillHideNotification,
|
|
UIResponder.keyboardDidHideNotification
|
|
]
|
|
keyboardNotifications.forEach { notification in
|
|
NotificationCenter.default.addObserver(
|
|
self,
|
|
selector: #selector(handleKeyboardNotification(_:)),
|
|
name: notification,
|
|
object: nil
|
|
)
|
|
}
|
|
}
|
|
|
|
// MARK: - Content
|
|
|
|
public func updateContent(with info: Info) {
|
|
self.info = info
|
|
internalOnBodyTap = nil
|
|
internalOnTextChanged = nil
|
|
internalOnConfirm = { modal in
|
|
if info.dismissOnConfirm {
|
|
modal.close()
|
|
}
|
|
|
|
info.onConfirm?(modal)
|
|
}
|
|
internalOnCancel = { modal in
|
|
guard info.onCancel != nil else { return modal.close() }
|
|
|
|
info.onCancel?(modal)
|
|
}
|
|
contentTapGestureRecognizer.isEnabled = true
|
|
imageViewTapGestureRecognizer.isEnabled = false
|
|
|
|
// Set the content based on the provided info
|
|
titleLabel.text = info.title
|
|
|
|
switch info.body {
|
|
case .none:
|
|
mainStackView.spacing = Values.smallSpacing
|
|
|
|
case .text(let text, let scrollMode):
|
|
mainStackView.spacing = Values.smallSpacing
|
|
explanationLabel.text = text
|
|
explanationLabel.scrollMode = scrollMode
|
|
explanationLabel.isHidden = false
|
|
|
|
case .attributedText(let attributedText, let scrollMode):
|
|
mainStackView.spacing = Values.smallSpacing
|
|
explanationLabel.attributedText = attributedText
|
|
explanationLabel.scrollMode = scrollMode
|
|
explanationLabel.isHidden = false
|
|
|
|
case .input(let explanation, let inputInfo, let onTextChanged):
|
|
explanationLabel.attributedText = explanation
|
|
explanationLabel.scrollMode = .never
|
|
explanationLabel.isHidden = (explanation == nil)
|
|
textField.placeholder = inputInfo.placeholder
|
|
textField.text = (inputInfo.initialValue ?? "")
|
|
textField.clearButtonMode = (inputInfo.clearButton ? .always : .never)
|
|
textField.isAccessibilityElement = true
|
|
textField.accessibilityIdentifier = inputInfo.accessibility?.identifier
|
|
textField.accessibilityLabel = inputInfo.accessibility?.label ?? textField.text
|
|
textFieldContainer.isHidden = false
|
|
internalOnTextChanged = { [weak textField, weak confirmButton, weak cancelButton] text, _ in
|
|
onTextChanged(text)
|
|
textField?.accessibilityLabel = text
|
|
confirmButton?.isEnabled = info.confirmEnabled.isValid(with: info)
|
|
cancelButton?.isEnabled = info.cancelEnabled.isValid(with: info)
|
|
}
|
|
|
|
case .dualInput(let explanation, let firstInputInfo, let secondInputInfo, let onTextChanged):
|
|
explanationLabel.attributedText = explanation
|
|
explanationLabel.scrollMode = .never
|
|
explanationLabel.isHidden = (explanation == nil)
|
|
textField.placeholder = firstInputInfo.placeholder
|
|
textField.text = (firstInputInfo.initialValue ?? "")
|
|
textField.clearButtonMode = (firstInputInfo.clearButton ? .always : .never)
|
|
textField.isAccessibilityElement = true
|
|
textField.accessibilityIdentifier = firstInputInfo.accessibility?.identifier
|
|
textField.accessibilityLabel = firstInputInfo.accessibility?.label
|
|
textFieldContainer.isHidden = false
|
|
textView.text = (secondInputInfo.initialValue ?? "")
|
|
textView.isAccessibilityElement = true
|
|
textView.accessibilityIdentifier = secondInputInfo.accessibility?.identifier
|
|
textView.accessibilityLabel = secondInputInfo.accessibility?.label
|
|
textViewPlaceholder.text = secondInputInfo.placeholder
|
|
textViewPlaceholder.isHidden = !textView.text.isEmpty
|
|
textViewContainer.isHidden = false
|
|
internalOnTextChanged = { [weak textField, weak textView, weak confirmButton, weak cancelButton] firstText, secondText in
|
|
onTextChanged(firstText, secondText)
|
|
textField?.accessibilityLabel = firstText
|
|
textView?.accessibilityLabel = secondText
|
|
confirmButton?.isEnabled = info.confirmEnabled.isValid(with: info)
|
|
cancelButton?.isEnabled = info.cancelEnabled.isValid(with: info)
|
|
}
|
|
|
|
case .radio(let explanation, let warning, let options):
|
|
mainStackView.spacing = 0
|
|
explanationLabel.attributedText = explanation
|
|
explanationLabel.scrollMode = .never
|
|
explanationLabel.isHidden = (explanation == nil)
|
|
warningLabel.attributedText = warning
|
|
warningLabel.isHidden = (warning == nil)
|
|
contentStackView.subviews.forEach { subview in
|
|
guard subview is RadioButton else { return }
|
|
|
|
subview.removeFromSuperview()
|
|
}
|
|
|
|
// Add the options
|
|
options.enumerated().forEach { index, optionInfo in
|
|
let radioButton: RadioButton = RadioButton(size: .medium, titleTextColor: .alert_text) { [weak self] button in
|
|
guard button.isEnabled && !button.isSelected else { return }
|
|
|
|
// If an option is selected then update the modal to show that one as selected
|
|
self?.updateContent(
|
|
with: info.with(
|
|
body: .radio(
|
|
explanation: explanation,
|
|
warning: warning,
|
|
options: options.enumerated().map { otherIndex, otherInfo in
|
|
(
|
|
otherInfo.title,
|
|
otherInfo.enabled,
|
|
(index == otherIndex),
|
|
otherInfo.accessibility
|
|
)
|
|
}
|
|
)
|
|
)
|
|
)
|
|
}
|
|
radioButton.text = optionInfo.title
|
|
radioButton.accessibilityLabel = optionInfo.accessibility?.label
|
|
radioButton.accessibilityIdentifier = optionInfo.accessibility?.identifier
|
|
radioButton.update(isEnabled: optionInfo.enabled, isSelected: optionInfo.selected)
|
|
contentStackView.addArrangedSubview(radioButton)
|
|
}
|
|
|
|
case .image(let placeholder, let value, let icon, let style, let accessibility, let onClick):
|
|
imageViewContainer.isAccessibilityElement = (accessibility != nil)
|
|
imageViewContainer.accessibilityIdentifier = accessibility?.identifier
|
|
imageViewContainer.accessibilityLabel = accessibility?.label
|
|
mainStackView.spacing = 0
|
|
imageViewContainer.isHidden = false
|
|
profileView.clipsToBounds = (style == .circular)
|
|
profileView.update(
|
|
ProfilePictureView.Info(
|
|
imageData: (value ?? placeholder),
|
|
icon: icon
|
|
)
|
|
)
|
|
internalOnBodyTap = onClick
|
|
contentTapGestureRecognizer.isEnabled = false
|
|
imageViewTapGestureRecognizer.isEnabled = true
|
|
}
|
|
|
|
confirmButton.accessibilityIdentifier = info.confirmTitle
|
|
confirmButton.isAccessibilityElement = true
|
|
confirmButton.setTitle(info.confirmTitle, for: .normal)
|
|
confirmButton.setThemeTitleColor(info.confirmStyle, for: .normal)
|
|
confirmButton.setThemeTitleColor(.disabled, for: .disabled)
|
|
confirmButton.isHidden = (info.confirmTitle == nil)
|
|
confirmButton.isEnabled = info.confirmEnabled.isValid(with: info)
|
|
|
|
cancelButton.accessibilityIdentifier = info.cancelTitle
|
|
cancelButton.isAccessibilityElement = true
|
|
cancelButton.setTitle(info.cancelTitle, for: .normal)
|
|
cancelButton.setThemeTitleColor(info.cancelStyle, for: .normal)
|
|
cancelButton.setThemeTitleColor(.disabled, for: .disabled)
|
|
cancelButton.isEnabled = info.cancelEnabled.isValid(with: info)
|
|
closeButton.isHidden = !info.hasCloseButton
|
|
|
|
titleLabel.isAccessibilityElement = true
|
|
titleLabel.accessibilityIdentifier = "Modal heading"
|
|
titleLabel.accessibilityLabel = titleLabel.text
|
|
|
|
explanationLabel.isAccessibilityElement = true
|
|
explanationLabel.accessibilityIdentifier = "Modal description"
|
|
explanationLabel.accessibilityLabel = explanationLabel.text
|
|
}
|
|
|
|
// MARK: - UITextFieldDelegate
|
|
|
|
public func textFieldShouldReturn(_ textField: UITextField) -> Bool {
|
|
textField.resignFirstResponder()
|
|
return true
|
|
}
|
|
|
|
public func textFieldShouldClear(_ textField: UITextField) -> Bool {
|
|
internalOnTextChanged?("", textView.text)
|
|
return true
|
|
}
|
|
|
|
public func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {
|
|
if let text: String = textField.text, let textRange: Range = Range(range, in: text) {
|
|
let updatedText = text.replacingCharacters(in: textRange, with: string)
|
|
|
|
internalOnTextChanged?(updatedText, textView.text)
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// MARK: - UITextViewDelegate
|
|
|
|
public func textViewDidChange(_ textView: UITextView) {
|
|
textViewPlaceholder.isHidden = !textView.text.isEmpty
|
|
internalOnTextChanged?((textField.text ?? ""), textView.text)
|
|
}
|
|
|
|
// MARK: - Interaction
|
|
|
|
@objc private func contentViewTapped() {
|
|
if textField.isFirstResponder {
|
|
textField.resignFirstResponder()
|
|
}
|
|
if textView.isFirstResponder {
|
|
textView.resignFirstResponder()
|
|
}
|
|
|
|
internalOnBodyTap?({ _ in })
|
|
}
|
|
|
|
@objc private func imageViewTapped() {
|
|
internalOnBodyTap?({ [weak self, info = self.info] valueUpdate in
|
|
switch (valueUpdate, info.body) {
|
|
case (.image(let updatedValueData), .image(let placeholderData, _, let icon, let style, let accessibility, let onClick)):
|
|
self?.updateContent(
|
|
with: info.with(
|
|
body: .image(
|
|
placeholderData: placeholderData,
|
|
valueData: updatedValueData,
|
|
icon: icon,
|
|
style: style,
|
|
accessibility: accessibility,
|
|
onClick: onClick
|
|
)
|
|
)
|
|
)
|
|
|
|
default: break
|
|
}
|
|
})
|
|
}
|
|
|
|
@objc internal func confirmationPressed() {
|
|
internalOnConfirm?(self)
|
|
}
|
|
|
|
override public func cancel() {
|
|
internalOnCancel?(self)
|
|
}
|
|
|
|
// MARK: - Keyboard Avoidance
|
|
|
|
@objc func handleKeyboardNotification(_ notification: Notification) {
|
|
guard
|
|
let userInfo: [AnyHashable: Any] = notification.userInfo,
|
|
var keyboardEndFrame: CGRect = userInfo[UIResponder.keyboardFrameEndUserInfoKey] as? CGRect
|
|
else { return }
|
|
|
|
// If reduce motion+crossfade transitions is on, in iOS 14 UIKit vends out a keyboard end frame
|
|
// of CGRect zero. This breaks the math below.
|
|
//
|
|
// If our keyboard end frame is CGRectZero, build a fake rect that's translated off the bottom edge.
|
|
if keyboardEndFrame == .zero {
|
|
keyboardEndFrame = CGRect(
|
|
x: UIScreen.main.bounds.minX,
|
|
y: UIScreen.main.bounds.maxY,
|
|
width: UIScreen.main.bounds.width,
|
|
height: 0
|
|
)
|
|
}
|
|
|
|
// Please refer to https://github.com/mapbox/mapbox-navigation-ios/issues/1600
|
|
// and https://stackoverflow.com/a/25260930 to better understand what we are
|
|
// doing with the UIViewAnimationOptions
|
|
let curveValue: Int = ((userInfo[UIResponder.keyboardAnimationCurveUserInfoKey] as? Int) ?? Int(UIView.AnimationOptions.curveEaseInOut.rawValue))
|
|
let options: UIView.AnimationOptions = UIView.AnimationOptions(rawValue: UInt(curveValue << 16))
|
|
let duration = ((userInfo[UIResponder.keyboardAnimationDurationUserInfoKey] as? TimeInterval) ?? 0)
|
|
|
|
guard duration > 0, !UIAccessibility.isReduceMotionEnabled else {
|
|
// UIKit by default (sometimes? never?) animates all changes in response to keyboard events.
|
|
// We want to suppress those animations if the view isn't visible,
|
|
// otherwise presentation animations don't work properly.
|
|
UIView.performWithoutAnimation {
|
|
self.updateKeyboardAvoidance(keyboardEndFrame: keyboardEndFrame)
|
|
}
|
|
return
|
|
}
|
|
|
|
UIView.animate(
|
|
withDuration: duration,
|
|
delay: 0,
|
|
options: options,
|
|
animations: { [weak self] in
|
|
self?.updateKeyboardAvoidance(keyboardEndFrame: keyboardEndFrame)
|
|
self?.view.layoutIfNeeded()
|
|
},
|
|
completion: nil
|
|
)
|
|
}
|
|
|
|
private func updateKeyboardAvoidance(keyboardEndFrame: CGRect) {
|
|
let contentCenteredBottom: CGFloat = (view.center.y + (contentView.bounds.height / 2))
|
|
contentTopConstraint?.isActive = (
|
|
((keyboardEndFrame.minY - contentCenteredBottom) < 10) &&
|
|
keyboardEndFrame.minY < (view.bounds.height - 100)
|
|
)
|
|
contentCenterYConstraint?.isActive = (contentTopConstraint?.isActive != true)
|
|
}
|
|
}
|
|
|
|
// MARK: - Types
|
|
|
|
public extension ConfirmationModal {
|
|
enum ValueUpdate {
|
|
case input(String)
|
|
case image(Data?)
|
|
}
|
|
|
|
struct Info: Equatable, Hashable {
|
|
internal let title: String
|
|
public let body: Body
|
|
public let showCondition: ShowCondition
|
|
internal let confirmTitle: String?
|
|
let confirmStyle: ThemeValue
|
|
let confirmEnabled: ButtonValidator
|
|
internal let cancelTitle: String
|
|
let cancelStyle: ThemeValue
|
|
let cancelEnabled: ButtonValidator
|
|
let hasCloseButton: Bool
|
|
let dismissOnConfirm: Bool
|
|
let dismissType: Modal.DismissType
|
|
let onConfirm: ((ConfirmationModal) -> ())?
|
|
let onCancel: ((ConfirmationModal) -> ())?
|
|
let afterClosed: (() -> ())?
|
|
|
|
// MARK: - Initialization
|
|
|
|
public init(
|
|
title: String,
|
|
body: Body = .none,
|
|
showCondition: ShowCondition = .none,
|
|
confirmTitle: String? = nil,
|
|
confirmStyle: ThemeValue = .alert_text,
|
|
confirmEnabled: ButtonValidator = true,
|
|
cancelTitle: String = SNUIKit.localizedString(for: "cancel"),
|
|
cancelStyle: ThemeValue = .danger,
|
|
cancelEnabled: ButtonValidator = true,
|
|
hasCloseButton: Bool = false,
|
|
dismissOnConfirm: Bool = true,
|
|
dismissType: Modal.DismissType = .recursive,
|
|
onConfirm: ((ConfirmationModal) -> ())? = nil,
|
|
onCancel: ((ConfirmationModal) -> ())? = nil,
|
|
afterClosed: (() -> ())? = nil
|
|
) {
|
|
self.title = title
|
|
self.body = body
|
|
self.showCondition = showCondition
|
|
self.confirmTitle = confirmTitle
|
|
self.confirmStyle = confirmStyle
|
|
self.confirmEnabled = confirmEnabled
|
|
self.cancelTitle = cancelTitle
|
|
self.cancelStyle = cancelStyle
|
|
self.cancelEnabled = cancelEnabled
|
|
self.hasCloseButton = hasCloseButton
|
|
self.dismissOnConfirm = dismissOnConfirm
|
|
self.dismissType = dismissType
|
|
self.onConfirm = onConfirm
|
|
self.onCancel = onCancel
|
|
self.afterClosed = afterClosed
|
|
}
|
|
|
|
// MARK: - Mutation
|
|
|
|
public func with(
|
|
body: Body? = nil,
|
|
onConfirm: ((ConfirmationModal) -> ())? = nil,
|
|
onCancel: ((ConfirmationModal) -> ())? = nil,
|
|
afterClosed: (() -> ())? = nil
|
|
) -> Info {
|
|
return Info(
|
|
title: self.title,
|
|
body: (body ?? self.body),
|
|
showCondition: self.showCondition,
|
|
confirmTitle: self.confirmTitle,
|
|
confirmStyle: self.confirmStyle,
|
|
confirmEnabled: self.confirmEnabled,
|
|
cancelTitle: self.cancelTitle,
|
|
cancelStyle: self.cancelStyle,
|
|
cancelEnabled: self.cancelEnabled,
|
|
hasCloseButton: self.hasCloseButton,
|
|
dismissOnConfirm: self.dismissOnConfirm,
|
|
dismissType: self.dismissType,
|
|
onConfirm: (onConfirm ?? self.onConfirm),
|
|
onCancel: (onCancel ?? self.onCancel),
|
|
afterClosed: (afterClosed ?? self.afterClosed)
|
|
)
|
|
}
|
|
|
|
// MARK: - Confirmance
|
|
|
|
public static func == (lhs: ConfirmationModal.Info, rhs: ConfirmationModal.Info) -> Bool {
|
|
return (
|
|
lhs.title == rhs.title &&
|
|
lhs.body == rhs.body &&
|
|
lhs.showCondition == rhs.showCondition &&
|
|
lhs.confirmTitle == rhs.confirmTitle &&
|
|
lhs.confirmStyle == rhs.confirmStyle &&
|
|
lhs.confirmEnabled.isValid(with: lhs) == rhs.confirmEnabled.isValid(with: rhs) &&
|
|
lhs.cancelTitle == rhs.cancelTitle &&
|
|
lhs.cancelStyle == rhs.cancelStyle &&
|
|
lhs.cancelEnabled.isValid(with: lhs) == rhs.cancelEnabled.isValid(with: rhs) &&
|
|
lhs.hasCloseButton == rhs.hasCloseButton &&
|
|
lhs.dismissOnConfirm == rhs.dismissOnConfirm &&
|
|
lhs.dismissType == rhs.dismissType
|
|
)
|
|
}
|
|
|
|
public func hash(into hasher: inout Hasher) {
|
|
title.hash(into: &hasher)
|
|
body.hash(into: &hasher)
|
|
showCondition.hash(into: &hasher)
|
|
confirmTitle.hash(into: &hasher)
|
|
confirmStyle.hash(into: &hasher)
|
|
confirmEnabled.isValid(with: self).hash(into: &hasher)
|
|
cancelTitle.hash(into: &hasher)
|
|
cancelStyle.hash(into: &hasher)
|
|
cancelEnabled.isValid(with: self).hash(into: &hasher)
|
|
hasCloseButton.hash(into: &hasher)
|
|
dismissOnConfirm.hash(into: &hasher)
|
|
dismissType.hash(into: &hasher)
|
|
}
|
|
}
|
|
}
|
|
|
|
public extension ConfirmationModal.Info {
|
|
// MARK: - ButtonValidator
|
|
|
|
class ButtonValidator: ExpressibleByBooleanLiteral {
|
|
public typealias BooleanLiteralType = Bool
|
|
|
|
/// Storage for the bool literal - should only ever access this via the `isValid` function which allows us to
|
|
/// override the result for other validator types
|
|
private let boolValue: Bool
|
|
|
|
required public init(booleanLiteral value: BooleanLiteralType) {
|
|
boolValue = value
|
|
}
|
|
|
|
func isValid(with info: ConfirmationModal.Info) -> Bool { boolValue }
|
|
}
|
|
|
|
/// The `AfterChangeValidator` will also return `false` for the initial validity check and will use the provided
|
|
/// value for subsequent checks
|
|
class AfterChangeValidator: ButtonValidator {
|
|
private(set) var hasDoneInitialValidCheck: Bool = false
|
|
let isValid: (ConfirmationModal.Info) -> Bool
|
|
|
|
required public init(booleanLiteral value: BooleanLiteralType) {
|
|
isValid = { _ in value }
|
|
|
|
super.init(booleanLiteral: value)
|
|
}
|
|
|
|
public init(isValid: @escaping (ConfirmationModal.Info) -> Bool) {
|
|
self.isValid = isValid
|
|
|
|
/// Default this value to false (we won't use it directly anywhere
|
|
super.init(booleanLiteral: false)
|
|
}
|
|
|
|
public override func isValid(with info: ConfirmationModal.Info) -> Bool {
|
|
guard hasDoneInitialValidCheck else {
|
|
hasDoneInitialValidCheck = true
|
|
return false
|
|
}
|
|
|
|
return self.isValid(info)
|
|
}
|
|
}
|
|
|
|
// MARK: - ShowCondition
|
|
|
|
enum ShowCondition {
|
|
case none
|
|
case enabled
|
|
case disabled
|
|
|
|
public func shouldShow(for value: Bool) -> Bool {
|
|
switch self {
|
|
case .none: return true
|
|
case .enabled: return (value == true)
|
|
case .disabled: return (value == false)
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - Body
|
|
|
|
enum Body: Equatable, Hashable {
|
|
public struct InputInfo: Equatable, Hashable {
|
|
public let placeholder: String
|
|
public let initialValue: String?
|
|
public let clearButton: Bool
|
|
public let accessibility: Accessibility?
|
|
|
|
public init(
|
|
placeholder: String,
|
|
initialValue: String? = nil,
|
|
clearButton: Bool = false,
|
|
accessibility: Accessibility? = nil
|
|
) {
|
|
self.placeholder = placeholder
|
|
self.initialValue = initialValue
|
|
self.clearButton = clearButton
|
|
self.accessibility = accessibility
|
|
}
|
|
}
|
|
public enum ImageStyle: Equatable, Hashable {
|
|
case inherit
|
|
case circular
|
|
}
|
|
|
|
case none
|
|
case text(
|
|
_ text: String,
|
|
scrollMode: ScrollableLabel.ScrollMode = .automatic
|
|
)
|
|
case attributedText(
|
|
_ attributedText: NSAttributedString,
|
|
scrollMode: ScrollableLabel.ScrollMode = .automatic
|
|
)
|
|
case input(
|
|
explanation: NSAttributedString?,
|
|
info: InputInfo,
|
|
onChange: (String) -> ()
|
|
)
|
|
case dualInput(
|
|
explanation: NSAttributedString?,
|
|
firstInfo: InputInfo,
|
|
secondInfo: InputInfo,
|
|
onChange: (String, String) -> ()
|
|
)
|
|
case radio(
|
|
explanation: NSAttributedString?,
|
|
warning: NSAttributedString?,
|
|
options: [(
|
|
title: String,
|
|
enabled: Bool,
|
|
selected: Bool,
|
|
accessibility: Accessibility?
|
|
)]
|
|
)
|
|
case image(
|
|
placeholderData: Data?,
|
|
valueData: Data?,
|
|
icon: ProfilePictureView.ProfileIcon = .none,
|
|
style: ImageStyle,
|
|
accessibility: Accessibility?,
|
|
onClick: ((@escaping (ConfirmationModal.ValueUpdate) -> Void) -> Void)
|
|
)
|
|
|
|
public static func == (lhs: ConfirmationModal.Info.Body, rhs: ConfirmationModal.Info.Body) -> Bool {
|
|
switch (lhs, rhs) {
|
|
case (.none, .none): return true
|
|
case (.text(let lhsText, _), .text(let rhsText, _)): return (lhsText == rhsText)
|
|
case (.attributedText(let lhsText, _), .attributedText(let rhsText, _)): return (lhsText == rhsText)
|
|
|
|
case (.input(let lhsExplanation, let lhsInfo, _), .input(let rhsExplanation, let rhsInfo, _)):
|
|
return (
|
|
lhsExplanation == rhsExplanation &&
|
|
lhsInfo == rhsInfo
|
|
)
|
|
|
|
case (.dualInput(let lhsExplanation, let lhsFirstInfo, let lhsSecondInfo, _), .dualInput(let rhsExplanation, let rhsFirstInfo, let rhsSecondInfo, _)):
|
|
return (
|
|
lhsExplanation == rhsExplanation &&
|
|
lhsFirstInfo == rhsFirstInfo &&
|
|
lhsSecondInfo == rhsSecondInfo
|
|
)
|
|
|
|
case (.radio(let lhsExplanation, let lhsWarning, let lhsOptions), .radio(let rhsExplanation, let rhsWarning, let rhsOptions)):
|
|
return (
|
|
lhsExplanation == rhsExplanation &&
|
|
lhsWarning == rhsWarning &&
|
|
lhsOptions.map { "\($0.0)-\($0.1)-\($0.2)" } == rhsOptions.map { "\($0.0)-\($0.1)-\($0.2)" }
|
|
)
|
|
|
|
case (.image(let lhsPlaceholder, let lhsValue, let lhsIcon, let lhsStyle, let lhsAccessibility, _), .image(let rhsPlaceholder, let rhsValue, let rhsIcon, let rhsStyle, let rhsAccessibility, _)):
|
|
return (
|
|
lhsPlaceholder == rhsPlaceholder &&
|
|
lhsValue == rhsValue &&
|
|
lhsIcon == rhsIcon &&
|
|
lhsStyle == rhsStyle &&
|
|
lhsAccessibility == rhsAccessibility
|
|
)
|
|
|
|
default: return false
|
|
}
|
|
}
|
|
|
|
public func hash(into hasher: inout Hasher) {
|
|
switch self {
|
|
case .none: break
|
|
case .text(let text, _): text.hash(into: &hasher)
|
|
case .attributedText(let text, _): text.hash(into: &hasher)
|
|
|
|
case .input(let explanation, let info, _):
|
|
explanation.hash(into: &hasher)
|
|
info.hash(into: &hasher)
|
|
|
|
case .dualInput(let explanation, let firstInfo, let secondInfo, _):
|
|
explanation.hash(into: &hasher)
|
|
firstInfo.hash(into: &hasher)
|
|
secondInfo.hash(into: &hasher)
|
|
|
|
case .radio(let explanation, let warning, let options):
|
|
explanation.hash(into: &hasher)
|
|
warning.hash(into: &hasher)
|
|
options.map { "\($0.0)-\($0.1)-\($0.2)" }.hash(into: &hasher)
|
|
|
|
case .image(let placeholder, let value, let icon, let style, let accessibility, _):
|
|
placeholder.hash(into: &hasher)
|
|
value.hash(into: &hasher)
|
|
icon.hash(into: &hasher)
|
|
style.hash(into: &hasher)
|
|
accessibility.hash(into: &hasher)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - DSL
|
|
|
|
public extension ConfirmationModal.Info.ButtonValidator {
|
|
static func bool(_ isValid: Bool) -> ConfirmationModal.Info.ButtonValidator {
|
|
return ConfirmationModal.Info.ButtonValidator(booleanLiteral: isValid)
|
|
}
|
|
|
|
static func afterChange(isValid: @escaping (ConfirmationModal.Info) -> Bool) -> ConfirmationModal.Info.AfterChangeValidator {
|
|
return ConfirmationModal.Info.AfterChangeValidator(isValid: isValid)
|
|
}
|
|
}
|