ActionSheet.swift 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. //
  2. // ActionSheet.swift
  3. // Sheeeeeeeeet
  4. //
  5. // Created by Daniel Saidi on 2017-11-26.
  6. // Copyright © 2017 Daniel Saidi. All rights reserved.
  7. //
  8. /*
  9. This is the main class in the Sheeeeeeeeet library. You can
  10. use it to create action sheets and present them in any view
  11. controller, from any source view or bar button item.
  12. To create an action sheet, just call the initializer with a
  13. list of items and buttons and a block that should be called
  14. whenever an item is selected.
  15. ## Custom presentation
  16. You can also inject a custom sheet presenter if you want to
  17. customize how your sheet is presented and dismissed. If you
  18. do not use a custom presenter, `ActionSheetDefaultPresenter`
  19. is used. It honors the default iOS behavior by using action
  20. sheets on iPhones and popovers on iPad.
  21. ## Subclassing
  22. `ActionSheet` can be subclassed, which may be nice whenever
  23. you use Sheeeeeeeeet in your own app and want to use an app
  24. specific domain model. For instance, if you want to present
  25. a list of `Food` items, you could create a `FoodActionSheet`
  26. subclass, that is responsible to populate itself with items.
  27. When you subclass `ActionSheet` you can either override the
  28. initializers. However, you could also just override `setup`
  29. and configure the action sheet in your override.
  30. ## Appearance
  31. Sheeeeeeeeet's action sheet appearance if easily customized.
  32. To change the global appearance for every action sheet that
  33. is used in your app, use `UIActionSheetAppearance.standard`.
  34. To change the appearance of a single action sheet, use it's
  35. `appearance` property. To change the appearance of a single
  36. item, use it's `appearance` property.
  37. ## Triggered actions
  38. `ActionSheet` has two actions that are triggered by tapping
  39. an item. `itemTapAction` is used by the sheet itself when a
  40. tap occurs on an item. You can override this if you want to,
  41. but you don't have to. `itemSelectAction`, however, must be
  42. set to detect when an item is selected after a tap. This is
  43. the main item action to observe, and the action you provide
  44. in the initializer.
  45. */
  46. import UIKit
  47. open class ActionSheet: UIViewController {
  48. // MARK: - Initialization
  49. public init(
  50. items: [ActionSheetItem],
  51. presenter: ActionSheetPresenter = ActionSheet.defaultPresenter,
  52. action: @escaping SelectAction) {
  53. self.presenter = presenter
  54. itemSelectAction = action
  55. super.init(nibName: nil, bundle: nil)
  56. setupItemsAndButtons(with: items)
  57. setup()
  58. }
  59. public required init?(coder aDecoder: NSCoder) {
  60. presenter = ActionSheet.defaultPresenter
  61. itemSelectAction = { _, _ in print("itemSelectAction is not set") }
  62. super.init(coder: aDecoder)
  63. setup()
  64. }
  65. deinit { print("\(type(of: self)) deinit") }
  66. // MARK: - Setup
  67. open func setup() {
  68. view.backgroundColor = .clear
  69. }
  70. // MARK: - View Controller Lifecycle
  71. open override func viewDidLayoutSubviews() {
  72. super.viewDidLayoutSubviews()
  73. prepareForPresentation()
  74. }
  75. // MARK: - Typealiases
  76. public typealias SelectAction = (ActionSheet, ActionSheetItem) -> ()
  77. public typealias TapAction = (ActionSheetItem) -> ()
  78. // MARK: - Dependencies
  79. open var appearance = ActionSheetAppearance(copy: .standard)
  80. open var presenter: ActionSheetPresenter
  81. // MARK: - Actions
  82. open var itemSelectAction: SelectAction
  83. open lazy var itemTapAction: TapAction = { [weak self] item in
  84. self?.handleTap(on: item)
  85. }
  86. // MARK: - Item Properties
  87. open var buttons = [ActionSheetButton]()
  88. open var items = [ActionSheetItem]()
  89. // MARK: - Properties
  90. open var availableItemHeight: CGFloat {
  91. return UIScreen.main.bounds.height
  92. - 2 * margin(at: .top)
  93. - margin(at: .bottom)
  94. - headerSectionHeight
  95. - buttonsSectionHeight
  96. }
  97. open var bottomPresentationFrame: CGRect {
  98. guard let view = view.superview else { return .zero }
  99. var frame = view.frame
  100. let leftMargin = margin(at: .left)
  101. let rightMargin = margin(at: .right)
  102. let maxMargin = max(leftMargin, rightMargin)
  103. frame = frame.insetBy(dx: maxMargin, dy: 0)
  104. frame.size.height = contentHeight
  105. frame.origin.y = view.frame.height - contentHeight
  106. frame.origin.y -= margin(at: .bottom)
  107. return frame
  108. }
  109. open var buttonsSectionHeight: CGFloat {
  110. return buttonsViewHeight
  111. }
  112. open var buttonsViewHeight: CGFloat {
  113. return buttons.reduce(0) { $0 + $1.appearance.height }
  114. }
  115. open var contentHeight: CGFloat {
  116. return headerSectionHeight + itemsSectionHeight + buttonsSectionHeight
  117. }
  118. open var contentWidth: CGFloat {
  119. return super.preferredContentSize.width
  120. }
  121. open var headerSectionHeight: CGFloat {
  122. guard headerViewHeight > 0 else { return 0 }
  123. return headerViewHeight + appearance.contentInset
  124. }
  125. open var headerViewHeight: CGFloat {
  126. return headerView?.frame.height ?? 0
  127. }
  128. open var itemsSectionHeight: CGFloat {
  129. guard itemsViewHeight > 0 else { return 0 }
  130. guard buttonsSectionHeight > 0 else { return itemsViewHeight }
  131. return itemsViewHeight + appearance.contentInset
  132. }
  133. open var itemsViewHeight: CGFloat {
  134. let required = requiredItemHeight
  135. let available = availableItemHeight
  136. return min(required, available)
  137. }
  138. open var itemsViewRequiresScrolling: Bool {
  139. let required = requiredItemHeight
  140. let available = availableItemHeight
  141. return available < required
  142. }
  143. open override var preferredContentSize: CGSize {
  144. get { return CGSize(width: contentWidth, height: contentHeight) }
  145. set { super.preferredContentSize = newValue }
  146. }
  147. open var preferredPopoverSize: CGSize {
  148. let width = appearance.popover.width
  149. return CGSize(width: width, height: contentHeight)
  150. }
  151. open var requiredItemHeight: CGFloat {
  152. return items.reduce(0) { $0 + $1.appearance.height }
  153. }
  154. // MARK: - View Properties
  155. open lazy var buttonsView: UITableView = {
  156. let tableView = createTableView(handler: buttonHandler)
  157. view.addSubview(tableView)
  158. return tableView
  159. }()
  160. open var headerView: UIView? {
  161. didSet {
  162. oldValue?.removeFromSuperview()
  163. guard let header = headerView else { return }
  164. view.addSubview(header)
  165. }
  166. }
  167. open lazy var itemsView: UITableView = {
  168. let tableView = createTableView(handler: itemHandler)
  169. view.addSubview(tableView)
  170. return tableView
  171. }()
  172. // MARK: - Data Properties
  173. public lazy var buttonHandler = ActionSheetItemHandler(actionSheet: self, handles: .buttons)
  174. public lazy var itemHandler = ActionSheetItemHandler(actionSheet: self, handles: .items)
  175. // MARK: - Presentation Functions
  176. open func applyAppearance() {
  177. itemsView.separatorColor = appearance.itemsSeparatorColor
  178. buttonsView.separatorColor = appearance.buttonsSeparatorColor
  179. }
  180. open func dismiss(completion: @escaping () -> ()) {
  181. presenter.dismiss { completion() }
  182. }
  183. open func present(in vc: UIViewController, from view: UIView?) {
  184. prepareForPresentation()
  185. presenter.present(sheet: self, in: vc.rootViewController, from: view)
  186. }
  187. open func present(in vc: UIViewController, from barButtonItem: UIBarButtonItem) {
  188. prepareForPresentation()
  189. presenter.present(sheet: self, in: vc.rootViewController, from: barButtonItem)
  190. }
  191. open func prepareForPresentation() {
  192. applyAppearance()
  193. items.forEach { $0.applyAppearance(appearance) }
  194. buttons.forEach { $0.applyAppearance(appearance) }
  195. applyRoundCorners()
  196. positionViews()
  197. }
  198. // MARK: - Public Functions
  199. open func margin(at margin: ActionSheetMargin) -> CGFloat {
  200. let minimum = appearance.contentInset
  201. return margin.value(in: view.superview, minimum: minimum)
  202. }
  203. public func item(at indexPath: IndexPath) -> ActionSheetItem {
  204. return items[indexPath.row]
  205. }
  206. open func reloadData() {
  207. itemsView.reloadData()
  208. buttonsView.reloadData()
  209. }
  210. open func setupItemsAndButtons(with items: [ActionSheetItem]) {
  211. self.items = items.filter { !($0 is ActionSheetButton) }
  212. buttons = items.compactMap { $0 as? ActionSheetButton }
  213. reloadData()
  214. }
  215. }
  216. // MARK: - Private Functions
  217. private extension ActionSheet {
  218. func applyRoundCorners() {
  219. applyRoundCorners(to: headerView)
  220. applyRoundCorners(to: itemsView)
  221. applyRoundCorners(to: buttonsView)
  222. }
  223. func applyRoundCorners(to view: UIView?) {
  224. view?.clipsToBounds = true
  225. view?.layer.cornerRadius = appearance.cornerRadius
  226. }
  227. func createTableView(handler: ActionSheetItemHandler) -> UITableView {
  228. let tableView = UITableView(frame: view.frame, style: .plain)
  229. tableView.isScrollEnabled = false
  230. tableView.tableFooterView = UIView.empty
  231. tableView.cellLayoutMarginsFollowReadableWidth = false
  232. tableView.dataSource = handler
  233. tableView.delegate = handler
  234. return tableView
  235. }
  236. func handleTap(on item: ActionSheetItem) {
  237. reloadData()
  238. if item.tapBehavior == .dismiss {
  239. dismiss { self.itemSelectAction(self, item) }
  240. } else {
  241. itemSelectAction(self, item)
  242. }
  243. }
  244. func positionViews() {
  245. let width = view.frame.width
  246. positionHeaderView(width: width)
  247. positionItemsView(width: width)
  248. positionButtonsView(width: width)
  249. positionSheet()
  250. }
  251. func positionSheet() {
  252. guard let superview = view.superview else { return }
  253. guard let frame = presenter.presentationFrame(for: self, in: superview) else { return }
  254. view.frame = frame
  255. }
  256. func positionButtonsView(width: CGFloat) {
  257. buttonsView.frame.origin.x = 0
  258. buttonsView.frame.origin.y = headerSectionHeight + itemsSectionHeight
  259. buttonsView.frame.size.width = width
  260. buttonsView.frame.size.height = buttonsViewHeight
  261. }
  262. func positionHeaderView(width: CGFloat) {
  263. guard let view = headerView else { return }
  264. view.frame.origin = .zero
  265. view.frame.size.width = width
  266. }
  267. func positionItemsView(width: CGFloat) {
  268. itemsView.frame.origin.x = 0
  269. itemsView.frame.origin.y = headerSectionHeight
  270. itemsView.frame.size.width = width
  271. itemsView.frame.size.height = itemsViewHeight
  272. itemsView.isScrollEnabled = itemsViewRequiresScrolling
  273. }
  274. }