假设我有一个自定义容器视图控制器(MainViewController),其中我执行以下操作:
- (void)viewDidLoad
{
[super viewDidLoad];
HomeViewController *homeVC = [[HomeViewController alloc] initWithNibName:@"HomeViewController" bundle:nil];
[self addChildViewController:homeVC];
[self.view addSubview:homeVC.view];
}
HomeViewController将有一个按钮,例如“go”,按下时需要前进到下一个视图控制器。所以我需要通知MainViewController这个动作。这样做的最佳方式是什么?
我正在使用自定义容器,因为我需要在视图控制器之间进行自定义转换。当按下“go”时,HomeViewController上的一些视图将生成动画,而新视图控制器中的视图将动画到位。
显然我可以给HomeViewController一个MainViewController类型的属性并以这种方式进行调用,但我希望有一个更简洁的容器视图控制器API方法。
答案 0 :(得分:15)
您可以使用委托或阻止;
使用委托
创建协议:
@protocol SomeProtocol <NSObject>
- (void)someAction;
@end
只需在HomeViewController.h中声明一个委托:
id<SomeProtocol> delegate;
然后在MainViewController的viewDidLoad中设置如下:
homeVC.delegate = self;
//some where in MainViewController implement the protocol method
-(void)someAction
{
//do something
}
然后当您按下homeVC中的按钮时,只需调用:
if ([self.delegate respondsToSelector:@selector(someAction)]) {
[self.delegate someAction];
}
使用阻止:
在HomeViewController.h中声明一个块属性:
typedef void (^ActionBlock)();
@property (nonatomic, copy) ActionBlock block;
然后在MainViewController ViewDidLoad:
homeVC.block = ^(){
//do something
};
然后当您按下homeVC中的按钮时,只需调用:
self.block();
答案 1 :(得分:10)
还有另一种方式......
每个视图控制器都有parentViewController
属性,因此使用它可以执行此操作...
在MainViewController
中定义您要执行的操作的方法...
- (void)someMethod:(BOOL)someParam;
然后在HomeViewController
你可以做......
MainViewController* parent = (MainViewController*)[self parentViewController];
[parent someMethod:paramValue];
HTH:)
答案 2 :(得分:5)
这是一种非常常见的模式。通过提供protocol
和default extension,父级将是将处理操作的实际实例。
在Swift 3中:
父视图控制器:
protocol SomeProtocol {
func foo()
}
extension ParentViewController: SomeProtocol {
func foo() {
// Parent handles it
}
}
子视图控制器:
@IBAction func tapGo(_ sender: Any) {
(parent as? SomeProtocol)?.foo()
}