iOS system swipe right to return to the global control solution

iOS system swipe right to return to the global control solution

Preface

Today, there is a small requirement. Before clicking the back button on the navigation bar, we need to call a certain API and pop up a UIAlertView to display it. According to the user's option, we determine whether to return or stay in the current controller. For example, when you click the back button in the upper left corner of the navigation bar, our API is called to prompt whether you know. If you click yes, you return. If you click no, you stay in the current controller.

Then the question is, the navigation's built-in right swipe back gesture will not be handled when clicking the system's back button, it is automatic, so we have to find a way to change it to leftBarButtonItem, but using leftBarButtonItem will eliminate the right swipe back gesture.

You can't have your cake and eat it too? I have a way!

The author tried to write a demo to verify what solutions there are, and tried the following four methods:

  • Only if the current controller complies with UIGestureRecognizerDelegate and sets the delegate to self
  • Put UIGestureRecognizerDelegate in a public base class controller and set the delegate to self, then subclasses override the delegate method
  • Put UIGestureRecognizerDelegate in the public navigation class HYBNavigationController, set the delegate to the navigation class, and then rewrite all push/pop related methods
  • Put UIGestureRecognizerDelegate in the public navigation class HYBNavigationController and set the delegate to the navigation class, but only follow the -gestureRecognizerShouldBegin: delegate method

Option 1 (Not feasible)

Solution 1: Only the current controller complies with UIGestureRecognizerDelegate and sets the delegate to self

Why is it not feasible? How can we know if we don't test it? It's hard to consider it comprehensively just by thinking about it. So I wrote a small demo to test it.

We write this in the controller:

  1. - (void)viewDidLoad {
  2.  
  3. [super viewDidLoad];
  4.  
  5.   
  6.  
  7. UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
  8.  
  9. [button setTitle:@ "return" forState:UIControlStateNormal];
  10.  
  11. [button addTarget:self action :@selector(onBack) forControlEvents:UIControlEventTouchUpInside];
  12.  
  13. [button sizeToFit];
  14.  
  15. [button setTitleColor:[UIColor blueColor] forState:UIControlStateNormal];
  16.  
  17. UIBarButtonItem *btnItem = [[UIBarButtonItem alloc] initWithCustomView:button];
  18.  
  19. self.navigationItem.leftBarButtonItem = btnItem;
  20.  
  21.   
  22.  
  23. //Key lines
  24.  
  25. self.navigationController.interactivePopGestureRecognizer.delegate = self;
  26.  
  27. }

Once the delegate is set to self, the click callback can be implemented by using leftBarButtonItem, and the right swipe gesture is still there.

However, the proxy of self.navigationController, which is the navigation controller object, is modified to a certain controller object. When this controller class is released, the proxy becomes nil, so there is no more right swipe back gesture.

Then someone might think, what if we set the proxy to self in -viewDidAppear: and set the proxy to the original proxy object in -viewDidDisappear:? This is also not possible. When A pushes to B, B pushes to C, and then returns from C, the proxy is no longer the original navigation proxy.

Therefore, this solution is not feasible.

Option 2 (Not feasible)

Solution 2: Put UIGestureRecognizerDelegate in a common base class controller and set the delegate to self, then override the delegate method in the subclass

I tried to put UIGestureRecognizerDelegate in HYBBaseViewControlle, and then implemented the proxy. By default, it returns YES, indicating that it supports right swipe back. If you want a controller not to support right swipe back or to perform some operations before returning, you can do so by overriding this proxy method.

This is possible when there is only one controller. However, when the controller is released, the proxy object becomes nil, so the proxy is for the navigation bar object, not a single controller.

Option 3 (feasible, but complicated)

Solution 3: Put UIGestureRecognizerDelegate in the public navigation class HYBNavigationController, set the delegate to the navigation class, and then rewrite all push/pop related methods.

