Dismissing Modal and Current UIViewControllers in the Same Delegate
[banner id=”salesfolio” url=”/sales_folio.pdf”]
Sometimes in iOS, something that seems like it should be easy to do can, be a bit more complicated than you would expect or can be difficult to remember since you don’t do it very often. This is often the case for me when I have a view controller that presents another view controller modally and then during the delegate callback for the modal, I want to not only dismiss the modal itself, but the current view controller that presented it to begin with. This post will simply describe how I normally accomplish this task.
First, I present the modal view controller to do whatever task I need it to do. In this example, I am doing a signature capture:
SignatureViewController *sigViewController = [[SignatureViewController alloc] init];
sigViewController.delegate = self;
sigViewController.modalPresentationStyle = UIModalPresentationFullScreen;
sigViewController.modalTransitionStyle = UIModalTransitionStyleCoverVertical;
[self presentViewController:sigViewController animated:YES completion:NULL];
Then after I collect the signature in the SignatureViewController, I call the appropriate delegate in the original view controller:
if (delegate && [delegate respondsToSelector:@selector(signatureController:signatureAsBase64:savePressed:)]) {
NSString *signature = [signaturePad getSignatureAsBase64];
[delegate signatureController:self signatureAsBase64:signature savePressed:sender];
}
All pretty standard fare. Now things get just a little different than normal in the delegate. For this example we will go back to the second view controller in our navigation controller, the first being a login screen that we don’t want the user to repeat.
// It's *VERY* important not to animate this. Otherwise it causes problems with
// popping back to the lookup screen. Causes the views to try to
// lay themselves out while being dismissed
__block OrderPickDetailViewController *me = self;
[self dismissViewControllerAnimated:NO completion:^{
UIViewController *popTo = nil;
NSArray *viewControllers = [me.navigationController viewControllers];
if (viewControllers && [viewControllers count] > 1) {
popTo = [viewControllers objectAtIndex:1];
}
if (popTo) {
[me.navigationController popToViewController:popTo animated:YES];
} else {
[me.navigationController popToRootViewControllerAnimated:YES];
}
}];
The important things to note here are:
- Do not animate the dismissal of the modal. Mixing that animation with the dismissal of the current view controller will cause the view to try to lay itself out while it is being released.
- We use the completion block of the modal dismissal to dismiss the current view controller.
- I’m holding a block level reference to the current view controller to make sure that I have access to it during the entire time I am setting up the navigation away from the current view controller. As soon as the block goes out of scope, ARC reclaims it and the view controller is released, so we do not have a retain loop.
That’s all there is to it. Generally the biggest thing to remember in these types of situations is not to have two animations going at the same time and to sequence things logically if you can.
Hey,
Thanks for the detailed explanation. I’m trying to do the same thing and it works, but for reason a gap appears after the navigationBar in the rootViewController.
It’s too long to explain it on here. I have posted the question on StackOverFlow. Thanks!
http://stackoverflow.com/questions/17790946/ios-containerview-with-modalview-gap-appears-after-dismissing-modal