说明:
项目中我们可能会针对全局的 push,pop,present,dismiss 等视图转场进行操作;
例如:
* 给 push/present 的下一级视图一个返回按钮。
* 给 push/present 的过程中隐藏所有的 tabbar。
* 给 push/present 的时候自定义一个转场动画。
。。。。。针对视图转换之间进行的操作
Push Pop
大家都知道 push 和 pop 操作是有导航控制器来进行控制的,所以这里我们讲一个在 push 界面的时候为下一级的导航控制器进行添加返回按钮和对 tabbar 的隐藏。
1.0 创建 TabbarController 自定义 NavigationController,设置自定义的 NavigationController 为 TabBarController 的 NavigationController
[self setupChildViewController:[[KlqNavViewController alloc] initWithRootViewController:[[KlqOneViewController alloc] init]] title:@"test1" image:@"tabbar_home_icon" selectedImage:@"tabbar_home_select_icon"];
[self setupChildViewController:[[KlqNavViewController alloc] initWithRootViewController:[[KlqTwoViewController alloc] init]] title:@"test2" image:@"tabbar_counters_icon" selectedImage:@"tabbar_counters_select_icon"];
[self setupChildViewController:[[KlqNavViewController alloc] initWithRootViewController:[[KlqThreeViewController alloc] init]] title:@"test4" image:@"tabbar_instr_icon" selectedImage:@"tabbar_instu_select_icon"];
[self setupChildViewController:[[KlqNavViewController alloc] initWithRootViewController:[[KlqFourViewController alloc] init]] title:@"test5" image:@"tabbar_mine_icon" selectedImage:@"tabbar_mine_select_icon"];
在 导航控制器进行 push 操作的时候
- (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated
{
if (self.childViewControllers.count > 0) {
UIButton *backButton = [UIButton buttonWithType:UIButtonTypeCustom];
[backButton setTitle:@"返回" forState:UIControlStateNormal];
[backButton setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
[backButton addTarget:self action:@selector(back) forControlEvents:UIControlEventTouchUpInside];
viewController.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithCustomView:backButton];
viewController.hidesBottomBarWhenPushed = YES;
}
[super pushViewController:viewController animated:YES];//这里进行下简单的说明,放到后面和放到前面是一样子的,但是要注意的是如果放到前面的话,if 判断中的 self.choildViewControllers 就要大于1才行了。
}
- (UIViewController *)popViewControllerAnimated:(BOOL)animated
{
return [super popViewControllerAnimated:YES];
}
通过这两个方法可以拦截到在自定义的 navigationController 导航控制器控制的所有的 viewControlelr 发生的 push 和 pop 方法,这时候就可以在这里面对返回按钮或者自定义push 动画进行操作。
Present Dismiss
说完了 push 和 pop 操作,我们这时候在说下如果获取到全局的 present 和 dismiss 操作,首先要了解的是我们的 present 和 dismiss 操作是不受导航控制器来进行控制的。因此我们不能够在导航控制器中写 present 和 dismiss 操作,我们用到的方法是 hook 编程,即用到 objective 中的Method Swizzling 来获取到 present 和 dismiss 方法,对它们进行转换,从而实现我们的需求。
Method Swizzling 方法的原理,根本原理就是在程序运行期间利用 IMP 和 SEL 动态的给两个方法进行互换。
- IMP -> 就是指向一个方法实现的指针,大家要知道 OC 中方法在编译期间会根据 runtime 中的操作将方法保存到内存中,因此我们调用方法的时候,就需要根据方法的指针来对方法进行调用。
- SEL -> 我们可以看到 objc/runtime.h 文件中的 objc_method 结构体中是这么写的,方法调用的时候会通过SEL找到 objc_method 结构体,从而找到方法的 IMP指针。
struct objc_method {
SEL method_name OBJC2_UNAVAILABLE;
char *method_types OBJC2_UNAVAILABLE;
IMP method_imp OBJC2_UNAVAILABLE;
}
下面我们看代码:这里我们展示的是获取到 present 的操作然后自定义下 present 的动画。
1. 我们这边定义了一个 ViewController 的 Category 来进行对 ViewController 的扩展。从而达到对 ViewController 的所有的方法的扩展。
2. 在 viewController Category 中重写 load 方法
+ (void)load{
[super load];
Method fromMethod = class_getInstanceMethod([self class], @selector(presentViewController:animated:completion:));
Method toMethod = class_getInstanceMethod([self class], @selector(swizzlingpresentViewController:animated:completion:));
if (!class_addMethod([self class], @selector(viewDidLoad), method_getImplementation(toMethod), method_getTypeEncoding(toMethod))) {
method_exchangeImplementations(fromMethod, toMethod);
}
}
3.重定义的 present 方法
- (void)swizzlingpresentViewController:(UIViewController *)viewControllerToPresent animated:(BOOL)flag completion:(void (^)(void))completion
{
//拦截到用户的所有的 present 然后对视图等信息进行修改,调整。
viewControllerToPresent.transitioningDelegate = self;
[self swizzlingpresentViewController:viewControllerToPresent animated:YES completion:nil];
}
4.自定义模态方法 ShowMBTransitionAnimation
#pragma mark - TransitioningDelegate
- (id<UIViewControllerAnimatedTransitioning>)animationControllerForPresentedController:(UIViewController *)presented presentingController:(UIViewController *)presenting sourceController:(UIViewController *)source
{
ShowMBTransitionAnimation *showMBTransition = [ShowMBTransitionAnimation transitionWithType:ShowMBTransitionAnimationPresentType];
showMBTransition.toViewHeight = 500;
return showMBTransition;
}
- (id<UIViewControllerAnimatedTransitioning>)animationControllerForDismissedController:(UIViewController *)dismissed
{
ShowMBTransitionAnimation *dismissMBTransition = [ShowMBTransitionAnimation transitionWithType:ShowMBTransitionAnimationDismissType];
dismissMBTransition.toViewHeight = 500;
return dismissMBTransition;
}