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

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

屏蔽右滑返回功能代码:

 

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

开启滑动返回功能代码:

 

  1. - (void)viewWillAppear:(BOOL)animated{
  2. [super viewWillAppear:animated];
  3. // 右滑返回
  4. if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) {
  5. self.navigationController.interactivePopGestureRecognizer.delegate = nil;
  6. }
  7. }

注意各种坑:

"在一级视图中,iOS样式返回的手势滑动一下,然后进入二级视图,发现画面卡住了,按Home键转入后台,再返回应用,发现并没有Crash掉,而是直接跳到了二级视图里,运行正常了,大家知道push和pop的原理是用进栈出栈完成的,可能因为在一级视图中滑动那一下,影响了视图在栈中的位置。 "

------有人提到通过以下方法处理:“一级视图中一定要加入self.navigationController.interactivePopGestureRecognizer.enabled = NO;,先把iOS7手势返回屏蔽掉,到二级视图再用self.navigationController.interactivePopGestureRecognizer.enabled = YES打开”

自己写了个demo试运行,发现self.navigationController.interactivePopGestureRecognizer.enabled 不能动态设置更改状态。因此该方法不可行。

解决方法:

  1. - (void)viewDidAppear:(BOOL)animated
  2. {
  3. __weak typeof(self) weakSelf = self;
  4. self.navigationController.interactivePopGestureRecognizer.delegate = weakSelf;
  5. }

实现手势协议:

 

  1. #pragma mark - UIGestureRecognizerDelegate
  2. - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer*)gestureRecognizer{
  3. //判断是否为rootViewController
  4. if (self.navigationController && self.navigationController.viewControllers.count == 1) {
  5. return NO;
  6. }
  7. return YES;
  8. }

但问题又来了,如果是一个显示成功/失败结果页,滑动返回不大符合正常思维,因为需要选择性屏蔽处理。

