This question seems to be very popular here on StackOverflow so I thought I would try and give a better answer to help out people starting in the world of iOS like me.
I hope this answer is clear enough for people to understand and that I have not missed anything.
Passing Data Forward
Passing data forward to a view controller from another view controller. You would use this method if you wanted to pass an object/value from one view controller to another view controller that you may be pushing on to a navigation stack.
For this example, we will have
ViewControllerA and ViewControllerB
To pass a
BOOL value from toViewControllerAViewControllerB we would do the following.- in
ViewControllerB.hcreating a property for theBOOL@property (nonatomic, assign) BOOL isSomethingEnabled; - in
ViewControllerAyou need to tell it aboutViewControllerBso use an#import "ViewControllerB.h"Then where you want to load the view eg.didSelectRowAtIndexor some youIBActionneed to set the property in beforeViewControllerByou push it onto nav stack.ViewControllerB *viewControllerB = [[ViewControllerB alloc] initWithNib:@"ViewControllerB" bundle:nil]; viewControllerB.isSomethingEnabled = YES; [self pushViewController:viewControllerB animated:YES];This will setisSomethingEnabledinViewControllerBto valueBOOLYES.
Passing Data Forward using Segues
If you are using Storyboards you are most likely using segues and will need this procedure to pass data forward. This is similar to the above but instead of passing the data before you push the view controller, you use a method called
-(void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender
So to pass a
BOOL from ViewControllerA to ViewControllerB we would do the following:- in
ViewControllerB.hcreate a property for theBOOL@property (nonatomic, assign) BOOL isSomethingEnabled; - in
ViewControllerAyou need to tell it aboutViewControllerBso use an#import "ViewControllerB.h" - Create a the segue from
ViewControllerAtoViewControllerBon the storyboard and give it an identifier, in this example, we'll call it"showDetailSegue" - Next, we need to add the method to that
ViewControllerAis called when any segue is performed, because of this we need to detect which segue was called and then do something. In our example, we will check for"showDetailSegue"and if that's performed we will pass ourBOOLvalue toViewControllerB-(void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender{ if([segue.identifier isEqualToString:@"showDetailSegue"]){ ViewControllerB *controller = (ViewControllerB *)segue.destinationViewController; controller.isSomethingEnabled = YES; } }If you have your views embedded in a navigation controller you need to change the method above slightly to the following-(void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender{ if([segue.identifier isEqualToString:@"showDetailSegue"]){ UINavigationController *navController = (UINavigationController *)segue.destinationViewController; ViewControllerB *controller = (ViewControllerB *)navController.topViewController; controller.isSomethingEnabled = YES; } }This will set inisSomethingEnabledViewControllerBto valueBOOLYES.
Passing Data Back
To pass data back from to
ViewControllerBViewControllerA you need to use Protocols and Delegates or Blocks, the latter can be used as a loosely coupled mechanism for callbacks.
To do this we will make a
ViewControllerA delegate of.ViewControllerB This allows ViewControllerBsending a message back to ViewControllerAenable us to send data back.
For to
ViewControllerA be a delegate of itViewControllerB must conform to ViewControllerB's protocol which we have to specify. This tells whichViewControllerA methods it must implement.- In,
ViewControllerB.hbelow them#import, but above you@interfacespecify the protocol.@class ViewControllerB; @protocol ViewControllerBDelegate <NSObject> - (void)addItemViewController:(ViewControllerB *)controller didFinishEnteringItem:(NSString *)item; @end - next still in the
ViewControllerB.hyou need to setup adelegateproperty and synthesize inViewControllerB.m@property (nonatomic, weak) id <ViewControllerBDelegate> delegate; - In
ViewControllerBwe call a message on thedelegatewhen we pop the view controller.NSString *itemToPassBack = @"Pass this value back to ViewControllerA"; [self.delegate addItemViewController:self didFinishEnteringItem:itemToPassBack]; - That's it for
ViewControllerB. Now inViewControllerA.h, tellViewControllerAto importViewControllerBand conform to its protocol.#import "ViewControllerB.h" @interface ViewControllerA : UIViewController <ViewControllerBDelegate> - In
ViewControllerA.mimplement the following method from our protocol- (void)addItemViewController:(ViewControllerB *)controller didFinishEnteringItem:(NSString *)item { NSLog(@"This was returned from ViewControllerB %@",item); } - Before pushing
viewControllerBto navigation stack we need to tellViewControllerBthatViewControllerAis its delegate, otherwise we will get an error.ViewControllerB *viewControllerB = [[ViewControllerB alloc] initWithNib:@"ViewControllerB" bundle:nil]; viewControllerB.delegate = self [[self navigationController] pushViewController:viewControllerB animated:YES];
NSNotification center It's another way to pass data.
// add observer in controller(s) where you want to receive data
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleDeepLinking:) name:@"handleDeepLinking" object:nil];
-(void) handleDeepLinking:(NSNotification *) notification {
id someObject = notification.object // some custom object that was passed with notification fire.
}
// post notification
id someObject;
[NSNotificationCenter.defaultCenter postNotificationName:@"handleDeepLinking" object:someObject];
Passing Data back from one class to another (A class can be any controller, Network/session manager, UIView subclass or any other class)
Blocks are anonymous functions.
This example passes data from Controller B to Controller A
define a block
@property void(^selectedVoucherBlock)(NSString *); // in ContollerA.h
add block handler (listener) where you need a value (for example you need your API response in ControllerA or you need ContorllerB data on A)
// in ContollerA.m
- (void)viewDidLoad {
[super viewDidLoad];
__unsafe_unretained typeof(self) weakSelf = self;
self.selectedVoucherBlock = ^(NSString *voucher) {
weakSelf->someLabel.text = voucher;
};
}
Go to Controller B
UIStoryboard *storyboard = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
ControllerB *vc = [storyboard instantiateViewControllerWithIdentifier:@"ControllerB"];
vc.sourceVC = self;
[self.navigationController pushViewController:vc animated:NO];
fire block
-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:
(NSIndexPath *)indexPath {
NSString *voucher = vouchersArray[indexPath.row];
if (sourceVC.selectVoucherBlock) {
sourceVC.selectVoucherBlock(voucher);
}
[self.navigationController popToViewController:sourceVC animated:YES];
}