How to implement it as follows:

  1. //HYBNavigationController.m
  2.  
  3. // NavRightPanGestureDemo
  4.  
  5. //
  6.  
  7. // Created by huangyibiao on 16/2/22.
  8.  
  9. // Copyright © 2016 huangyibiao. All rights reserved.
  10.  
  11. //
  12.  
  13.   
  14.  
  15. #import "HYBNavigationController.h"  
  16.  
  17. #import "HYBBaseViewController.h"  
  18.  
  19.   
  20.  
  21. @interface HYBNavigationController ()
  22.  
  23.   
  24.  
  25. @property (nonatomic, assign) BOOL enableRightGesture;
  26.  
  27.   
  28.  
  29. @ end  
  30.  
  31.   
  32.  
  33. @implementation HYBNavigationController
  34.  
  35.   
  36.  
  37. - (void)viewDidLoad {
  38.  
  39. [super viewDidLoad];
  40.  
  41.   
  42.  
  43. self.enableRightGesture = YES;
  44.  
  45. self.interactivePopGestureRecognizer.delegate = self;
  46.  
  47. }
  48.  
  49.   
  50.  
  51. - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer {
  52.  
  53. return self.enableRightGesture;
  54.  
  55. }
  56.  
  57.   
  58.  
  59. - (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated {
  60.  
  61. if ([viewController isKindOfClass:[HYBBaseViewController class]]) {
  62.  
  63. if ([viewController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) {
  64.  
  65. HYBBaseViewController *vc = (HYBBaseViewController *)viewController;
  66.  
  67. self.enableRightGesture = [vc gestureRecognizerShouldBegin];
  68.  
  69. }
  70.  
  71. }
  72.  
  73.   
  74.  
  75. [super pushViewController:viewController animated:YES];
  76.  
  77. }
  78.  
  79.   
  80.  
  81. - (NSArray *)popToRootViewControllerAnimated:(BOOL)animated {
  82.  
  83. self.enableRightGesture = YES;
  84.  
  85. return [super popToRootViewControllerAnimated:animated];
  86.  
  87. }
  88.  
  89.   
  90.  
  91. - (UIViewController *)popViewControllerAnimated:(BOOL)animated {
  92.  
  93. if ( self.viewControllers.count == 1) {
  94.  
  95. self.enableRightGesture = YES;
  96.  
  97. } else {
  98.  
  99. NSUInteger index = self.viewControllers.count - 2;
  100.  
  101. UIViewController *destinationController = [self.viewControllers objectAtIndex: index ];
  102.  
  103. if ([destinationController isKindOfClass:[HYBBaseViewController class]]) {
  104.  
  105. if ([destinationController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) {
  106.  
  107. HYBBaseViewController *vc = (HYBBaseViewController *)destinationController;
  108.  
  109. self.enableRightGesture = [vc gestureRecognizerShouldBegin];
  110.  
  111. }
  112.  
  113. }
  114.  
  115. }
  116.  
  117.   
  118.  
  119. return [super popViewControllerAnimated:animated];
  120.  
  121. }
  122.  
  123.   
  124.  
  125. - (NSArray *)popToViewController:(UIViewController *)viewController animated:(BOOL)animated {
  126.  
  127. if ( self.viewControllers.count == 1) {
  128.  
  129. self.enableRightGesture = YES;
  130.  
  131. } else {
  132.  
  133. UIViewController *destinationController = viewController;
  134.  
  135. if ([destinationController isKindOfClass:[HYBBaseViewController class]]) {
  136.  
  137. if ([destinationController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) {
  138.  
  139. HYBBaseViewController *vc = (HYBBaseViewController *)destinationController;
  140.  
  141. self.enableRightGesture = [vc gestureRecognizerShouldBegin];
  142.  
  143. }
  144.  
  145. }
  146.  
  147. }
  148.  
  149.   
  150.  
  151. return [super popToViewController:viewController animated:animated];
  152.  
  153. }
  154.  
  155.   
  156.  
  157. @ end  

This is done by rewriting all pop/push related methods and determining whether right swipe is required. Then, we need to make a controller class call our API to determine before swiping right or clicking back, as follows:

  1. #import "HYBBController.h"  
  2.  
  3.   
  4.  
  5. @implementation HYBBController
  6.  
  7.   
  8.  
  9. - (void)viewDidLoad {
  10.  
  11. [super viewDidLoad];
  12.  
  13.   
  14.  
  15. UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
  16.  
  17. [button setTitle:@ "return" forState:UIControlStateNormal];
  18.  
  19. [button addTarget:self action :@selector(onBack) forControlEvents:UIControlEventTouchUpInside];
  20.  
  21. [button sizeToFit];
  22.  
  23. [button setTitleColor:[UIColor blueColor] forState:UIControlStateNormal];
  24.  
  25. UIBarButtonItem *btnItem = [[UIBarButtonItem alloc] initWithCustomView:button];
  26.  
  27. self.navigationItem.leftBarButtonItem = btnItem;
  28.  
  29. }
  30.  
  31.   
  32.  
  33. - (BOOL)gestureRecognizerShouldBegin {
  34.  
  35. [self onBack];
  36.  
  37. return   NO ;
  38.  
  39. }
  40.  
  41.   
  42.  
  43. - (void)onBack {
  44.  
  45. UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@ "Brother Biao's Technology Blog"  
  46.  
  47. message:@ "Do you know what the blog address is?"  
  48.  
  49. delegate:self
  50.  
  51. cancelButtonTitle:@ "Don't know"  
  52.  
  53. otherButtonTitles:@ "Know" , nil];
  54.  
  55. [alertView show];
  56.  
  57. }
  58.  
  59.   
  60.  
  61. #pragma mark - UIAlertViewDelegate
  62.  
  63. - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {
  64.  
  65. if (buttonIndex == 0) {
  66.  
  67.   
  68.  
  69. } else {
  70.  
  71. if ([self.navigationItem.title isEqualToString:@ "VC6" ]) {
  72.  
  73. NSUInteger index = self.navigationController.viewControllers.count - 3;
  74.  
  75. UIViewController *vc = [self.navigationController.viewControllers objectAtIndex: index ];
  76.  
  77. [self.navigationController popToViewController:vc animated:YES];
  78.  
  79. } else {
  80.  
  81. [self.navigationController popViewControllerAnimated:YES];
  82.  
  83. }
  84.  
  85. }
  86.  
  87. }
  88.  
  89.   
  90.  
  91. @ end  

This solution does meet our needs. But is there a simpler solution? Maybe because my eyes are a little sleepy today, I didn't realize the fourth solution when I was researching. When I was preparing to write this article, I read the logic again and found that there is a very simple solution that can meet my needs.

Option 4 (reliable, ***)

Solution 4: Put UIGestureRecognizerDelegate in the public navigation class HYBNavigationController and set the delegate to the navigation class, but only follow the -gestureRecognizerShouldBegin: delegate method.

  1. @interface HYBNavigationController ()
  2.  
  3.   
  4.  
  5. @ end  
  6.  
  7.   
  8.  
  9. @implementation HYBNavigationController
  10.  
  11.   
  12.  
  13. - (void)viewDidLoad {
  14.  
  15. [super viewDidLoad];
  16.  
  17.   
  18.  
  19. self.interactivePopGestureRecognizer.delegate = self;
  20.  
  21. }
  22.  
  23.   
  24.  
  25. - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer {
  26.  
  27. BOOL ok = YES; // By default, right swipe is supported
  28.  
  29. if ([self.topViewController isKindOfClass:[HYBBaseViewController class]]) {
  30.  
  31. if ([self.topViewController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) {
  32.  
  33. HYBBaseViewController *vc = (HYBBaseViewController *)self.topViewController;
  34.  
  35. ok = [vc gestureRecognizerShouldBegin];
  36.  
  37. }
  38.  
  39. }
  40.  
  41.   
  42.  
  43. return ok;
  44.  
  45. }
  46.  
  47.   
  48.  
  49. @ end  

The usage is the same as the third solution. Isn't it very simplified? It seems that the Lantern Festival gave me a gift. I suddenly thought of this method. I have never studied the interactivePopGestureRecognizer attribute before. This attribute is only available after iOS7. Therefore, I can't directly use leftBarButtonItem to handle it in the project, unless that interface does not need to swipe right to return.

Now, everything is clear. It is very simple to use the leftBarButtonItem to call the API in the common base class controller to set it up uniformly. The right swipe back gesture can also be used normally~

What are you waiting for? Try it now!

***

If the project you are using also has such a requirement, why not give it a try! The author provides a demo, so you can download the demo first to see the effect! After many tests, the author believes that this is a feasible solution. If you encounter any problems during use, please give feedback to the author. I also want to understand what the situation is. Of course, we also need to find a solution and make progress together.

<<:  Some methods to detect iOS APP performance

>>:  25 New Android Libraries You Definitely Want to Try in Early 2017

Recommend

User Growth Formula: Pinduoduo’s Growth Game Thinking

Nowadays, we can often see some gamification sett...

How to make an executable plan to attract new users?

What should you do when the KPI indicator is &quo...

Cook: Poor sales in China, iPhone sales lower than expected

According to foreign media reports, Apple's s...

How to master new media marketing? 4 keys and 8 ways!

As one of the new and rapidly emerging industries...

5 Kotlin features Android developers need to know

[51CTO.com Quick Translation] It has always been ...

Google's Fuchsia OS has been launched on the first generation Nest Hub

In May of this year, Google officially confirmed ...

Zhihu Product Analysis Report

This article is for self-study and use, and is a ...

How to achieve growth at low prices? Let you know Xiaomi's business model

From launching MiTalk and MIUI systems to making ...