暂无描述

PageViewController.swift 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. //
  2. // PageViewController.swift
  3. // PaiAi
  4. //
  5. // Created by ffib on 2018/12/6.
  6. // Copyright © 2018 yb. All rights reserved.
  7. //
  8. import UIKit
  9. fileprivate let baseTag = 55161750
  10. open class PageViewController: UIViewController {
  11. private enum ScrollDirection {
  12. case left
  13. case right
  14. }
  15. private var contentRect = CGRect.zero
  16. //animation auxiliary
  17. private var currentIndex: Int = 0
  18. private var distance: CGFloat = 0
  19. private var currentItem: UILabel?
  20. private var startOffset: CGFloat = 0
  21. private var isBeginScroll: Bool = false
  22. private var menuItemWidths: [CGFloat] = []
  23. private var direction: ScrollDirection = .right
  24. private var sliderConstraint: NSLayoutConstraint?
  25. public private(set) lazy var scrollView: UIScrollView = {
  26. let scrollView = UIScrollView()
  27. scrollView.bounces = false
  28. scrollView.delegate = self
  29. scrollView.isPagingEnabled = true
  30. scrollView.alwaysBounceVertical = false
  31. scrollView.showsVerticalScrollIndicator = false
  32. scrollView.showsHorizontalScrollIndicator = false
  33. return scrollView
  34. }()
  35. public private(set) lazy var menuView: UIView = {
  36. let view = UIView()
  37. return view
  38. }()
  39. public private(set) lazy var sliderView: UIView = {
  40. let view = UIView()
  41. view.layer.cornerRadius = 2.5
  42. view.backgroundColor = option.selectedColor
  43. return view
  44. }()
  45. public var option = PageOption()
  46. public var pageItems = [PageItem]() {
  47. didSet {
  48. setPageItems()
  49. setMenuItems()
  50. }
  51. }
  52. override open func viewDidLoad() {
  53. super.viewDidLoad()
  54. contentRect = CGRect(x: 0, y: 0, width: view.frame.width, height: view.frame.height)
  55. if #available(iOS 11, *) {
  56. scrollView.contentInsetAdjustmentBehavior = .never
  57. }
  58. constructViewHierarchy()
  59. activateConstraints()
  60. setMenuGestureRecognizer()
  61. setupNavigationBarInteractivePopDelegate()
  62. }
  63. open override func viewDidLayoutSubviews() {
  64. super.viewDidLayoutSubviews()
  65. cacheMenuItemWidth()
  66. }
  67. private func cacheMenuItemWidth() {
  68. guard menuItemWidths.isEmpty else { return }
  69. menuItemWidths = Array(repeating: 0, count: menuView.subviews.count - 1)
  70. for view in menuView.subviews {
  71. guard let label = view as? UILabel else { continue }
  72. menuItemWidths[label.tag - baseTag] = label.frame.width
  73. }
  74. }
  75. private func constructViewHierarchy() {
  76. view.addSubview(scrollView)
  77. navigationController?.navigationBar.addSubview(menuView)
  78. menuView.addSubview(sliderView)
  79. }
  80. }
  81. /// set PageItem and MenuItem
  82. fileprivate extension PageViewController {
  83. func setPageItems() {
  84. guard !pageItems.isEmpty else { return }
  85. var navigationBarHeight: CGFloat
  86. if UIApplication.shared.statusBarFrame.height == 44 {
  87. navigationBarHeight = 88
  88. } else {
  89. navigationBarHeight = 64
  90. }
  91. scrollView.contentSize = CGSize(width: contentRect.width * CGFloat(pageItems.count),
  92. height: contentRect.height - navigationBarHeight)
  93. var last: UIView?
  94. for item in pageItems {
  95. addChild(item.viewController)
  96. scrollView.addSubview(item.viewController.view)
  97. item.viewController.view.translatesAutoresizingMaskIntoConstraints = false
  98. let width = item.viewController.view
  99. .widthAnchor
  100. .constraint(equalTo: scrollView.widthAnchor)
  101. let height = item.viewController.view
  102. .heightAnchor
  103. .constraint(equalTo: scrollView.heightAnchor)
  104. let top = item.viewController.view
  105. .topAnchor
  106. .constraint(equalTo: scrollView.topAnchor)
  107. let leading = item.viewController.view
  108. .leadingAnchor
  109. .constraint(equalTo: last?.trailingAnchor ?? scrollView.leadingAnchor)
  110. NSLayoutConstraint.activate([width, height, top, leading])
  111. last = item.viewController.view
  112. }
  113. }
  114. func setMenuItems() {
  115. guard !pageItems.isEmpty else { return }
  116. var last: UILabel?
  117. for (i, item) in pageItems.enumerated() {
  118. let label = UILabel()
  119. label.text = item.title
  120. label.tag = baseTag + i
  121. label.font = option.font
  122. label.textAlignment = .center
  123. label.textColor = option.normalColor
  124. label.translatesAutoresizingMaskIntoConstraints = false
  125. menuView.addSubview(label)
  126. let left: NSLayoutConstraint
  127. if let lastLabel = last {
  128. left = label.leftAnchor
  129. .constraint(equalTo: lastLabel.rightAnchor, constant: option.spacing)
  130. } else {
  131. left = label.leadingAnchor
  132. .constraint(equalTo: menuView.leadingAnchor)
  133. }
  134. let centerY = label.centerYAnchor
  135. .constraint(equalTo: menuView.centerYAnchor)
  136. if i == 0 {
  137. label.textColor = option.selectedColor
  138. label.font = UIFont.systemFont(ofSize: option.font.pointSize + 1)
  139. currentItem = label
  140. } else if i == pageItems.count - 1 {
  141. NSLayoutConstraint.activate([label.trailingAnchor
  142. .constraint(equalTo: menuView.trailingAnchor)])
  143. }
  144. NSLayoutConstraint.activate([left, centerY])
  145. last = label
  146. }
  147. setSliderViewDetail()
  148. }
  149. func setSliderViewDetail() {
  150. guard let label = menuView.viewWithTag(baseTag) else { return }
  151. sliderConstraint = sliderView.centerXAnchor
  152. .constraint(equalTo: label.centerXAnchor)
  153. NSLayoutConstraint.activate([sliderConstraint!])
  154. }
  155. }
  156. /// layout
  157. fileprivate extension PageViewController {
  158. func activateConstraints() {
  159. activateConstraintsMenuView()
  160. activateConstraintsSliderView()
  161. activateConstraintsScrollView()
  162. }
  163. func activateConstraintsScrollView() {
  164. scrollView.translatesAutoresizingMaskIntoConstraints = false
  165. let top = scrollView.topAnchor.constraint(equalTo: view.topAnchor)
  166. let width = scrollView.widthAnchor.constraint(equalTo: view.widthAnchor)
  167. let bottom = scrollView.bottomAnchor.constraint(equalTo: view.bottomAnchor)
  168. let height = scrollView.heightAnchor.constraint(equalTo: view.heightAnchor)
  169. let leading = scrollView.leadingAnchor.constraint(equalTo: view.leadingAnchor)
  170. let trailing = scrollView.trailingAnchor.constraint(equalTo: view.trailingAnchor)
  171. NSLayoutConstraint.activate([width, height, top, leading, bottom, trailing])
  172. }
  173. func activateConstraintsMenuView() {
  174. guard let barContentView = navigationController?.navigationBar else { return }
  175. menuView.translatesAutoresizingMaskIntoConstraints = false
  176. NSLayoutConstraint.activate([
  177. menuView.topAnchor.constraint(equalTo: barContentView.topAnchor),
  178. menuView.bottomAnchor.constraint(equalTo: barContentView.bottomAnchor),
  179. menuView.centerXAnchor.constraint(equalTo: barContentView.centerXAnchor)
  180. ])
  181. }
  182. func activateConstraintsSliderView() {
  183. sliderView.translatesAutoresizingMaskIntoConstraints = false
  184. NSLayoutConstraint.activate([
  185. sliderView.widthAnchor.constraint(equalToConstant: 5),
  186. sliderView.heightAnchor.constraint(equalToConstant: 5),
  187. sliderView.bottomAnchor.constraint(equalTo: menuView.bottomAnchor, constant: -2)
  188. ])
  189. }
  190. }
  191. /// GuestureRecognizer
  192. fileprivate extension PageViewController {
  193. func setMenuGestureRecognizer() {
  194. let tap = UITapGestureRecognizer(target: self, action: #selector(tapMenu(tap:)))
  195. menuView.addGestureRecognizer(tap)
  196. }
  197. @objc func tapMenu(tap: UITapGestureRecognizer) {
  198. var x = tap.location(in: menuView).x
  199. for (i, width) in menuItemWidths.enumerated() {
  200. x -= (width + option.spacing / 2)
  201. guard x <= 0 else { continue }
  202. if i != currentIndex {
  203. didSelect(i)
  204. }
  205. return
  206. }
  207. }
  208. }
  209. /// UIScrollViewDelegate implementation
  210. extension PageViewController: UIScrollViewDelegate {
  211. public func scrollViewWillBeginDragging(_ scrollView: UIScrollView) {
  212. startOffset = self.scrollView.contentOffset.x
  213. isBeginScroll = true
  214. }
  215. public func scrollViewDidScroll(_ scrollView: UIScrollView) {
  216. guard scrollView.isDragging || scrollView.isDecelerating || scrollView.isTracking else { return }
  217. initializeScrollParameter()
  218. moveSlider(percentage: (self.scrollView.contentOffset.x - startOffset ) / contentRect.width)
  219. }
  220. public func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool) {
  221. guard !decelerate else { return }
  222. pageViewDidEndScroll()
  223. }
  224. public func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
  225. pageViewDidEndScroll()
  226. }
  227. public func scrollViewDidEndScrollingAnimation(_ scrollView: UIScrollView) {
  228. pageViewDidEndScroll()
  229. }
  230. }
  231. /// linkage
  232. fileprivate extension PageViewController {
  233. func didSelect(_ index: Int) {
  234. startOffset = self.scrollView.contentOffset.x
  235. sliderAnimation(index)
  236. didSelectPageItem(index)
  237. }
  238. func initializeScrollParameter() {
  239. if isBeginScroll {
  240. direction = self.scrollView.contentOffset.x - startOffset > 0 ? .right : .left
  241. guard let label = currentItem else { return }
  242. switch direction {
  243. case .left:
  244. guard label.tag - baseTag - 1 > 0 else { return }
  245. distance = menuItemWidths[label.tag - baseTag] / 2 + menuItemWidths[label.tag - baseTag - 1] / 2 + option.spacing
  246. case .right:
  247. guard label.tag - baseTag + 1 < menuItemWidths.count else { return }
  248. distance = menuItemWidths[label.tag - baseTag] / 2 + menuItemWidths[label.tag - baseTag + 1] / 2 + option.spacing
  249. }
  250. isBeginScroll = false
  251. }
  252. }
  253. }
  254. /// menu linkage
  255. fileprivate extension PageViewController {
  256. func moveSlider(percentage: CGFloat) {
  257. sliderConstraint?.constant = percentage * distance
  258. }
  259. func didSelectMenuItem(_ index: Int) {
  260. guard let currentLabel = currentItem,
  261. let label = menuView.viewWithTag(baseTag + index) as? UILabel else { return }
  262. currentItem = label
  263. currentLabel.font = option.font
  264. currentLabel.textColor = option.normalColor
  265. label.textColor = option.selectedColor
  266. label.font = UIFont.systemFont(ofSize: option.font.pointSize + 1)
  267. currentIndex = index
  268. updateSliderConstraint()
  269. }
  270. func sliderAnimation(_ index: Int) {
  271. let isLeft = currentIndex - index > 0
  272. var animationDistance: CGFloat = 0
  273. for i in isLeft ? (index..<currentIndex) : (currentIndex..<index) {
  274. animationDistance += menuItemWidths[i] / 2 + menuItemWidths[i + 1] / 2 + option.spacing
  275. }
  276. UIView.animate(withDuration: 0.25) {
  277. self.sliderConstraint?.constant = isLeft ? -animationDistance : animationDistance
  278. self.menuView.layoutIfNeeded()
  279. }
  280. }
  281. func updateSliderConstraint() {
  282. NSLayoutConstraint.deactivate([sliderConstraint!])
  283. sliderConstraint = sliderView.centerXAnchor.constraint(equalTo: currentItem!.centerXAnchor)
  284. NSLayoutConstraint.activate([sliderConstraint!])
  285. }
  286. }
  287. /// page linkage
  288. fileprivate extension PageViewController {
  289. func didSelectPageItem(_ index: Int) {
  290. scrollView.setContentOffset(CGPoint.init(x: CGFloat(index) * contentRect.width, y: 0), animated: true)
  291. }
  292. func pageViewDidEndScroll() {
  293. guard self.scrollView.contentOffset.x != startOffset else { return }
  294. let index = Int(self.scrollView.contentOffset.x / contentRect.width)
  295. pageItems[index].viewController.didMove(toParent: self)
  296. switch direction {
  297. case .left:
  298. didSelectMenuItem(index)
  299. case .right:
  300. didSelectMenuItem(index)
  301. }
  302. }
  303. }
  304. extension PageViewController: NavigationBarInteractiveViewController {
  305. public var navigationView: UIView {
  306. return menuView
  307. }
  308. }