终极解决方法:自定义全屏滑动手势UIPanGestureRecognizer

    1. //
    2. //  BasicNavigationController.m
    3. //
    4. //
    5. //  Copyright (c) 2016年 [email protected] All rights reserved.
    6. //
    7. #import "BasicNavigationController.h"
    8. #import "BaseResultViewController.h"
    9. @interface BasicNavigationController() <UIGestureRecognizerDelegate>
    10. @end
    11. @implementation BasicNavigationController
    12. - (void)viewDidLoad
    13. {
    14. [super viewDidLoad];
    15. [self.navigationBar setTitleTextAttributes:@{NSForegroundColorAttributeName : WhiteColor}];
    16. // 获取系统自带滑动手势的target对象
    17. id target = self.interactivePopGestureRecognizer.delegate;
    18. // 创建全屏滑动手势,调用系统自带滑动手势的target的action方法
    19. UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)];
    20. // 设置手势代理,拦截手势触发
    21. pan.delegate = self;
    22. // 给导航控制器的view添加全屏滑动手势
    23. [self.view addGestureRecognizer:pan];
    24. // 禁止使用系统自带的滑动手势
    25. self.interactivePopGestureRecognizer.enabled = NO;
    26. }
    27. - (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated
    28. {
    29. [viewController.navigationItem.backBarButtonItem setTitleTextAttributes:@{NSFontAttributeName : [UIFont systemFontOfSize:Scale_Size_Smaller()]} forState:UIControlStateNormal];
    30. if (self.childViewControllers.count > 0) {
    31. viewController.hidesBottomBarWhenPushed = YES;
    32. }
    33. [super pushViewController:viewController animated:YES];
    34. }
    35. - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer
    36. {
    37. // 注意:只有非根控制器才有滑动返回功能,根控制器没有。
    38. // 判断导航控制器是否只有一个子控制器,如果只有一个子控制器,肯定是根控制器
    39. if (self.childViewControllers.count == 1) {
    40. // 表示用户在根控制器界面,就不需要触发滑动手势,
    41. return NO;
    42. }
    43. // 当前页面是显示结果页,不响应滑动手势
    44. UIViewController *vc = [self.childViewControllers lastObject];
    45. if ([vc isKindOfClass:[BaseResultViewController class]]) {
    46. return NO;
    47. }
    48. return YES;
    49. }
    50. @end

===============================

iOS自定义导航条(页面手势返回时 导航条更随控制器一起滑动)

前段时间被问到一个问题就是一个JD的UI效果如图

IMG_0485.PNG

其实就是一个原生的导航条 一个是自定义的导航条
第一想法 肯定是 将原生的navgationBar 隐藏掉 让后搞一个UIview 覆盖到原来的位置, 然后在UIview 的上面分别添加左button 和右button 和中间的titleview 当然这样做汪全是可以的 但是我感觉那么 navgationBar 使用的时候就不够方便了

所以 我们可以单独创建一个NavgationBar 然后添加到视图控制器上

那么首先我们创建一个基类 让其继承与UIviewcontroler 并且项目中的其他项目全部继承与这个基类

#import <UIKit/UIKit.h>
#import "ZJNavgationbar.h"
@interface ZJbaseViewController : UIViewController

@property (nonatomic,strong)UINavigationBar * NavigationBar;
@property (nonatomic,strong)UINavigationItem *ZJNavigationItem;

@end

将UINavigationBar定义成基类控制器的公开属性

- (void)viewWillAppear:(BOOL)animated{
    [super viewWillAppear:animated];

    if (!self.NavigationBar) {
        self.NavigationBar = [[UINavigationBar alloc]initWithFrame:CGRectMake(0,0,self.view.frame.size.width,64)];
        [self.view addSubview:self.NavigationBar];
        self.ZJNavigationItem = [[UINavigationItem alloc]initWithTitle:@"hellows"];
        [self.NavigationBar pushNavigationItem:self.ZJNavigationItem animated:YES];
        UIBarButtonItem * logeItem = [[UIBarButtonItem alloc]initWithTitle:@"登录" style:UIBarButtonItemStylePlain target:self action:@selector(logeClick)];
        NSArray * arr = @[logeItem];
                self.ZJNavigationItem.rightBarButtonItems = arr;
        if ([self.navigationController.childViewControllers count]>1) {

        UIBarButtonItem * backItem = [[UIBarButtonItem alloc]initWithTitle:@"返回" style:UIBarButtonItemStylePlain target:self action:@selector(backclick)];
        NSArray * arrleft = @[backItem];
        self.ZJNavigationItem.leftBarButtonItems = arrleft;
        }
    }

}

在控制器的 viewWillAppear 中判断是否存在 NavgationBar 存在的话就不创建 然则反之.
并且把 navgationbar 添加在 控制器的self.view 上
那么 说有集成于基类的控制器 都会拥有这个导航栏

2.导航栏添加名称
首先分析 原生的导航栏添加标题

self.navigationItem.title = @"名称";

我们可以看出来 标题时添加在navgationitem 上 而且 navgationitem 是控制器的属性 而且是只读的 这时候在通过这个属性赋值的话 就会失效

那么为了方便 我们应该同事也自定义一个UINavigationItem
同时也将 属性暴漏出来 如上述代码
这时候呢 那么 添加左右button 也就可以和原来一样 而且可以实现上图的效果.
但是还有一点问题就是 如果控制器是带xib 文件的话 那么导航栏的宽度会比控制器款

F9B1A4A9-0D36-455A-88F0-D29EF472AAB5.png

正常的控制器效果是这样的

6495EAEE-55FD-47AB-B5C7-A29CE41795BF.png

大概原因因该是我现在了 viewwillapper 中的缘故
有啥好的解决办法 可以告诉我哦

以后还会更加深入的解析

感觉好像忘了点啥 那就是 让界面允许全屏滑动返回
很简单只要一句代码搞定
在基类的viewvillappear中添加这一句话 这是kvc 的方法

 [self.navigationController.interactivePopGestureRecognizer setValue:@([UIScreen mainScreen].bounds.size.width) forKeyPath:@"_recognizer._settings._edgeSettings._edgeRegionSize"];

效果如下

好吧 我不会传动态图

今天看了一下昨天遗留下的问题 就是在xib 继承基类是 导航条宽度比屏幕要宽,现在解决的
就是在创建 navgationBar 是frame的宽 不能写 self.view.bounds,size.width
应该写成

 self.NavigationBar = [[UINavigationBar alloc]initWithFrame:CGRectMake(0,0,[UIScreen mainScreen].bounds.size.width,64)];

不上图了 哈哈哈哈
这些问题都解决了 还有一个就是 之所以自定义导航栏有一个原因是 不同的iOS系统版本 导航栏格式不同 自定义是为了统一格式. 很显然 我一上的自定义办法不太好 达不到这样的效果啊

========================

系统自带的向右滑动手势返回上一个界面,ios7--手势

当从控制器A push到控制器B,我们返回控制器A,除了使用按钮返回

[self.navigationController pushViewController:Vc animated:YES];

还可以使用ios7出来的向右滑动,返回控制器A

文档中是这样定义的:

@property(nullable, nonatomic, weak) id<UINavigationControllerDelegate> delegate;

@property(nullable, nonatomic, readonly) UIGestureRecognizer *interactivePopGestureRecognizer NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED;

----------------------------------------------------------------------

我们在控制器B中的viewDidLoad中

if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 7.0) {
        self.navigationController.interactivePopGestureRecognizer.enabled = YES;      // 手势有效设置为YES  无效为NO
        self.navigationController.interactivePopGestureRecognizer.delegate = self;    // 手势的代理设置为self
}

但是当回到控制器A中时,再想push到控制器B,就会出现卡屏,不会动的现象,因为rootView也会有向右滑动返回的问题

要解决这个问题,我们只需在控制器A的viewDidAppear中设置,interactivePopGestureRecognizer为NO:

-(void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:animated];
    if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 7.0) {
        self.navigationController.interactivePopGestureRecognizer.enabled = NO;
    }

}

这样即可以保证再B中向右滑返回A动后再次pushB时不会卡在A界面。

时间: 2024-08-06 16:05:00

iOS之手势滑动返回功能-b的相关文章

iOS 7的手势滑动返回功能

现在使用默认模板创建的iOS App都支持手势返回功能,如果导航栏的返回按钮是自定义的那么则会失效,也可以参考这里手动设置无效. if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) { self.navigationController.interactivePopGestureRecognizer.enabled = NO; } 如果是因为自定义导航按钮而导

iOS 7的手势滑动返回

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

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 禁止手势滑动翻页

- (void)viewDidAppear:(BOOL)animated{ [super viewDidAppear:animated]; // 禁用 iOS7 返回手势 if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) { self.navigationController.interactivePopGestureRecognizer.enabled =