LNNestedScrollView.swift 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. //
  2. // LNNestedScrollView.swift
  3. // Gami
  4. //
  5. // Created by OneeChan on 2026/3/12.
  6. //
  7. import Foundation
  8. import UIKit
  9. private class ScrollViewWeakWrapper {
  10. weak var delegate: LNNestedScrollViewDelegate?
  11. weak var childListView: UIScrollView?
  12. }
  13. private var scrollViewWeakWrapperKey: UInt8 = 0
  14. private extension UIScrollView {
  15. var minTopOffset: CGFloat {
  16. contentSize.height - bounds.height - contentInset.top + contentInset.bottom
  17. }
  18. var weakWrapper: ScrollViewWeakWrapper {
  19. get {
  20. var wrapper = objc_getAssociatedObject(self, &scrollViewWeakWrapperKey)
  21. as? ScrollViewWeakWrapper
  22. if wrapper == nil {
  23. wrapper = ScrollViewWeakWrapper()
  24. objc_setAssociatedObject(
  25. self,
  26. &scrollViewWeakWrapperKey,
  27. wrapper,
  28. .OBJC_ASSOCIATION_RETAIN_NONATOMIC
  29. )
  30. }
  31. return wrapper!
  32. }
  33. set {
  34. objc_setAssociatedObject(
  35. self,
  36. &scrollViewWeakWrapperKey,
  37. newValue,
  38. .OBJC_ASSOCIATION_RETAIN_NONATOMIC
  39. )
  40. }
  41. }
  42. weak var curChildListView: UIScrollView? {
  43. get { weakWrapper.childListView }
  44. set { weakWrapper.childListView = newValue }
  45. }
  46. }
  47. extension UIScrollView: LNNestedScrollViewDelegate {
  48. func listViewDidScroll(_ scrollView: UIScrollView) -> Bool {
  49. curChildListView = scrollView
  50. if contentOffset.y < minTopOffset - 0.001 {
  51. return true
  52. }
  53. contentOffset.y = minTopOffset
  54. return false
  55. }
  56. }
  57. extension UIScrollView {
  58. weak var observerDelegate: LNNestedScrollViewDelegate? {
  59. get { weakWrapper.delegate }
  60. set { weakWrapper.delegate = newValue }
  61. }
  62. }
  63. protocol LNNestedScrollViewDelegate: NSObject {
  64. func listViewDidScroll(_ scrollView: UIScrollView) -> Bool
  65. }
  66. private class ScrollViewDelegateProxy: NSObject, UIScrollViewDelegate {
  67. weak var originalDelegate: UIScrollViewDelegate?
  68. weak var observerDelegate: UIScrollViewDelegate?
  69. func scrollViewDidScroll(_ scrollView: UIScrollView) {
  70. originalDelegate?.scrollViewDidScroll?(scrollView)
  71. observerDelegate?.scrollViewDidScroll?(scrollView)
  72. }
  73. override func forwardingTarget(for aSelector: Selector!) -> Any? {
  74. if originalDelegate?.responds(to: aSelector) == true {
  75. return originalDelegate
  76. }
  77. return super.forwardingTarget(for: aSelector)
  78. }
  79. override func responds(to aSelector: Selector!) -> Bool {
  80. return super.responds(to: aSelector) || originalDelegate?.responds(to: aSelector) == true
  81. }
  82. }
  83. class LNNestedScrollView: UIScrollView, UIScrollViewDelegate, UIGestureRecognizerDelegate {
  84. weak var scrollDelegate: UIScrollViewDelegate? {
  85. didSet {
  86. proxy.originalDelegate = scrollDelegate
  87. delegate = nil
  88. delegate = proxy
  89. }
  90. }
  91. private var proxy = ScrollViewDelegateProxy()
  92. override init(frame: CGRect) {
  93. super.init(frame: frame)
  94. proxy.originalDelegate = scrollDelegate
  95. proxy.observerDelegate = self
  96. delegate = proxy
  97. }
  98. required init?(coder: NSCoder) {
  99. fatalError("init(coder:) has not been implemented")
  100. }
  101. func scrollViewDidScroll(_ scrollView: UIScrollView) {
  102. if observerDelegate?.listViewDidScroll(self) == true {
  103. scrollView.contentOffset.y = -contentInset.top
  104. }
  105. if let curChildListView,
  106. curChildListView.contentOffset.y > -curChildListView.contentInset.top {
  107. scrollView.contentOffset.y = minTopOffset
  108. }
  109. }
  110. public func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool {
  111. gestureRecognizer.isKind(of: UIPanGestureRecognizer.self) && otherGestureRecognizer.isKind(of: UIPanGestureRecognizer.self)
  112. }
  113. }
  114. private class TableViewDelegateProxy: NSObject, UITableViewDelegate {
  115. weak var originalDelegate: UITableViewDelegate?
  116. weak var observerDelegate: UITableViewDelegate?
  117. func scrollViewDidScroll(_ scrollView: UIScrollView) {
  118. originalDelegate?.scrollViewDidScroll?(scrollView)
  119. observerDelegate?.scrollViewDidScroll?(scrollView)
  120. }
  121. override func forwardingTarget(for aSelector: Selector!) -> Any? {
  122. if originalDelegate?.responds(to: aSelector) == true {
  123. return originalDelegate
  124. }
  125. return super.forwardingTarget(for: aSelector)
  126. }
  127. override func responds(to aSelector: Selector!) -> Bool {
  128. super.responds(to: aSelector)
  129. || originalDelegate?.responds(to: aSelector) == true
  130. }
  131. }
  132. class LNNestedTableView: UITableView, UITableViewDelegate, UIGestureRecognizerDelegate {
  133. weak var tableDelegate: UITableViewDelegate? {
  134. didSet {
  135. proxy.originalDelegate = tableDelegate
  136. delegate = nil
  137. delegate = proxy
  138. }
  139. }
  140. private var proxy = TableViewDelegateProxy()
  141. override init(frame: CGRect, style: UITableView.Style) {
  142. super.init(frame: frame, style: style)
  143. proxy.originalDelegate = tableDelegate
  144. proxy.observerDelegate = self
  145. delegate = proxy
  146. }
  147. required init?(coder: NSCoder) {
  148. fatalError("init(coder:) has not been implemented")
  149. }
  150. func scrollViewDidScroll(_ scrollView: UIScrollView) {
  151. if observerDelegate?.listViewDidScroll(self) == true {
  152. scrollView.contentOffset.y = -contentInset.top
  153. }
  154. if let curChildListView,
  155. curChildListView.contentOffset.y > -curChildListView.contentInset.top {
  156. scrollView.contentOffset.y = minTopOffset
  157. }
  158. }
  159. public func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool {
  160. gestureRecognizer.isKind(of: UIPanGestureRecognizer.self) && otherGestureRecognizer.isKind(of: UIPanGestureRecognizer.self)
  161. }
  162. }