FLEXManager+Extensibility.m 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. //
  2. // FLEXManager+Extensibility.m
  3. // FLEX
  4. //
  5. // Created by Tanner on 2/2/20.
  6. // Copyright © 2020 FLEX Team. All rights reserved.
  7. //
  8. #import "FLEXManager+Extensibility.h"
  9. #import "FLEXManager+Private.h"
  10. #import "FLEXNavigationController.h"
  11. #import "FLEXGlobalsEntry.h"
  12. #import "FLEXObjectExplorerFactory.h"
  13. #import "FLEXKeyboardShortcutManager.h"
  14. #import "FLEXExplorerViewController.h"
  15. #import "FLEXNetworkMITMViewController.h"
  16. #import "FLEXKeyboardHelpViewController.h"
  17. #import "FLEXFileBrowserController.h"
  18. #import "FLEXUtility.h"
  19. @interface FLEXManager (ExtensibilityPrivate)
  20. @property (nonatomic, readonly) UIViewController *topViewController;
  21. @end
  22. @implementation FLEXManager (Extensibility)
  23. #pragma mark - Globals Screen Entries
  24. - (void)registerGlobalEntryWithName:(NSString *)entryName objectFutureBlock:(id (^)(void))objectFutureBlock {
  25. NSParameterAssert(entryName);
  26. NSParameterAssert(objectFutureBlock);
  27. NSAssert(NSThread.isMainThread, @"This method must be called from the main thread.");
  28. entryName = entryName.copy;
  29. FLEXGlobalsEntry *entry = [FLEXGlobalsEntry entryWithNameFuture:^NSString *{
  30. return entryName;
  31. } viewControllerFuture:^UIViewController *{
  32. return [FLEXObjectExplorerFactory explorerViewControllerForObject:objectFutureBlock()];
  33. }];
  34. [self.userGlobalEntries addObject:entry];
  35. }
  36. - (void)registerGlobalEntryWithName:(NSString *)entryName viewControllerFutureBlock:(UIViewController * (^)(void))viewControllerFutureBlock {
  37. NSParameterAssert(entryName);
  38. NSParameterAssert(viewControllerFutureBlock);
  39. NSAssert(NSThread.isMainThread, @"This method must be called from the main thread.");
  40. entryName = entryName.copy;
  41. FLEXGlobalsEntry *entry = [FLEXGlobalsEntry entryWithNameFuture:^NSString *{
  42. return entryName;
  43. } viewControllerFuture:^UIViewController *{
  44. UIViewController *viewController = viewControllerFutureBlock();
  45. NSCAssert(viewController, @"'%@' entry returned nil viewController. viewControllerFutureBlock should never return nil.", entryName);
  46. return viewController;
  47. }];
  48. [self.userGlobalEntries addObject:entry];
  49. }
  50. - (void)clearGlobalEntries
  51. {
  52. [self.userGlobalEntries removeAllObjects];
  53. }
  54. #pragma mark - Simulator Shortcuts
  55. - (void)registerSimulatorShortcutWithKey:(NSString *)key modifiers:(UIKeyModifierFlags)modifiers action:(dispatch_block_t)action description:(NSString *)description {
  56. #if TARGET_OS_SIMULATOR
  57. [FLEXKeyboardShortcutManager.sharedManager registerSimulatorShortcutWithKey:key modifiers:modifiers action:action description:description allowOverride:YES];
  58. #endif
  59. }
  60. - (void)setSimulatorShortcutsEnabled:(BOOL)simulatorShortcutsEnabled {
  61. #if TARGET_OS_SIMULATOR
  62. [FLEXKeyboardShortcutManager.sharedManager setEnabled:simulatorShortcutsEnabled];
  63. #endif
  64. }
  65. - (BOOL)simulatorShortcutsEnabled {
  66. #if TARGET_OS_SIMULATOR
  67. return FLEXKeyboardShortcutManager.sharedManager.isEnabled;
  68. #else
  69. return NO;
  70. #endif
  71. }
  72. #pragma mark - Shortcuts Defaults
  73. - (void)registerDefaultSimulatorShortcutWithKey:(NSString *)key modifiers:(UIKeyModifierFlags)modifiers action:(dispatch_block_t)action description:(NSString *)description {
  74. #if TARGET_OS_SIMULATOR
  75. // Don't allow override to avoid changing keys registered by the app
  76. [FLEXKeyboardShortcutManager.sharedManager registerSimulatorShortcutWithKey:key modifiers:modifiers action:action description:description allowOverride:NO];
  77. #endif
  78. }
  79. - (void)registerDefaultSimulatorShortcuts {
  80. [self registerDefaultSimulatorShortcutWithKey:@"f" modifiers:0 action:^{
  81. [self toggleExplorer];
  82. } description:@"Toggle FLEX toolbar"];
  83. [self registerDefaultSimulatorShortcutWithKey:@"g" modifiers:0 action:^{
  84. [self showExplorerIfNeeded];
  85. [self.explorerViewController toggleMenuTool];
  86. } description:@"Toggle FLEX globals menu"];
  87. [self registerDefaultSimulatorShortcutWithKey:@"v" modifiers:0 action:^{
  88. [self showExplorerIfNeeded];
  89. [self.explorerViewController toggleViewsTool];
  90. } description:@"Toggle view hierarchy menu"];
  91. [self registerDefaultSimulatorShortcutWithKey:@"s" modifiers:0 action:^{
  92. [self showExplorerIfNeeded];
  93. [self.explorerViewController toggleSelectTool];
  94. } description:@"Toggle select tool"];
  95. [self registerDefaultSimulatorShortcutWithKey:@"m" modifiers:0 action:^{
  96. [self showExplorerIfNeeded];
  97. [self.explorerViewController toggleMoveTool];
  98. } description:@"Toggle move tool"];
  99. [self registerDefaultSimulatorShortcutWithKey:@"n" modifiers:0 action:^{
  100. [self toggleTopViewControllerOfClass:[FLEXNetworkMITMViewController class]];
  101. } description:@"Toggle network history view"];
  102. // 't' is for testing: quickly present an object explorer for debugging
  103. [self registerDefaultSimulatorShortcutWithKey:@"t" modifiers:0 action:^{
  104. [self showExplorerIfNeeded];
  105. [self.explorerViewController toggleToolWithViewControllerProvider:^UINavigationController *{
  106. return [FLEXNavigationController withRootViewController:[FLEXObjectExplorerFactory
  107. explorerViewControllerForObject:NSBundle.mainBundle
  108. ]];
  109. } completion:nil];
  110. } description:@"Present an object explorer for debugging"];
  111. [self registerDefaultSimulatorShortcutWithKey:UIKeyInputDownArrow modifiers:0 action:^{
  112. if (self.isHidden || ![self.explorerViewController handleDownArrowKeyPressed]) {
  113. [self tryScrollDown];
  114. }
  115. } description:@"Cycle view selection\n\t\tMove view down\n\t\tScroll down"];
  116. [self registerDefaultSimulatorShortcutWithKey:UIKeyInputUpArrow modifiers:0 action:^{
  117. if (self.isHidden || ![self.explorerViewController handleUpArrowKeyPressed]) {
  118. [self tryScrollUp];
  119. }
  120. } description:@"Cycle view selection\n\t\tMove view up\n\t\tScroll up"];
  121. [self registerDefaultSimulatorShortcutWithKey:UIKeyInputRightArrow modifiers:0 action:^{
  122. if (!self.isHidden) {
  123. [self.explorerViewController handleRightArrowKeyPressed];
  124. }
  125. } description:@"Move selected view right"];
  126. [self registerDefaultSimulatorShortcutWithKey:UIKeyInputLeftArrow modifiers:0 action:^{
  127. if (self.isHidden) {
  128. [self tryGoBack];
  129. } else {
  130. [self.explorerViewController handleLeftArrowKeyPressed];
  131. }
  132. } description:@"Move selected view left"];
  133. [self registerDefaultSimulatorShortcutWithKey:@"?" modifiers:0 action:^{
  134. [self toggleTopViewControllerOfClass:[FLEXKeyboardHelpViewController class]];
  135. } description:@"Toggle (this) help menu"];
  136. [self registerDefaultSimulatorShortcutWithKey:UIKeyInputEscape modifiers:0 action:^{
  137. [[self.topViewController presentingViewController] dismissViewControllerAnimated:YES completion:nil];
  138. } description:@"End editing text\n\t\tDismiss top view controller"];
  139. [self registerDefaultSimulatorShortcutWithKey:@"o" modifiers:UIKeyModifierCommand|UIKeyModifierShift action:^{
  140. [self toggleTopViewControllerOfClass:[FLEXFileBrowserController class]];
  141. } description:@"Toggle file browser menu"];
  142. }
  143. + (void)load {
  144. dispatch_async(dispatch_get_main_queue(), ^{
  145. [self.sharedManager registerDefaultSimulatorShortcuts];
  146. });
  147. }
  148. #pragma mark - Private
  149. - (UIEdgeInsets)contentInsetsOfScrollView:(UIScrollView *)scrollView {
  150. if (@available(iOS 11, *)) {
  151. return scrollView.adjustedContentInset;
  152. }
  153. return scrollView.contentInset;
  154. }
  155. - (void)tryScrollDown {
  156. UIScrollView *scrollview = [self firstScrollView];
  157. UIEdgeInsets insets = [self contentInsetsOfScrollView:scrollview];
  158. CGPoint contentOffset = scrollview.contentOffset;
  159. CGFloat maxYOffset = scrollview.contentSize.height - scrollview.bounds.size.height + insets.bottom;
  160. contentOffset.y = MIN(contentOffset.y + 200, maxYOffset);
  161. [scrollview setContentOffset:contentOffset animated:YES];
  162. }
  163. - (void)tryScrollUp {
  164. UIScrollView *scrollview = [self firstScrollView];
  165. UIEdgeInsets insets = [self contentInsetsOfScrollView:scrollview];
  166. CGPoint contentOffset = scrollview.contentOffset;
  167. contentOffset.y = MAX(contentOffset.y - 200, -insets.top);
  168. [scrollview setContentOffset:contentOffset animated:YES];
  169. }
  170. - (UIScrollView *)firstScrollView {
  171. NSMutableArray<UIView *> *views = FLEXUtility.appKeyWindow.subviews.mutableCopy;
  172. UIScrollView *scrollView = nil;
  173. while (views.count > 0) {
  174. UIView *view = views.firstObject;
  175. [views removeObjectAtIndex:0];
  176. if ([view isKindOfClass:[UIScrollView class]]) {
  177. scrollView = (UIScrollView *)view;
  178. break;
  179. } else {
  180. [views addObjectsFromArray:view.subviews];
  181. }
  182. }
  183. return scrollView;
  184. }
  185. - (void)tryGoBack {
  186. UINavigationController *navigationController = nil;
  187. UIViewController *topViewController = self.topViewController;
  188. if ([topViewController isKindOfClass:[UINavigationController class]]) {
  189. navigationController = (UINavigationController *)topViewController;
  190. } else {
  191. navigationController = topViewController.navigationController;
  192. }
  193. [navigationController popViewControllerAnimated:YES];
  194. }
  195. - (UIViewController *)topViewController {
  196. return [FLEXUtility topViewControllerInWindow:UIApplication.sharedApplication.keyWindow];
  197. }
  198. - (void)toggleTopViewControllerOfClass:(Class)class {
  199. UINavigationController *topViewController = (id)self.topViewController;
  200. if ([topViewController isKindOfClass:[FLEXNavigationController class]]) {
  201. if ([topViewController.topViewController isKindOfClass:[class class]]) {
  202. if (topViewController.viewControllers.count == 1) {
  203. // Dismiss since we are already presenting it
  204. [topViewController.presentingViewController dismissViewControllerAnimated:YES completion:nil];
  205. } else {
  206. // Pop since we are viewing it but it's not the only thing on the stack
  207. [topViewController popViewControllerAnimated:YES];
  208. }
  209. } else {
  210. // Push it on the existing navigation stack
  211. [topViewController pushViewController:[class new] animated:YES];
  212. }
  213. } else {
  214. // Present it in an entirely new navigation controller
  215. [self.explorerViewController presentViewController:
  216. [FLEXNavigationController withRootViewController:[class new]]
  217. animated:YES completion:nil];
  218. }
  219. }
  220. - (void)showExplorerIfNeeded {
  221. if (self.isHidden) {
  222. [self showExplorer];
  223. }
  224. }
  225. @end