iOS 7的手势滑动返回功能

  现在使用默认模板创建的iOS App都支持手势返回功能,如果导航栏的返回按钮是自定义的那么则会失效,也可以参考这里手动设置无效。

if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) {
    self.navigationController.interactivePopGestureRecognizer.enabled = NO;
}

  如果是因为自定义导航按钮而导致手势返回失效,那么可以在NavigationController的viewDidLoad函数中添加如下代码:

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view.  

    __weak typeof (self) weakSelf = self;
    if ([self respondsToSelector:@selector(interactivePopGestureRecognizer)]) {
        self.interactivePopGestureRecognizer.delegate = weakSelf;
    }
}

  这样写了以后就可以通过手势滑动返回上一层了,但是如果在push过程中触发手势滑动返回,会导致导航栏崩溃(从日志中可以看出)。针对这个问题,我们需要在pop过程禁用手势滑动返回功能,需要实现协议“UINavigationControllerDelegate”

- (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated
{
    // fix ‘nested pop animation can result in corrupted navigation bar‘
    if ([self respondsToSelector:@selector(interactivePopGestureRecognizer)]) {
        self.interactivePopGestureRecognizer.enabled = NO;
    }  

    [super pushViewController:viewController animated:animated];
} 

- (void)navigationController:(UINavigationController *)navigationController
       didShowViewController:(UIViewController *)viewController
                    animated:(BOOL)animated
{
    if ([navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) {
        navigationController.interactivePopGestureRecognizer.enabled = YES;
    }
}

  除了使用系统默认的动画,还可以使用自定义过渡动画(丰满的文档):

- (id<UIViewControllerAnimatedTransitioning>)navigationController:(UINavigationController *)navigationController
                                  animationControllerForOperation:(UINavigationControllerOperation)operation
                                               fromViewController:(UIViewController *)fromVC
                                                 toViewController:(UIViewController *)toVC
{
    if (operation == UINavigationControllerOperationPop) {
        if (self.popAnimator == nil) {
            self.popAnimator = [WQPopAnimator new];
        }
        return self.popAnimator;
    }  

    return nil;
}  

- (id<UIViewControllerInteractiveTransitioning>)navigationController:(UINavigationController *)navigationController
                         interactionControllerForAnimationController:(id<UIViewControllerAnimatedTransitioning>)animationController
{
    return self.popInteractionController;
}  

#pragma mark -   

- (void)enablePanToPopForNavigationController:(UINavigationController *)navigationController
{
    UIScreenEdgePanGestureRecognizer *left2rightSwipe = [[UIScreenEdgePanGestureRecognizer alloc]
                                                         initWithTarget:self
                                                         action:@selector(didPanToPop:)];
    //[left2rightSwipe setDelegate:self];
    [left2rightSwipe setEdges:UIRectEdgeLeft];
    [navigationController.view addGestureRecognizer:left2rightSwipe];  

    self.popAnimator = [WQPopAnimator new];
    self.supportPan2Pop = YES;
}  

- (void)didPanToPop:(UIPanGestureRecognizer *)panGesture
{
    if (!self.supportPan2Pop) return ;  

    UIView *view = self.navigationController.view;  

    if (panGesture.state == UIGestureRecognizerStateBegan) {
        self.popInteractionController = [UIPercentDrivenInteractiveTransition new];
        [self.navigationController popViewControllerAnimated:YES];
    } else if (panGesture.state == UIGestureRecognizerStateChanged) {
        CGPoint translation = [panGesture translationInView:view];
        CGFloat d = fabs(translation.x / CGRectGetWidth(view.bounds));
        [self.popInteractionController updateInteractiveTransition:d];
    } else if (panGesture.state == UIGestureRecognizerStateEnded) {
        if ([panGesture velocityInView:view].x > 0) {
            [self.popInteractionController finishInteractiveTransition];
        } else {
            [self.popInteractionController cancelInteractiveTransition];
        }
        self.popInteractionController = nil;
    }
}

  如下这个代理方法是用来提供一个非交互式的过渡动画的:

- (id<UIViewControllerAnimatedTransitioning>)navigationController:(UINavigationController *)navigationController
                                  animationControllerForOperation:(UINavigationControllerOperation)operation
                                               fromViewController:(UIViewController *)fromVC
                                                 toViewController:(UIViewController *)toVC
{
    if (operation == UINavigationControllerOperationPop) {
        if (self.popAnimator == nil) {
            self.popAnimator = [WQPopAnimator new];
        }
        return self.popAnimator;
    }  

    return nil;
}

  而下面这个代理方法则是提供交互式动画:

- (id<UIViewControllerInteractiveTransitioning>)navigationController:(UINavigationController *)navigationController
                         interactionControllerForAnimationController:(id<UIViewControllerAnimatedTransitioning>)animationController
{
    return self.popInteractionController;
}

  这两个组合起来使用。首先,我们需要有个动画:

@interface WQPopAnimator : NSObject <UIViewControllerAnimatedTransitioning>  

@end

  

#import "WQPopAnimator.h"  

@implementation WQPopAnimator  

- (NSTimeInterval)transitionDuration:(id <UIViewControllerContextTransitioning>)transitionContext
{
    return 0.4;
}  

- (void)animateTransition:(id<UIViewControllerContextTransitioning>)transitionContext
{
    UIViewController *toViewController = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];
    UIViewController *fromViewController = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];
    [[transitionContext containerView] insertSubview:toViewController.view belowSubview:fromViewController.view];  

    __block CGRect toRect = toViewController.view.frame;
    CGFloat originX = toRect.origin.x;
    toRect.origin.x -= toRect.size.width / 3;
    toViewController.view.frame = toRect;  

    [UIView animateWithDuration:[self transitionDuration:transitionContext] animations:^{
        CGRect fromRect = fromViewController.view.frame;
        fromRect.origin.x += fromRect.size.width;
        fromViewController.view.frame = fromRect;  

        toRect.origin.x = originX;
        toViewController.view.frame = toRect;
    } completion:^(BOOL finished) {
        [transitionContext completeTransition:![transitionContext transitionWasCancelled]];
    }];
}  

