11

I want to perform a action when the user pressed the back button on my UINavigationController when arrived at a certain UIViewController.

Unfortunately it looks like UINavigationControllerDelegate doesn't have any methods to get notified of the popping of views.

As a workaround I now have in the viewDidDisappear method my action, that only gets fired when animated is YES. This works, but it's a bit ugly.

How should I do this properly?

Vadim Kotov
  • 8,084
  • 8
  • 48
  • 62
Peterdk
  • 15,625
  • 20
  • 101
  • 140

5 Answers5

30

The most popular way of handling a pop from navigation view controller (as well as from modal) is implementing viewWillDisappear for your view controller.

- (void)viewWillDisappear:(BOOL)animated {
    [super viewWillDisappear:animated];

    if (self.isMovingFromParentViewController || self.isBeingDismissed) {
        // This view controller is being popped or dismissed
    }
}
samwize
  • 25,675
  • 15
  • 141
  • 186
5

If you have a reference to the controller down the stack, the one which will show when this one is popped, you can register as a delegate and check it through

navigationController:willShowViewController:animated:
Bo Persson
  • 90,663
  • 31
  • 146
  • 203
Max_B
  • 59
  • 1
  • 2
0

you can observe to the UINavigationControllerDelegate and check if transition will happened:

- (void)navigationController:(UINavigationController *)navigationController
          willShowViewController:(UIViewController *)viewController
                        animated:(BOOL)animated
    {
        if([navigationController.viewControllers containsObject:self])
        {
            NSLog(@"push");
        }
        else
        {
            NSLog(@"pop");
        }
    }
Ron Fridman
  • 304
  • 1
  • 6
  • 1
    If you have a UITabBarController, you should use this solution because the viewWillDisappear is call each time you change the current tab index. – FouZ Jun 13 '17 at 17:41
0

You can either call a delegate method when viewWillDisappear or set logic on viewWillAppear for certain UIViewController.

alloc_iNit
  • 5,173
  • 2
  • 26
  • 54
0

First you need to conform UINavigationControllerDelegate to navigation controller and implement this method

 public func navigationController(
    _ navigationController: UINavigationController,
    didShow viewController: UIViewController,
    animated: Bool) {
    guard let dismissedViewController =
            navigationController.transitionCoordinator?
            .viewController(forKey: .from),
          !navigationController.viewControllers
            .contains(dismissedViewController) else {
        return
    }
    performOnDismissed(for: dismissedViewController)
}

In performOnDismissed func, you can check dismissedViewController, like

if (dismissedViewController is DesireViewController)

then, fire your method.

Shahriar Nasim Nafi
  • 1,160
  • 15
  • 19