导航控制器左滑返回(下)

  • 上一部分我们只是解决了在自定义了左侧返回按钮的时候覆盖了原有按钮导致左滑返回功能丢失的解决办法,有时候我们觉得自带的左滑返回功能所监听的手势范围太小,我们可以自定义一个手势来做响应的操作。首先让我们来看一下这是如何实现的。
  • 首先让我们看看系统这个interactivePopGesttureRecognizer手势干了啥
interactivePopGestureRecognizer.png
// _UINavigationInteractiveTransition原来是通过这个target 调用了handleNavigationTransition这个action
  • 当我们进入这个类的头头文件的时候发现只有一个属性
@property (readwrite, nonatomic, assign) UIRectEdge edges; //< The edges on which this gesture recognizes, relative to the current interface orientation

// 这个枚举的具体属性是
typedef NS_OPTIONS(NSUInteger, UIRectEdge) {
    UIRectEdgeNone   = 0,
    UIRectEdgeTop    = 1 << 0,
    UIRectEdgeLeft   = 1 << 1,
    UIRectEdgeBottom = 1 << 2,
    UIRectEdgeRight  = 1 << 3,
    UIRectEdgeAll    = UIRectEdgeTop | UIRectEdgeLeft | UIRectEdgeBottom | UIRectEdgeRight
} NS_ENUM_AVAILABLE_IOS(7_0);

// 这个时候我们来打印系统的这个值
2016-03-08 22:25:55.130 57 - 彩票竞技场[18827:319119] edges ==== 2
  • 为2,UIRectEdgeLeft,原来是根据这个值将这个手势固定在左边了
  • 这个时候突发奇想,我们可不可以把系统的这个手势禁用了,自定义一个手势用这个target调用这个action呢
  • 可是我们找遍了头文件和父类也没有找到这个类,估计他是影藏的API,我们只能通过运行时来尝试着找到他
- (void)viewDidLoad
{
    self.delegate = self;

    // 保存手势代理
    self.gestureDelegate = self.interactivePopGestureRecognizer.delegate;

    unsigned int count = 0;
    Ivar *ivars = class_copyIvarList([UIGestureRecognizer class], &count);

    for (int i = 0; i < count; i++) {
        // 获取成员变量
        Ivar ivar = ivars[i];

        // 获取变量名
       NSString *ivarName =  @(ivar_getName(ivar));

        NSLog(@"%@",ivarName);
    }
}
打印结果.png
  • 茫茫多的结果,一页都装不下,但是请我们看这个,我们找到这个target了,现在我们通过kvc来看下这个值
    id objc = [self.interactivePopGestureRecognizer valueForKeyPath:@"_targets"];
    NSLog(@"%@",objc);
  • 妈呀,原来是一个只有一个数据的数组,我们再通过数组的第0个角标再取一次,发现还是开始那个结果,看来重写过了discription方法,不过我们可以通过断点调试找到他,现在我们再通过这个key来取他的值
    NSArray *targets = [self.interactivePopGestureRecognizer valueForKeyPath:@"_targets"];
    id target = [targets firstObject];
    id objc = [target valueForKeyPath:@"_target"];
    NSLog(@"%@",objc);
  • 现在我们拿到这个对象了,可以愉快的调用原有的方法了
    // 取消原有手势防治冲突
    self.interactivePopGestureRecognizer.enabled = NO;

    NSArray *targets = [self.interactivePopGestureRecognizer valueForKeyPath:@"_targets"];
    id target = [targets firstObject];
    id objc = [target valueForKeyPath:@"_target"];

    UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:objc action:@selector(handleNavigationTransition:)];

    [self.view addGestureRecognizer:pan];
  • 然后现在我们的就可以愉快的随意滚动了,但是别开心,我们还是会出现我们上一篇说的bug,因为本来原有的手势的代理会做一些事情,我们可以实现自定义手势的代理的是否触发的方法,在里面判断在非根控制器才让手势生效.
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer
{
    // 如果当前控制器不是根控制器才让手势可用
    return self.topViewController != self.childViewControllers[0];
}
  • 这样才大功告成,最后最后其实我们打印一下这个代理
    NSLog(@"%@",self.interactivePopGestureRecognizer.delegate);
  • 发现原来这个代理就是这个target。OTZ。。。。。轻拍~
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容