UIControl+QMUI.m 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. /**
  2. * Tencent is pleased to support the open source community by making QMUI_iOS available.
  3. * Copyright (C) 2016-2021 THL A29 Limited, a Tencent company. All rights reserved.
  4. * Licensed under the MIT License (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
  5. * http://opensource.org/licenses/MIT
  6. * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
  7. */
  8. //
  9. // UIControl+QMUI.m
  10. // qmui
  11. //
  12. // Created by QMUI Team on 15/7/20.
  13. //
  14. #import "UIControl+QMUI.h"
  15. #import "QMUICore.h"
  16. @interface UIControl ()
  17. @property(nonatomic,assign) BOOL qmuictl_canSetHighlighted;
  18. @property(nonatomic,assign) NSInteger qmuictl_touchEndCount;
  19. @end
  20. @implementation UIControl (QMUI)
  21. QMUISynthesizeBOOLProperty(qmuictl_canSetHighlighted, setQmuictl_canSetHighlighted)
  22. QMUISynthesizeNSIntegerProperty(qmuictl_touchEndCount, setQmuictl_touchEndCount)
  23. #pragma mark - Automatically Adjust Touch Highlighted In ScrollView
  24. static char kAssociatedObjectKey_automaticallyAdjustTouchHighlightedInScrollView;
  25. - (void)setQmui_automaticallyAdjustTouchHighlightedInScrollView:(BOOL)qmui_automaticallyAdjustTouchHighlightedInScrollView {
  26. objc_setAssociatedObject(self, &kAssociatedObjectKey_automaticallyAdjustTouchHighlightedInScrollView, @(qmui_automaticallyAdjustTouchHighlightedInScrollView), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
  27. if (qmui_automaticallyAdjustTouchHighlightedInScrollView) {
  28. [QMUIHelper executeBlock:^{
  29. OverrideImplementation([UIControl class], @selector(touchesBegan:withEvent:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  30. return ^(UIControl *selfObject, NSSet *touches, UIEvent *event) {
  31. // call super
  32. void (^callSuperBlock)(void) = ^{
  33. void (*originSelectorIMP)(id, SEL, NSSet *, UIEvent *);
  34. originSelectorIMP = (void (*)(id, SEL, NSSet *, UIEvent *))originalIMPProvider();
  35. originSelectorIMP(selfObject, originCMD, touches, event);
  36. };
  37. selfObject.qmuictl_touchEndCount = 0;
  38. if (selfObject.qmui_automaticallyAdjustTouchHighlightedInScrollView) {
  39. selfObject.qmuictl_canSetHighlighted = YES;
  40. callSuperBlock();
  41. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  42. if (selfObject.qmuictl_canSetHighlighted) {
  43. [selfObject setHighlighted:YES];
  44. }
  45. });
  46. } else {
  47. callSuperBlock();
  48. }
  49. };
  50. });
  51. OverrideImplementation([UIControl class], @selector(touchesMoved:withEvent:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  52. return ^(UIControl *selfObject, NSSet *touches, UIEvent *event) {
  53. if (selfObject.qmui_automaticallyAdjustTouchHighlightedInScrollView) {
  54. selfObject.qmuictl_canSetHighlighted = NO;
  55. }
  56. // call super
  57. void (*originSelectorIMP)(id, SEL, NSSet *, UIEvent *);
  58. originSelectorIMP = (void (*)(id, SEL, NSSet *, UIEvent *))originalIMPProvider();
  59. originSelectorIMP(selfObject, originCMD, touches, event);
  60. };
  61. });
  62. OverrideImplementation([UIControl class], @selector(touchesEnded:withEvent:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  63. return ^(UIControl *selfObject, NSSet *touches, UIEvent *event) {
  64. if (selfObject.qmui_automaticallyAdjustTouchHighlightedInScrollView) {
  65. selfObject.qmuictl_canSetHighlighted = NO;
  66. if (selfObject.touchInside) {
  67. [selfObject setHighlighted:YES];
  68. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.02 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  69. // 如果延迟时间太长,会导致快速点击两次,事件会触发两次
  70. // 对于 3D Touch 的机器,如果点击按钮的时候在按钮上停留事件稍微长一点点,那么 touchesEnded 会被调用两次
  71. // 把 super touchEnded 放到延迟里调用会导致长按无法触发点击,先这么改,再想想怎么办。// [selfObject qmui_touchesEnded:touches withEvent:event];
  72. [selfObject sendActionsForAllTouchEventsIfCan];
  73. if (selfObject.highlighted) {
  74. [selfObject setHighlighted:NO];
  75. }
  76. });
  77. } else {
  78. [selfObject setHighlighted:NO];
  79. }
  80. return;
  81. }
  82. // call super
  83. void (*originSelectorIMP)(id, SEL, NSSet *, UIEvent *);
  84. originSelectorIMP = (void (*)(id, SEL, NSSet *, UIEvent *))originalIMPProvider();
  85. originSelectorIMP(selfObject, originCMD, touches, event);
  86. };
  87. });
  88. OverrideImplementation([UIControl class], @selector(touchesCancelled:withEvent:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  89. return ^(UIControl *selfObject, NSSet *touches, UIEvent *event) {
  90. // call super
  91. void (^callSuperBlock)(void) = ^{
  92. void (*originSelectorIMP)(id, SEL, NSSet *, UIEvent *);
  93. originSelectorIMP = (void (*)(id, SEL, NSSet *, UIEvent *))originalIMPProvider();
  94. originSelectorIMP(selfObject, originCMD, touches, event);
  95. };
  96. if (selfObject.qmui_automaticallyAdjustTouchHighlightedInScrollView) {
  97. selfObject.qmuictl_canSetHighlighted = NO;
  98. callSuperBlock();
  99. if (selfObject.highlighted) {
  100. [selfObject setHighlighted:NO];
  101. }
  102. return;
  103. }
  104. callSuperBlock();
  105. };
  106. });
  107. } oncePerIdentifier:@"UIControl automaticallyAdjustTouchHighlightedInScrollView"];
  108. }
  109. }
  110. - (BOOL)qmui_automaticallyAdjustTouchHighlightedInScrollView {
  111. return [((NSNumber *)objc_getAssociatedObject(self, &kAssociatedObjectKey_automaticallyAdjustTouchHighlightedInScrollView)) boolValue];
  112. }
  113. // 这段代码需要以一个独立的方法存在,因为一旦有坑,外面可以直接通过runtime调用这个方法
  114. // 但,不要开放到.h文件里,理论上外面不应该用到它
  115. - (void)sendActionsForAllTouchEventsIfCan {
  116. self.qmuictl_touchEndCount += 1;
  117. if (self.qmuictl_touchEndCount == 1) {
  118. [self sendActionsForControlEvents:UIControlEventAllTouchEvents];
  119. }
  120. }
  121. #pragma mark - Prevents Repeated TouchUpInside Event
  122. static char kAssociatedObjectKey_preventsRepeatedTouchUpInsideEvent;
  123. - (void)setQmui_preventsRepeatedTouchUpInsideEvent:(BOOL)qmui_preventsRepeatedTouchUpInsideEvent {
  124. objc_setAssociatedObject(self, &kAssociatedObjectKey_preventsRepeatedTouchUpInsideEvent, @(qmui_preventsRepeatedTouchUpInsideEvent), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
  125. if (qmui_preventsRepeatedTouchUpInsideEvent) {
  126. [QMUIHelper executeBlock:^{
  127. OverrideImplementation([UIControl class], @selector(sendAction:to:forEvent:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  128. return ^(UIControl *selfObject, SEL action, id target, UIEvent *event) {
  129. if (selfObject.qmui_preventsRepeatedTouchUpInsideEvent) {
  130. NSArray<NSString *> *actions = [selfObject actionsForTarget:target forControlEvent:UIControlEventTouchUpInside];
  131. if (!actions) {
  132. // iOS 10 UIBarButtonItem 里的 UINavigationButton 点击事件用的是 UIControlEventPrimaryActionTriggered
  133. actions = [selfObject actionsForTarget:target forControlEvent:UIControlEventPrimaryActionTriggered];
  134. }
  135. if ([actions containsObject:NSStringFromSelector(action)]) {
  136. UITouch *touch = event.allTouches.anyObject;
  137. if (touch.tapCount > 1) {
  138. return;
  139. }
  140. }
  141. }
  142. // call super
  143. void (*originSelectorIMP)(id, SEL, SEL, id, UIEvent *);
  144. originSelectorIMP = (void (*)(id, SEL, SEL, id, UIEvent *))originalIMPProvider();
  145. originSelectorIMP(selfObject, originCMD, action, target, event);
  146. };
  147. });
  148. } oncePerIdentifier:@"UIControl preventsRepeatedTouchUpInsideEvent"];
  149. }
  150. }
  151. - (BOOL)qmui_preventsRepeatedTouchUpInsideEvent {
  152. return [((NSNumber *)objc_getAssociatedObject(self, &kAssociatedObjectKey_preventsRepeatedTouchUpInsideEvent)) boolValue];
  153. }
  154. #pragma mark - Highlighted Block
  155. static char kAssociatedObjectKey_setHighlightedBlock;
  156. - (void)setQmui_setHighlightedBlock:(void (^)(BOOL))qmui_setHighlightedBlock {
  157. objc_setAssociatedObject(self, &kAssociatedObjectKey_setHighlightedBlock, qmui_setHighlightedBlock, OBJC_ASSOCIATION_COPY_NONATOMIC);
  158. if (qmui_setHighlightedBlock) {
  159. [QMUIHelper executeBlock:^{
  160. OverrideImplementation([UIControl class], @selector(setHighlighted:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  161. return ^(UIControl *selfObject, BOOL highlighted) {
  162. // call super
  163. void (*originSelectorIMP)(id, SEL, BOOL);
  164. originSelectorIMP = (void (*)(id, SEL, BOOL))originalIMPProvider();
  165. originSelectorIMP(selfObject, originCMD, highlighted);
  166. if (selfObject.qmui_setHighlightedBlock) {
  167. selfObject.qmui_setHighlightedBlock(highlighted);
  168. }
  169. };
  170. });
  171. } oncePerIdentifier:@"UIControl setHighlighted:"];
  172. }
  173. }
  174. - (void (^)(BOOL))qmui_setHighlightedBlock {
  175. return (void (^)(BOOL))objc_getAssociatedObject(self, &kAssociatedObjectKey_setHighlightedBlock);
  176. }
  177. #pragma mark - Selected Block
  178. static char kAssociatedObjectKey_setSelectedBlock;
  179. - (void)setQmui_setSelectedBlock:(void (^)(BOOL))qmui_setSelectedBlock {
  180. objc_setAssociatedObject(self, &kAssociatedObjectKey_setSelectedBlock, qmui_setSelectedBlock, OBJC_ASSOCIATION_COPY_NONATOMIC);
  181. if (qmui_setSelectedBlock) {
  182. [QMUIHelper executeBlock:^{
  183. OverrideImplementation([UIControl class], @selector(setSelected:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  184. return ^(UIControl *selfObject, BOOL selected) {
  185. // call super
  186. void (*originSelectorIMP)(id, SEL, BOOL);
  187. originSelectorIMP = (void (*)(id, SEL, BOOL))originalIMPProvider();
  188. originSelectorIMP(selfObject, originCMD, selected);
  189. if (selfObject.qmui_setSelectedBlock) {
  190. selfObject.qmui_setSelectedBlock(selected);
  191. }
  192. };
  193. });
  194. } oncePerIdentifier:@"UIControl setSelected:"];
  195. }
  196. }
  197. - (void (^)(BOOL))qmui_setSelectedBlock {
  198. return (void (^)(BOOL))objc_getAssociatedObject(self, &kAssociatedObjectKey_setSelectedBlock);
  199. }
  200. #pragma mark - Enabled Block
  201. static char kAssociatedObjectKey_setEnabledBlock;
  202. - (void)setQmui_setEnabledBlock:(void (^)(BOOL))qmui_setEnabledBlock {
  203. objc_setAssociatedObject(self, &kAssociatedObjectKey_setEnabledBlock, qmui_setEnabledBlock, OBJC_ASSOCIATION_COPY_NONATOMIC);
  204. if (qmui_setEnabledBlock) {
  205. [QMUIHelper executeBlock:^{
  206. OverrideImplementation([UIControl class], @selector(setEnabled:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  207. return ^(UIControl *selfObject, BOOL enabled) {
  208. // call super
  209. void (*originSelectorIMP)(id, SEL, BOOL);
  210. originSelectorIMP = (void (*)(id, SEL, BOOL))originalIMPProvider();
  211. originSelectorIMP(selfObject, originCMD, enabled);
  212. if (selfObject.qmui_setEnabledBlock) {
  213. selfObject.qmui_setEnabledBlock(enabled);
  214. }
  215. };
  216. });
  217. } oncePerIdentifier:@"UIControl setEnabled:"];
  218. }
  219. }
  220. - (void (^)(BOOL))qmui_setEnabledBlock {
  221. return (void (^)(BOOL))objc_getAssociatedObject(self, &kAssociatedObjectKey_setEnabledBlock);
  222. }
  223. #pragma mark - Tap Block
  224. static char kAssociatedObjectKey_tapBlock;
  225. - (void)setQmui_tapBlock:(void (^)(__kindof UIControl *))qmui_tapBlock {
  226. if (qmui_tapBlock) {
  227. [QMUIHelper executeBlock:^{
  228. OverrideImplementation([UIControl class], @selector(removeTarget:action:forControlEvents:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
  229. return ^(UIControl *selfObject, id target, SEL action, UIControlEvents controlEvents) {
  230. // call super
  231. void (*originSelectorIMP)(id, SEL, id, SEL, UIControlEvents);
  232. originSelectorIMP = (void (*)(id, SEL, id, SEL, UIControlEvents))originalIMPProvider();
  233. originSelectorIMP(selfObject, originCMD, target, action, controlEvents);
  234. BOOL isTouchUpInsideEvent = controlEvents & UIControlEventTouchUpInside;
  235. BOOL shouldRemoveTouchUpInsideSelector = (action == @selector(qmui_handleTouchUpInside:)) || (target == selfObject && !action) || (!target && !action);
  236. if (isTouchUpInsideEvent && shouldRemoveTouchUpInsideSelector) {
  237. // 避免触发 setter 又反过来 removeTarget,然后就死循环了
  238. objc_setAssociatedObject(selfObject, &kAssociatedObjectKey_tapBlock, nil, OBJC_ASSOCIATION_COPY_NONATOMIC);
  239. }
  240. };
  241. });
  242. } oncePerIdentifier:@"UIControl tapBlock"];
  243. }
  244. SEL action = @selector(qmui_handleTouchUpInside:);
  245. if (!qmui_tapBlock) {
  246. [self removeTarget:self action:action forControlEvents:UIControlEventTouchUpInside];
  247. } else {
  248. [self addTarget:self action:action forControlEvents:UIControlEventTouchUpInside];
  249. }
  250. objc_setAssociatedObject(self, &kAssociatedObjectKey_tapBlock, qmui_tapBlock, OBJC_ASSOCIATION_COPY_NONATOMIC);
  251. }
  252. - (void (^)(__kindof UIControl *))qmui_tapBlock {
  253. return (void (^)(__kindof UIControl *))objc_getAssociatedObject(self, &kAssociatedObjectKey_tapBlock);
  254. }
  255. - (void)qmui_handleTouchUpInside:(__kindof UIControl *)sender {
  256. if (self.qmui_tapBlock) {
  257. self.qmui_tapBlock(self);
  258. }
  259. }
  260. @end