OC -> 获取全局的 Pop 和 Present 操作

说明:
项目中我们可能会针对全局的 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 动态的给两个方法进行互换。

  1. IMP -> 就是指向一个方法实现的指针,大家要知道 OC 中方法在编译期间会根据 runtime 中的操作将方法保存到内存中,因此我们调用方法的时候,就需要根据方法的指针来对方法进行调用。
  2. 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)animationControllerForPresentedController:(UIViewController *)presented presentingController:(UIViewController *)presenting sourceController:(UIViewController *)source
{
    ShowMBTransitionAnimation *showMBTransition = [ShowMBTransitionAnimation transitionWithType:ShowMBTransitionAnimationPresentType];
    showMBTransition.toViewHeight               = 500;
    return showMBTransition;
}

- (id)animationControllerForDismissedController:(UIViewController *)dismissed
{
    ShowMBTransitionAnimation *dismissMBTransition = [ShowMBTransitionAnimation transitionWithType:ShowMBTransitionAnimationDismissType];
    dismissMBTransition.toViewHeight               = 500;
    return dismissMBTransition;
}


你可能感兴趣的:(OC -> 获取全局的 Pop 和 Present 操作)