Passing Data between View Controllers :
Passing Data ForwardPassing 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 ViewControllerBTo pass a
BOOL value from ViewControllerA to ViewControllerB we would do the following.- in
ViewControllerB.hcreate a property for theBOOL
@property(nonatomic) BOOL *isSomethingEnabled; - in
ViewControllerAyou need to tell it aboutViewControllerBso use an
#import "ViewControllerB.h"
Then where you want to load the view eg.didSelectRowAtIndexor someIBActionyou need to set the property inViewControllerBbefore you push it onto nav stack.
ViewControllerB *viewControllerB = [[ViewControllerB alloc] initWithNib:@"ViewControllerB" bundle:nil]; viewControllerB.isSomethingEnabled = YES; [self pushViewController:viewControllerB animated:YES];
This will setisSomethingEnabledinViewControllerBtoBOOLvalueYES.
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) 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
ViewControllerAthat is 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 thats 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 setisSomethingEnabledinViewControllerBtoBOOLvalueYES.
Passing Data Back
To pass data back from
ViewControllerB to ViewControllerA 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
ViewControllerA a delegate of ViewControllerB. This allows ViewControllerB to send a message back to ViewControllerA enabling us to send data back.For
ViewControllerA to be delegate of ViewControllerB it must conform to ViewControllerB's protocol which we have to specify. This tells ViewControllerA which methods it must implement.- In
ViewControllerB.h, below the#import, but above@interfaceyou specify 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); } - The last thing we need to do is tell
ViewControllerBthatViewControllerAis its delegate before we pushViewControllerBon to nav stack.
ViewControllerB *viewControllerB = [[ViewControllerB alloc] initWithNib:@"ViewControllerB" bundle:nil]; viewControllerB.delegate = self [[self navigationController] pushViewController:viewControllerB animated:YES];
References
- Using Delegation to Communicate With Other View Controllers in the View Controller Programming Guide
- Delegate Pattern
No comments:
Post a Comment