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.
403 lines
14 KiB
Swift
403 lines
14 KiB
Swift
//
|
|
// Copyright (c) 2018 Open Whisper Systems. All rights reserved.
|
|
//
|
|
|
|
import Foundation
|
|
|
|
@objc
|
|
public class MenuAction: NSObject {
|
|
let block: (MenuAction) -> Void
|
|
let image: UIImage
|
|
let title: String
|
|
let subtitle: String?
|
|
|
|
public init(image: UIImage, title: String, subtitle: String?, block: @escaping (MenuAction) -> Void) {
|
|
self.image = image
|
|
self.title = title
|
|
self.subtitle = subtitle
|
|
self.block = block
|
|
}
|
|
}
|
|
|
|
@objc
|
|
protocol MenuActionsViewControllerDelegate: class {
|
|
func menuActionsDidHide(_ menuActionsViewController: MenuActionsViewController)
|
|
func menuActions(_ menuActionsViewController: MenuActionsViewController, isPresentingWithVerticalFocusChange: CGFloat)
|
|
func menuActions(_ menuActionsViewController: MenuActionsViewController, isDismissingWithVerticalFocusChange: CGFloat)
|
|
}
|
|
|
|
@objc
|
|
class MenuActionsViewController: UIViewController, MenuActionSheetDelegate {
|
|
|
|
@objc
|
|
weak var delegate: MenuActionsViewControllerDelegate?
|
|
|
|
private let focusedView: UIView
|
|
private let actionSheetView: MenuActionSheetView
|
|
|
|
deinit {
|
|
Logger.verbose("\(logTag) in \(#function)")
|
|
assert(didInformDelegateOfDismissalAnimation)
|
|
assert(didInformDelegateThatDisappearenceCompleted)
|
|
}
|
|
|
|
@objc
|
|
required init(focusedView: UIView, actions: [MenuAction]) {
|
|
self.focusedView = focusedView
|
|
|
|
self.actionSheetView = MenuActionSheetView(actions: actions)
|
|
super.init(nibName: nil, bundle: nil)
|
|
|
|
actionSheetView.delegate = self
|
|
}
|
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
|
|
// MARK: View LifeCycle
|
|
|
|
var actionSheetViewVerticalConstraint: NSLayoutConstraint?
|
|
|
|
override func loadView() {
|
|
self.view = UIView()
|
|
|
|
view.addSubview(actionSheetView)
|
|
|
|
actionSheetView.autoPinWidthToSuperview()
|
|
actionSheetView.setContentHuggingVerticalHigh()
|
|
actionSheetView.setCompressionResistanceHigh()
|
|
self.actionSheetViewVerticalConstraint = actionSheetView.autoPinEdge(.top, to: .bottom, of: self.view)
|
|
|
|
let tapGesture = UITapGestureRecognizer(target: self, action: #selector(didTapBackground))
|
|
self.view.addGestureRecognizer(tapGesture)
|
|
|
|
let swipeGesture = UISwipeGestureRecognizer(target: self, action: #selector(didSwipeBackground))
|
|
swipeGesture.direction = .down
|
|
self.view.addGestureRecognizer(swipeGesture)
|
|
}
|
|
|
|
override func viewDidAppear(_ animated: Bool) {
|
|
super.viewDidAppear(true)
|
|
|
|
self.animatePresentation()
|
|
}
|
|
|
|
override func viewDidDisappear(_ animated: Bool) {
|
|
Logger.debug("\(logTag) in \(#function)")
|
|
super.viewDidDisappear(animated)
|
|
|
|
// When the user has manually dismissed the menu, we do a nice animation
|
|
// but if the view otherwise disappears (e.g. due to resigning active),
|
|
// we still want to give the delegate the information it needs to restore it's UI.
|
|
ensureDelegateIsInformedOfDismissalAnimation()
|
|
ensureDelegateIsInformedThatDisappearenceCompleted()
|
|
}
|
|
|
|
// MARK: Present / Dismiss animations
|
|
|
|
var presentationFocusOffset: CGFloat?
|
|
var snapshotView: UIView?
|
|
|
|
private func addSnapshotFocusedView() -> UIView? {
|
|
guard let snapshotView = self.focusedView.snapshotView(afterScreenUpdates: false) else {
|
|
owsFail("\(self.logTag) in \(#function) snapshotView was unexpectedly nil")
|
|
return nil
|
|
}
|
|
view.addSubview(snapshotView)
|
|
|
|
guard let focusedViewSuperview = focusedView.superview else {
|
|
owsFail("\(self.logTag) in \(#function) focusedViewSuperview was unexpectedly nil")
|
|
return nil
|
|
}
|
|
|
|
let convertedFrame = view.convert(focusedView.frame, from: focusedViewSuperview)
|
|
snapshotView.frame = convertedFrame
|
|
|
|
return snapshotView
|
|
}
|
|
|
|
private func animatePresentation() {
|
|
guard let actionSheetViewVerticalConstraint = self.actionSheetViewVerticalConstraint else {
|
|
owsFail("\(self.logTag) in \(#function) actionSheetViewVerticalConstraint was unexpectedly nil")
|
|
return
|
|
}
|
|
|
|
guard let focusedViewSuperview = focusedView.superview else {
|
|
owsFail("\(self.logTag) in \(#function) focusedViewSuperview was unexpectedly nil")
|
|
return
|
|
}
|
|
|
|
// darken background
|
|
guard let snapshotView = addSnapshotFocusedView() else {
|
|
owsFail("\(self.logTag) in \(#function) snapshotView was unexpectedly nil")
|
|
return
|
|
}
|
|
|
|
self.snapshotView = snapshotView
|
|
snapshotView.superview?.layoutIfNeeded()
|
|
|
|
let backgroundDuration: TimeInterval = 0.1
|
|
UIView.animate(withDuration: backgroundDuration) {
|
|
self.view.backgroundColor = UIColor.black.withAlphaComponent(0.4)
|
|
}
|
|
|
|
self.actionSheetView.superview?.layoutIfNeeded()
|
|
|
|
let oldFocusFrame = self.view.convert(focusedView.frame, from: focusedViewSuperview)
|
|
NSLayoutConstraint.deactivate([actionSheetViewVerticalConstraint])
|
|
self.actionSheetViewVerticalConstraint = self.actionSheetView.autoPinEdge(toSuperviewEdge: .bottom)
|
|
UIView.animate(withDuration: 0.2,
|
|
delay: backgroundDuration,
|
|
options: .curveEaseOut,
|
|
animations: {
|
|
self.actionSheetView.superview?.layoutIfNeeded()
|
|
let newSheetFrame = self.actionSheetView.frame
|
|
|
|
var newFocusFrame = oldFocusFrame
|
|
|
|
// Position focused item just over the action sheet.
|
|
let padding: CGFloat = 10
|
|
let overlap: CGFloat = (oldFocusFrame.maxY + padding) - newSheetFrame.minY
|
|
newFocusFrame.origin.y = oldFocusFrame.origin.y - overlap
|
|
|
|
snapshotView.frame = newFocusFrame
|
|
|
|
let offset = -overlap
|
|
self.presentationFocusOffset = offset
|
|
self.delegate?.menuActions(self, isPresentingWithVerticalFocusChange: offset)
|
|
},
|
|
completion: nil)
|
|
}
|
|
|
|
private func animateDismiss(action: MenuAction?) {
|
|
guard let actionSheetViewVerticalConstraint = self.actionSheetViewVerticalConstraint else {
|
|
owsFail("\(self.logTag) in \(#function) actionSheetVerticalConstraint was unexpectedly nil")
|
|
self.delegate?.menuActionsDidHide(self)
|
|
return
|
|
}
|
|
|
|
guard let snapshotView = self.snapshotView else {
|
|
owsFail("\(self.logTag) in \(#function) snapshotView was unexpectedly nil")
|
|
self.delegate?.menuActionsDidHide(self)
|
|
return
|
|
}
|
|
|
|
guard let presentationFocusOffset = self.presentationFocusOffset else {
|
|
owsFail("\(self.logTag) in \(#function) presentationFocusOffset was unexpectedly nil")
|
|
self.delegate?.menuActionsDidHide(self)
|
|
return
|
|
}
|
|
|
|
self.actionSheetView.superview?.layoutIfNeeded()
|
|
NSLayoutConstraint.deactivate([actionSheetViewVerticalConstraint])
|
|
|
|
let dismissDuration: TimeInterval = 0.2
|
|
self.actionSheetViewVerticalConstraint = self.actionSheetView.autoPinEdge(.top, to: .bottom, of: self.view)
|
|
UIView.animate(withDuration: dismissDuration,
|
|
delay: 0,
|
|
options: .curveEaseOut,
|
|
animations: {
|
|
self.view.backgroundColor = UIColor.clear
|
|
self.actionSheetView.superview?.layoutIfNeeded()
|
|
snapshotView.frame.origin.y -= presentationFocusOffset
|
|
// this helps when focused view is above navbars, etc.
|
|
snapshotView.alpha = 0
|
|
self.ensureDelegateIsInformedOfDismissalAnimation()
|
|
},
|
|
completion: { _ in
|
|
self.view.isHidden = true
|
|
self.ensureDelegateIsInformedThatDisappearenceCompleted()
|
|
if let action = action {
|
|
action.block(action)
|
|
}
|
|
})
|
|
}
|
|
|
|
var didInformDelegateThatDisappearenceCompleted = false
|
|
func ensureDelegateIsInformedThatDisappearenceCompleted() {
|
|
guard !didInformDelegateThatDisappearenceCompleted else {
|
|
Logger.debug("\(logTag) in \(#function) ignoring redundant 'disappeared' notification")
|
|
return
|
|
}
|
|
didInformDelegateThatDisappearenceCompleted = true
|
|
|
|
self.delegate?.menuActionsDidHide(self)
|
|
}
|
|
|
|
var didInformDelegateOfDismissalAnimation = false
|
|
func ensureDelegateIsInformedOfDismissalAnimation() {
|
|
guard !didInformDelegateOfDismissalAnimation else {
|
|
Logger.debug("\(logTag) in \(#function) ignoring redundant 'dismissal' notification")
|
|
return
|
|
}
|
|
didInformDelegateOfDismissalAnimation = true
|
|
|
|
guard let presentationFocusOffset = self.presentationFocusOffset else {
|
|
owsFail("\(self.logTag) in \(#function) presentationFocusOffset was unexpectedly nil")
|
|
self.delegate?.menuActionsDidHide(self)
|
|
return
|
|
}
|
|
|
|
self.delegate?.menuActions(self, isDismissingWithVerticalFocusChange: presentationFocusOffset)
|
|
}
|
|
|
|
// MARK: Actions
|
|
|
|
@objc
|
|
func didTapBackground() {
|
|
animateDismiss(action: nil)
|
|
}
|
|
|
|
@objc
|
|
func didSwipeBackground(gesture: UISwipeGestureRecognizer) {
|
|
animateDismiss(action: nil)
|
|
}
|
|
|
|
// MARK: MenuActionSheetDelegate
|
|
|
|
func actionSheet(_ actionSheet: MenuActionSheetView, didSelectAction action: MenuAction) {
|
|
animateDismiss(action: action)
|
|
}
|
|
}
|
|
|
|
protocol MenuActionSheetDelegate: class {
|
|
func actionSheet(_ actionSheet: MenuActionSheetView, didSelectAction action: MenuAction)
|
|
}
|
|
|
|
class MenuActionSheetView: UIView, MenuActionViewDelegate {
|
|
|
|
private let actionStackView: UIStackView
|
|
private var actions: [MenuAction]
|
|
weak var delegate: MenuActionSheetDelegate?
|
|
|
|
override var bounds: CGRect {
|
|
didSet {
|
|
updateMask()
|
|
}
|
|
}
|
|
|
|
convenience init(actions: [MenuAction]) {
|
|
self.init(frame: CGRect.zero)
|
|
actions.forEach { self.addAction($0) }
|
|
}
|
|
|
|
override init(frame: CGRect) {
|
|
actionStackView = UIStackView()
|
|
actionStackView.axis = .vertical
|
|
actionStackView.spacing = CGHairlineWidth()
|
|
|
|
actions = []
|
|
|
|
super.init(frame: frame)
|
|
|
|
backgroundColor = UIColor.ows_light10
|
|
addSubview(actionStackView)
|
|
actionStackView.ows_autoPinToSuperviewEdges()
|
|
|
|
self.clipsToBounds = true
|
|
|
|
// Prevent panning from percolating to the superview, which would
|
|
// cause us to dismiss
|
|
let panGestureSink = UIPanGestureRecognizer(target: nil, action: nil)
|
|
self.addGestureRecognizer(panGestureSink)
|
|
}
|
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
fatalError("not implemented")
|
|
}
|
|
|
|
public func addAction(_ action: MenuAction) {
|
|
let actionView = MenuActionView(action: action)
|
|
actionView.delegate = self
|
|
actions.append(action)
|
|
self.actionStackView.addArrangedSubview(actionView)
|
|
}
|
|
|
|
// MARK: MenuActionViewDelegate
|
|
|
|
func actionView(_ actionView: MenuActionView, didSelectAction action: MenuAction) {
|
|
self.delegate?.actionSheet(self, didSelectAction: action)
|
|
}
|
|
|
|
// MARK:
|
|
|
|
private func updateMask() {
|
|
let cornerRadius: CGFloat = 16
|
|
let path: UIBezierPath = UIBezierPath(roundedRect: bounds, byRoundingCorners: [.topLeft, .topRight], cornerRadii: CGSize(width: cornerRadius, height: cornerRadius))
|
|
let mask = CAShapeLayer()
|
|
mask.path = path.cgPath
|
|
self.layer.mask = mask
|
|
}
|
|
}
|
|
|
|
protocol MenuActionViewDelegate: class {
|
|
func actionView(_ actionView: MenuActionView, didSelectAction action: MenuAction)
|
|
}
|
|
|
|
class MenuActionView: UIButton {
|
|
public weak var delegate: MenuActionViewDelegate?
|
|
private let action: MenuAction
|
|
|
|
required init(action: MenuAction) {
|
|
self.action = action
|
|
|
|
super.init(frame: CGRect.zero)
|
|
|
|
isUserInteractionEnabled = true
|
|
backgroundColor = .white
|
|
|
|
let imageView = UIImageView(image: action.image)
|
|
let imageWidth: CGFloat = 24
|
|
imageView.autoSetDimensions(to: CGSize(width: imageWidth, height: imageWidth))
|
|
imageView.isUserInteractionEnabled = false
|
|
|
|
let titleLabel = UILabel()
|
|
titleLabel.font = UIFont.ows_dynamicTypeBody
|
|
titleLabel.textColor = UIColor.ows_light90
|
|
titleLabel.text = action.title
|
|
titleLabel.isUserInteractionEnabled = false
|
|
|
|
let subtitleLabel = UILabel()
|
|
subtitleLabel.font = UIFont.ows_dynamicTypeSubheadline
|
|
subtitleLabel.textColor = UIColor.ows_light60
|
|
subtitleLabel.text = action.subtitle
|
|
subtitleLabel.isUserInteractionEnabled = false
|
|
|
|
let textColumn = UIStackView(arrangedSubviews: [titleLabel, subtitleLabel])
|
|
textColumn.axis = .vertical
|
|
textColumn.alignment = .leading
|
|
textColumn.isUserInteractionEnabled = false
|
|
|
|
let contentRow = UIStackView(arrangedSubviews: [imageView, textColumn])
|
|
contentRow.axis = .horizontal
|
|
contentRow.alignment = .center
|
|
contentRow.spacing = 12
|
|
contentRow.isLayoutMarginsRelativeArrangement = true
|
|
contentRow.layoutMargins = UIEdgeInsets(top: 7, left: 16, bottom: 7, right: 16)
|
|
contentRow.isUserInteractionEnabled = false
|
|
|
|
self.addSubview(contentRow)
|
|
contentRow.ows_autoPinToSuperviewMargins()
|
|
contentRow.autoSetDimension(.height, toSize: 56, relation: .greaterThanOrEqual)
|
|
|
|
self.addTarget(self, action: #selector(didPress(sender:)), for: .touchUpInside)
|
|
}
|
|
|
|
override var isHighlighted: Bool {
|
|
didSet {
|
|
self.backgroundColor = isHighlighted ? UIColor.ows_light10 : UIColor.white
|
|
}
|
|
}
|
|
|
|
@objc
|
|
func didPress(sender: Any) {
|
|
Logger.debug("\(logTag) in \(#function)")
|
|
self.delegate?.actionView(self, didSelectAction: action)
|
|
}
|
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
fatalError("not implemented")
|
|
}
|
|
}
|