session-ios/SignalMessaging/ViewControllers/ContactShareApprovalViewController.swift

514 lines
16 KiB
Swift
Raw Normal View History

2018-05-03 16:47:42 +02:00
//
2019-01-16 16:04:33 +01:00
// Copyright (c) 2019 Open Whisper Systems. All rights reserved.
2018-05-03 16:47:42 +02:00
//
import Foundation
import SignalServiceKit
@objc
public protocol ContactShareApprovalViewControllerDelegate: class {
func approveContactShare(_ approveContactShare: ContactShareApprovalViewController,
2018-05-10 17:09:35 +02:00
didApproveContactShare contactShare: ContactShareViewModel)
func approveContactShare(_ approveContactShare: ContactShareApprovalViewController,
2018-05-10 17:09:35 +02:00
didCancelContactShare contactShare: ContactShareViewModel)
2018-05-03 16:47:42 +02:00
}
2018-05-07 15:47:22 +02:00
protocol ContactShareField: class {
var isAvatar: Bool { get }
2018-05-07 15:47:22 +02:00
func localizedLabel() -> String
func isIncluded() -> Bool
func setIsIncluded(_ isIncluded: Bool)
2018-05-07 18:32:31 +02:00
func applyToContact(contact: ContactShareViewModel)
2018-05-07 15:47:22 +02:00
}
2018-05-03 16:47:42 +02:00
// MARK: -
2018-05-07 15:47:22 +02:00
class ContactShareFieldBase<ContactFieldType: OWSContactField>: NSObject, ContactShareField {
let value: ContactFieldType
2018-05-03 16:47:42 +02:00
2018-05-03 20:07:49 +02:00
private var isIncludedFlag = true
2018-05-03 16:47:42 +02:00
var isAvatar: Bool { return false }
2018-05-07 15:47:22 +02:00
required init(_ value: ContactFieldType) {
self.value = value
super.init()
}
2018-05-03 16:47:42 +02:00
func localizedLabel() -> String {
2018-05-07 15:47:22 +02:00
return value.localizedLabel()
2018-05-03 16:47:42 +02:00
}
func isIncluded() -> Bool {
return isIncludedFlag
}
2018-05-03 20:07:49 +02:00
func setIsIncluded(_ isIncluded: Bool) {
2018-05-03 16:47:42 +02:00
isIncludedFlag = isIncluded
}
2018-05-03 20:07:49 +02:00
2018-05-05 04:32:29 +02:00
func applyToContact(contact: ContactShareViewModel) {
2018-05-03 20:07:49 +02:00
preconditionFailure("This method must be overridden")
}
2018-05-03 16:47:42 +02:00
}
// MARK: -
2018-05-07 15:47:22 +02:00
class ContactSharePhoneNumber: ContactShareFieldBase<OWSContactPhoneNumber> {
2018-05-03 20:07:49 +02:00
2018-05-05 04:32:29 +02:00
override func applyToContact(contact: ContactShareViewModel) {
2018-05-03 20:07:49 +02:00
assert(isIncluded())
var values = [OWSContactPhoneNumber]()
2018-05-04 19:41:24 +02:00
values += contact.phoneNumbers
2018-05-03 20:07:49 +02:00
values.append(value)
contact.phoneNumbers = values
}
2018-05-03 16:47:42 +02:00
}
// MARK: -
2018-05-07 15:47:22 +02:00
class ContactShareEmail: ContactShareFieldBase<OWSContactEmail> {
2018-05-03 20:07:49 +02:00
2018-05-05 04:32:29 +02:00
override func applyToContact(contact: ContactShareViewModel) {
2018-05-03 20:07:49 +02:00
assert(isIncluded())
var values = [OWSContactEmail]()
2018-05-04 19:41:24 +02:00
values += contact.emails
2018-05-03 20:07:49 +02:00
values.append(value)
contact.emails = values
}
2018-05-03 16:47:42 +02:00
}
// MARK: -
2018-05-07 15:47:22 +02:00
class ContactShareAddress: ContactShareFieldBase<OWSContactAddress> {
2018-05-03 20:07:49 +02:00
2018-05-05 04:32:29 +02:00
override func applyToContact(contact: ContactShareViewModel) {
2018-05-03 20:07:49 +02:00
assert(isIncluded())
var values = [OWSContactAddress]()
2018-05-04 19:41:24 +02:00
values += contact.addresses
2018-05-03 20:07:49 +02:00
values.append(value)
contact.addresses = values
}
2018-05-03 16:47:42 +02:00
}
// Stub class so that avatars conform to OWSContactField.
class OWSContactAvatar: NSObject, OWSContactField {
public let avatarImage: UIImage
public let avatarData: Data
required init(avatarImage: UIImage, avatarData: Data) {
self.avatarImage = avatarImage
self.avatarData = avatarData
super.init()
}
public func ows_isValid() -> Bool {
return true
}
public func localizedLabel() -> String {
return ""
}
override public var debugDescription: String {
return "Avatar"
}
}
class ContactShareAvatarField: ContactShareFieldBase<OWSContactAvatar> {
override var isAvatar: Bool { return true }
override func applyToContact(contact: ContactShareViewModel) {
assert(isIncluded())
contact.avatarImageData = value.avatarData
}
}
2018-05-03 16:47:42 +02:00
// MARK: -
2018-05-04 17:40:23 +02:00
protocol ContactShareFieldViewDelegate: class {
func contactShareFieldViewDidChangeSelectedState()
}
// MARK: -
class ContactShareFieldView: UIStackView {
2018-05-03 16:47:42 +02:00
2018-05-04 17:40:23 +02:00
weak var delegate: ContactShareFieldViewDelegate?
2018-05-03 16:47:42 +02:00
let field: ContactShareField
let previewViewBlock : (() -> UIView)
private var checkbox: UIButton!
// MARK: - Initializers
@available(*, unavailable, message: "use init(call:) constructor instead.")
required init(coder aDecoder: NSCoder) {
2018-08-27 16:21:03 +02:00
notImplemented()
2018-05-03 16:47:42 +02:00
}
2018-05-04 17:40:23 +02:00
required init(field: ContactShareField, previewViewBlock : @escaping (() -> UIView), delegate: ContactShareFieldViewDelegate) {
2018-05-03 16:47:42 +02:00
self.field = field
self.previewViewBlock = previewViewBlock
2018-05-04 17:40:23 +02:00
self.delegate = delegate
2018-05-03 16:47:42 +02:00
super.init(frame: CGRect.zero)
self.isUserInteractionEnabled = true
self.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(wasTapped)))
createContents()
}
let hSpacing = CGFloat(10)
let hMargin = CGFloat(16)
2018-05-03 16:47:42 +02:00
func createContents() {
self.axis = .horizontal
self.spacing = hSpacing
self.alignment = .center
self.layoutMargins = UIEdgeInsets(top: 0, left: hMargin, bottom: 0, right: hMargin)
self.isLayoutMarginsRelativeArrangement = true
2018-05-03 16:47:42 +02:00
let checkbox = UIButton(type: .custom)
self.checkbox = checkbox
2018-05-09 20:56:11 +02:00
let checkedIcon = #imageLiteral(resourceName: "contact_checkbox_checked")
let uncheckedIcon = #imageLiteral(resourceName: "contact_checkbox_unchecked")
checkbox.setImage(uncheckedIcon, for: .normal)
checkbox.setImage(checkedIcon, for: .selected)
2018-05-03 16:47:42 +02:00
checkbox.isSelected = field.isIncluded()
// Disable the checkbox; the entire row is hot.
checkbox.isUserInteractionEnabled = false
self.addArrangedSubview(checkbox)
2018-05-03 16:47:42 +02:00
checkbox.setCompressionResistanceHigh()
checkbox.setContentHuggingHigh()
let previewView = previewViewBlock()
self.addArrangedSubview(previewView)
2018-05-03 16:47:42 +02:00
}
2018-05-25 18:54:25 +02:00
@objc func wasTapped(sender: UIGestureRecognizer) {
2018-08-23 16:37:34 +02:00
Logger.info("")
2018-05-03 16:47:42 +02:00
guard sender.state == .recognized else {
return
}
2018-05-03 20:07:49 +02:00
field.setIsIncluded(!field.isIncluded())
checkbox.isSelected = field.isIncluded()
2018-05-04 17:40:23 +02:00
2018-05-07 23:05:44 +02:00
delegate?.contactShareFieldViewDidChangeSelectedState()
2018-05-03 16:47:42 +02:00
}
}
// MARK: -
2018-05-04 19:06:26 +02:00
// TODO: Rename to ContactShareApprovalViewController
2018-05-03 16:47:42 +02:00
@objc
public class ContactShareApprovalViewController: OWSViewController, EditContactShareNameViewControllerDelegate, ContactShareFieldViewDelegate {
2018-05-04 17:40:23 +02:00
weak var delegate: ContactShareApprovalViewControllerDelegate?
2018-05-03 16:47:42 +02:00
2018-05-03 20:07:49 +02:00
let contactsManager: OWSContactsManager
2018-05-03 16:47:42 +02:00
2018-05-05 04:32:29 +02:00
var contactShare: ContactShareViewModel
2018-05-03 16:47:42 +02:00
var fieldViews = [ContactShareFieldView]()
2018-05-03 20:07:49 +02:00
var nameLabel: UILabel!
2018-05-03 16:47:42 +02:00
// MARK: Initializers
@available(*, unavailable, message:"use other constructor instead.")
required public init?(coder aDecoder: NSCoder) {
2018-08-27 16:21:03 +02:00
notImplemented()
2018-05-03 16:47:42 +02:00
}
@objc
required public init(contactShare: ContactShareViewModel, contactsManager: OWSContactsManager, delegate: ContactShareApprovalViewControllerDelegate) {
2018-05-03 16:47:42 +02:00
self.contactsManager = contactsManager
self.contactShare = contactShare
self.delegate = delegate
super.init(nibName: nil, bundle: nil)
buildFields()
}
func buildFields() {
var fieldViews = [ContactShareFieldView]()
let previewInsets = UIEdgeInsets(top: 5, left: 0, bottom: 5, right: 0)
if let avatarData = contactShare.avatarImageData {
if let avatarImage = contactShare.avatarImage {
let field = ContactShareAvatarField(OWSContactAvatar(avatarImage: avatarImage, avatarData: avatarData))
let fieldView = ContactShareFieldView(field: field, previewViewBlock: {
return ContactFieldView.contactFieldView(forAvatarImage: avatarImage, layoutMargins: previewInsets, actionBlock: nil)
},
delegate: self)
fieldViews.append(fieldView)
} else {
2018-08-27 16:27:48 +02:00
owsFailDebug("could not load avatar image.")
}
}
2018-05-04 19:41:24 +02:00
for phoneNumber in contactShare.phoneNumbers {
let field = ContactSharePhoneNumber(phoneNumber)
let fieldView = ContactShareFieldView(field: field, previewViewBlock: {
return ContactFieldView.contactFieldView(forPhoneNumber: phoneNumber, layoutMargins: previewInsets, actionBlock: nil)
2018-05-04 17:40:23 +02:00
},
delegate: self)
2018-05-04 19:41:24 +02:00
fieldViews.append(fieldView)
2018-05-03 16:47:42 +02:00
}
2018-05-04 17:40:23 +02:00
2018-05-04 19:41:24 +02:00
for email in contactShare.emails {
let field = ContactShareEmail(email)
let fieldView = ContactShareFieldView(field: field, previewViewBlock: {
return ContactFieldView.contactFieldView(forEmail: email, layoutMargins: previewInsets, actionBlock: nil)
2018-05-04 17:40:23 +02:00
},
delegate: self)
2018-05-04 19:41:24 +02:00
fieldViews.append(fieldView)
2018-05-03 16:47:42 +02:00
}
2018-05-04 17:40:23 +02:00
2018-05-04 19:41:24 +02:00
for address in contactShare.addresses {
let field = ContactShareAddress(address)
let fieldView = ContactShareFieldView(field: field, previewViewBlock: {
return ContactFieldView.contactFieldView(forAddress: address, layoutMargins: previewInsets, actionBlock: nil)
2018-05-04 17:40:23 +02:00
},
delegate: self)
2018-05-04 19:41:24 +02:00
fieldViews.append(fieldView)
2018-05-03 16:47:42 +02:00
}
self.fieldViews = fieldViews
}
// MARK: - View Lifecycle
override public func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
updateNavigationBar()
}
override public func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
}
override public func viewWillDisappear(_ animated: Bool) {
super.viewWillDisappear(animated)
}
override public func viewDidDisappear(_ animated: Bool) {
super.viewDidDisappear(animated)
}
override public func loadView() {
super.loadView()
self.navigationItem.title = NSLocalizedString("CONTACT_SHARE_APPROVAL_VIEW_TITLE",
comment: "Title for the 'Approve contact share' view.")
2018-07-23 21:51:12 +02:00
self.view.backgroundColor = Theme.backgroundColor
2018-05-03 16:47:42 +02:00
updateContent()
updateNavigationBar()
}
2018-05-04 17:40:23 +02:00
func isAtLeastOneFieldSelected() -> Bool {
for fieldView in fieldViews {
if fieldView.field.isIncluded(), !fieldView.field.isAvatar {
2018-05-04 17:40:23 +02:00
return true
}
}
return false
2018-05-03 16:47:42 +02:00
}
func updateNavigationBar() {
2018-05-03 17:33:01 +02:00
self.navigationItem.leftBarButtonItem = UIBarButtonItem(barButtonSystemItem: .cancel,
target: self,
action: #selector(didPressCancel))
2018-05-09 20:34:24 +02:00
self.navigationItem.rightBarButtonItem = UIBarButtonItem(title: NSLocalizedString("ATTACHMENT_APPROVAL_SEND_BUTTON",
comment: "Label for 'send' button in the 'attachment approval' dialog."),
style: .plain, target: self, action: #selector(didPressSendButton))
2018-05-03 17:33:01 +02:00
}
2018-05-03 16:47:42 +02:00
private func updateContent() {
AssertIsOnMainThread()
2018-05-03 16:47:42 +02:00
guard let rootView = self.view else {
2018-08-27 16:27:48 +02:00
owsFailDebug("missing root view.")
2018-05-03 16:47:42 +02:00
return
}
for subview in rootView.subviews {
subview.removeFromSuperview()
}
let scrollView = UIScrollView()
scrollView.preservesSuperviewLayoutMargins = false
self.view.addSubview(scrollView)
scrollView.layoutMargins = .zero
2019-01-16 16:04:33 +01:00
scrollView.autoPinEdge(toSuperviewSafeArea: .leading)
scrollView.autoPinEdge(toSuperviewSafeArea: .trailing)
2018-05-03 16:47:42 +02:00
scrollView.autoPin(toTopLayoutGuideOf: self, withInset: 0)
scrollView.autoPinEdge(toSuperviewEdge: .bottom)
let fieldsView = createFieldsView()
scrollView.addSubview(fieldsView)
2018-05-07 15:47:22 +02:00
// Use layoutMarginsGuide for views inside UIScrollView
// that should have same width as scroll view.
2018-05-03 16:47:42 +02:00
fieldsView.autoPinLeadingToSuperviewMargin()
fieldsView.autoPinTrailingToSuperviewMargin()
fieldsView.autoPinEdge(toSuperviewEdge: .top)
fieldsView.autoPinEdge(toSuperviewEdge: .bottom)
2018-05-07 15:47:22 +02:00
fieldsView.setContentHuggingHorizontalLow()
2018-05-03 16:47:42 +02:00
}
private func createFieldsView() -> UIView {
AssertIsOnMainThread()
2018-05-03 16:47:42 +02:00
2018-05-04 19:57:29 +02:00
var rows = [UIView]()
2018-05-03 16:47:42 +02:00
2018-05-04 19:57:29 +02:00
rows.append(createNameRow())
2018-05-03 17:33:01 +02:00
2018-05-03 16:47:42 +02:00
for fieldView in fieldViews {
2018-05-04 19:57:29 +02:00
rows.append(fieldView)
2018-05-03 16:47:42 +02:00
}
2018-05-04 19:57:29 +02:00
return ContactFieldView(rows: rows, hMargin: hMargin)
2018-05-03 16:47:42 +02:00
}
private let hMargin = CGFloat(16)
2018-05-03 17:33:01 +02:00
func createNameRow() -> UIView {
let nameVMargin = CGFloat(16)
2018-05-03 16:47:42 +02:00
2018-05-25 20:59:36 +02:00
let stackView = TappableStackView(actionBlock: { [weak self] in
guard let strongSelf = self else { return }
strongSelf.didPressEditName()
})
2018-05-03 16:47:42 +02:00
2018-05-03 17:33:01 +02:00
stackView.axis = .horizontal
stackView.alignment = .center
stackView.layoutMargins = UIEdgeInsets(top: nameVMargin, left: hMargin, bottom: nameVMargin, right: hMargin)
2018-05-03 20:07:49 +02:00
stackView.spacing = 10
stackView.isLayoutMarginsRelativeArrangement = true
2018-05-03 16:47:42 +02:00
2018-05-03 17:33:01 +02:00
let nameLabel = UILabel()
2018-05-03 20:07:49 +02:00
self.nameLabel = nameLabel
2018-05-10 17:09:35 +02:00
nameLabel.text = contactShare.name.displayName
nameLabel.font = UIFont.ows_dynamicTypeBody.ows_mediumWeight()
2018-07-23 21:51:12 +02:00
nameLabel.textColor = Theme.primaryColor
2018-05-03 17:33:01 +02:00
nameLabel.lineBreakMode = .byTruncatingTail
stackView.addArrangedSubview(nameLabel)
let editNameLabel = UILabel()
editNameLabel.text = NSLocalizedString("CONTACT_EDIT_NAME_BUTTON", comment: "Label for the 'edit name' button in the contact share approval view.")
editNameLabel.font = UIFont.ows_dynamicTypeBody
editNameLabel.textColor = UIColor.ows_materialBlue
2018-05-03 17:33:01 +02:00
stackView.addArrangedSubview(editNameLabel)
editNameLabel.setContentHuggingHigh()
editNameLabel.setCompressionResistanceHigh()
2018-05-04 19:48:58 +02:00
return stackView
2018-05-03 16:47:42 +02:00
}
// MARK: -
2018-05-05 04:32:29 +02:00
func filteredContactShare() -> ContactShareViewModel {
2018-05-10 17:09:35 +02:00
let result = self.contactShare.newContact(withName: self.contactShare.name)
2018-05-03 20:07:49 +02:00
for fieldView in fieldViews {
if fieldView.field.isIncluded() {
fieldView.field.applyToContact(contact: result)
}
}
return result
}
// MARK: -
2018-05-25 18:54:25 +02:00
@objc func didPressSendButton() {
AssertIsOnMainThread()
2018-05-09 20:34:24 +02:00
guard isAtLeastOneFieldSelected() else {
OWSAlerts.showErrorAlert(message: NSLocalizedString("CONTACT_SHARE_NO_FIELDS_SELECTED",
comment: "Error indicating that at least one contact field must be selected before sharing a contact."))
return
}
guard contactShare.ows_isValid else {
OWSAlerts.showErrorAlert(message: NSLocalizedString("CONTACT_SHARE_INVALID_CONTACT",
comment: "Error indicating that an invalid contact cannot be shared."))
return
}
2018-05-04 17:40:23 +02:00
2018-08-23 16:37:34 +02:00
Logger.info("")
2018-05-03 16:47:42 +02:00
2018-05-03 17:33:01 +02:00
guard let delegate = self.delegate else {
2018-08-27 16:27:48 +02:00
owsFailDebug("missing delegate.")
2018-05-03 17:33:01 +02:00
return
}
2018-05-03 16:47:42 +02:00
2018-05-03 20:07:49 +02:00
let filteredContactShare = self.filteredContactShare()
2018-05-05 04:32:29 +02:00
assert(filteredContactShare.ows_isValid)
2018-05-03 20:07:49 +02:00
delegate.approveContactShare(self, didApproveContactShare: filteredContactShare)
2018-05-03 17:33:01 +02:00
}
2018-05-25 18:54:25 +02:00
@objc func didPressCancel() {
2018-08-23 16:37:34 +02:00
Logger.info("")
2018-05-03 17:33:01 +02:00
guard let delegate = self.delegate else {
2018-08-27 16:27:48 +02:00
owsFailDebug("missing delegate.")
2018-05-03 17:33:01 +02:00
return
}
delegate.approveContactShare(self, didCancelContactShare: contactShare)
}
func didPressEditName() {
2018-08-23 16:37:34 +02:00
Logger.info("")
2018-05-03 16:47:42 +02:00
2018-05-03 20:07:49 +02:00
let view = EditContactShareNameViewController(contactShare: contactShare, delegate: self)
self.navigationController?.pushViewController(view, animated: true)
}
// MARK: - EditContactShareNameViewControllerDelegate
2018-05-10 17:09:35 +02:00
public func editContactShareNameView(_ editContactShareNameView: EditContactShareNameViewController,
didEditContactShare contactShare: ContactShareViewModel) {
2018-05-03 20:07:49 +02:00
self.contactShare = contactShare
2018-05-10 17:09:35 +02:00
nameLabel.text = contactShare.name.displayName
2018-05-03 20:07:49 +02:00
self.updateNavigationBar()
2018-05-03 16:47:42 +02:00
}
2018-05-04 17:40:23 +02:00
// MARK: - ContactShareFieldViewDelegate
public func contactShareFieldViewDidChangeSelectedState() {
self.updateNavigationBar()
}
2018-05-03 16:47:42 +02:00
}