@end

  其次,交互式动画是通过

UIPercentDrivenInteractiveTransition

  来维护的,在滑动过程中根据滑动距离来进行更新:

} else if (panGesture.state == UIGestureRecognizerStateChanged) {
        CGPoint translation = [panGesture translationInView:view];
        CGFloat d = fabs(translation.x / CGRectGetWidth(view.bounds));
        [self.popInteractionController updateInteractiveTransition:d];

  当手势结束时要做出收尾动作:

} else if (panGesture.state == UIGestureRecognizerStateEnded) {
        if ([panGesture velocityInView:view].x > 0) {
            [self.popInteractionController finishInteractiveTransition];
        } else {
            [self.popInteractionController cancelInteractiveTransition];
        }
        self.popInteractionController = nil;
    }

  同样地,自定义的动画也会有上面提到的导航栏崩溃问题,也可以通过类似的方法来解决:

- (void)navigationController:(UINavigationController *)navigationController
       didShowViewController:(UIViewController *)viewController
                    animated:(BOOL)animated
{
    if (viewController == self.navigationController.pushingViewController) {
        self.supportPan2Pop = YES;
        self.navigationController.pushingViewController = nil;
    }

  补充:位于当前navgationController的第一个([0])viewController时需要设置手势代理,不响应。

原文地址:http://blog.csdn.net/jasonblog/article/details/28282147

  

时间: 2024-11-25 04:37:02

iOS 7的手势滑动返回功能的相关文章

iOS 7的手势滑动返回

如今使用默认模板创建的iOS App都支持手势返回功能,假设导航栏的返回button是自己定义的那么则会失效,也能够參考这里手动设置无效. [cpp] view plaincopy if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) { self.navigationController.interactivePopGestureRecognizer.enab

iOS之手势滑动返回功能-b

iOS中如果不自定义UINavigationBar,通过手势向右滑是可以实现返回的,这时左边的标题文字提示的是上一个ViewController的标题,如果需要把文字改为简约风格,例如弄过箭头返回啥的,那么你需要自定义UINavigationBar,但当你自定义navigationBar后,这个功能就会自动失效. 屏蔽右滑返回功能代码:   if ([self.navigationController respondsToSelector:@selector(interactivePopGest

iOS彩票项目--第五天,新特性引导页的封装、返回按钮的自定义、导航控制器的滑动返回以及自定义滑动返回功能

一.上次实现了在AppDelegate中通过判断app版本决定是否进入新特性页面,今天将AppDelegate中的一坨进行了封装.将self.window的根控制器到底应该为新特性界面,还是主页面,封装到了导航工具类ChaosGuideTool 封装,先决定外面怎么用,然后实现方法.外部通过类方法调用  + (UIViewController *)chooseRootVC; 外部的APPDelegate 只是调用方法 之前的业务判断没有改变,只是将数据的存储进行了封装 二.返回按钮的自定义 <1

滑动返回功能被覆盖的解决思路

在OC开发中,导航控制器是一个非常常见的控件,而且在不少使用时候,我们需要自定义导航条NavigationBar.但是这个做法可能带来一些小麻烦,下面是我遇到的问题以及解决方案的思维过程. 在苹果内部,返回功能的实现自带了一个边缘的滑动返回功能.但是一旦使用我们自定义的NavigationBar,子控制器这个功能便会消失.如果我们既要用到自定义,又要保留滑动返回功能,那我们首先要分析消失的原因,再来寻找解决方案. 既然替换NavigationBar会导致滑动返回功能的消失,那我们基本可以确定,在

Android-通过SlidingMenu高仿微信6.2最新版手势滑动返回(二)

转载请标明出处: http://blog.csdn.net/hanhailong726188/article/details/46453627 本文出自:[海龙的博客] 一.概述 在上一篇博文中,博文地址Android-通过SlidingPaneLayout高仿微信6.2最新版手势滑动返回(一),我们是通过官方自带的SlidingPaneLayout来实现的手势滑动返回.在这篇博文中,我们将採用SlidingMenu来高仿. 事实上实现的原理都一样.仅仅只是是把SlidingPaneLayout

Android-通过SlidingPaneLayout高仿微信6.2最新版手势滑动返回(一)

最近更新了微信版本到6.2,发现里面有个非常好的体验,就是在第二个页面Activity能手势向右滑动返回,在手势滑动的过程中能看到第一个页面,这种体验非常赞,这里高仿了一下.这里使用的是v4包里面的SlidingPaneLayout来手势滑动,在下一篇博文中将采用SlidingMenu来高仿,下面是SlidingPaneLayout高仿后的效果,效果还是蛮不错的. 最重要的是,每一个页面都是Activity,而非Fragment哦,使用Activity和正常的Activity一样 这里给出dem

ios7去除手势滑动返回

if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) { self.navigationController.interactivePopGestureRecognizer.enabled = NO; } ios7去除手势滑动返回

滑动返回的延伸(全局滑动返回功能)

上一篇文章提到自定义导航条(NavigationBar)覆盖系统返回按钮,导致滑动功能消失的解决方法后,有朋友提出有时候会遇到一些别的要求,比如要设置一个全屏滑动返回的功能,顾名思义在屏幕任何位置都能实现右滑返回. 其实如果滑动功能的解决思路和过程,这个方案并不难实现,或者说比解决滑动功能消失更简单.首先我们可以先解决以下问题:为什么系统手势不能全屏?那我们可以把手势属性打印出来分析. Edge是边缘的意思,那很有可能系统用这个方法只能控制边缘的滑动返回,我们可以点进去进一步观察. 可见 UIS

iOS开发—向右滑动手势实现返回

在navigationController中实现向右滑动返回功能 系统提供的backbarbuttonitem,不用添加任何代码即可实现向右滑动后退功能,但是往往要对按钮修改样式等时,就需要自定义leftbarbuttonitem,此时向右滑动即失效.通过下面方法即可解决该问题. 主要是通过设置navigationController.interactivePopGestureRecognizer UIGestureRecognizerDelegate 1. self.navigationCon