美文网首页导航栏IOS
iOS关于自定义返回按钮

iOS关于自定义返回按钮

作者: 独孤红雨 | 来源:发表于2018-03-06 14:55 被阅读0次

    前言:

    就在上个月也就是年末的时候,打开了以前的一个项目,忽然发现导航栏返回按钮几乎看不到了,一直忙另外一个项目,直到这几天才处理一下,总结一下遇到的一些问题和处理方法。

    问题截图如下:

    返回按钮错位截图

    原来没有任何问题,但是iOS11更新之后就这样了,运行下之后发现,下面打印了一堆,仔细一看好像是说约束重复的问题,网上搜索下,目前没找到原因(可能是iOS11系统在导航栏里面的布局和控件都变化了)。下面说一些自定义返回按钮的一些方法总结,以及最后如何解决这个问题。

    代码如下:

    UIImage *backButtonImage = [[UIImage imageNamed:@"箭头left40x40.png"] resizableImageWithCapInsets:UIEdgeInsetsMake(0, 40, 0, 0) resizingMode:UIImageResizingModeTile]; 

    [[UIBarButtonItem appearance] setBackButtonBackgroundImage:backButtonImage forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];

    [[UIBarButtonItem appearance] setBackButtonTitlePositionAdjustment:UIOffsetMake(NSIntegerMin, NSIntegerMin) forBarMetrics:UIBarMetricsDefault];


    自定义返回按钮的一些方法:

    1、在父视图中修改

    UIBarButtonItem *backItem = [[UIBarButtonItem alloc] initWithTitle:@"返回上级界面" style:UIBarButtonItemStylePlain target:nil action:nil];     self.navigationItem.backBarButtonItem = backItem;

    ●这种方法只是修改文字部分,如果文字设置问空,但是箭头右边空白部分依然可以响应返回方法。

    UIBarButtonItem *backItem = [[UIBarButtonItem alloc] initWithImage:[UIImage imageNamed:@"箭头left40x40"] style:UIBarButtonItemStylePlain target:nil action:nil];  self.navigationItem.backBarButtonItem = backItem;

    ●这种方法只是改变返回按钮的文字部分,文字被替换成我们设置的图片,左边原本的箭头依然存在。

    UIButton *backButton = [UIButton buttonWithType:UIButtonTypeCustom];     backButton.frame = CGRectMake(0, 0, 30, 30);    

    [backButton setTitle:@"返回" forState:UIControlStateNormal];    

    [backButton setImage:[[UIImage imageNamed:@"箭头left40x40"] imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal] forState:UIControlStateNormal];    

    self.navigationItem.backBarButtonItem = [[UIBarButtonItem alloc] initWithCustomView:backButton];

    ●这种方法会使返回按钮文字部分消失,图片也不起效果。

    UIBarButtonItem *backItem = [[UIBarButtonItem alloc] initWithTitle:@"返回" style:UIBarButtonItemStylePlain target:nil action:nil];     self.navigationController.navigationBar.backIndicatorImage = [[UIImage imageNamed:@"箭头left40x40"] imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal];     self.navigationController.navigationBar.backIndicatorTransitionMaskImage = [[UIImage imageNamed:@"箭头left40x40"] imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal];     self.navigationItem.backBarButtonItem = backItem;

    ●这种方法会使返回按钮部分文字消失,图片需要渲染成原始的才有效果,而且图片会应用于本导航控制器下的所有有返回按钮的页面。


    2、在本视图中修改

    修改成只有图片的:

    UIBarButtonItem *backItem = [[UIBarButtonItem alloc] initWithImage:[UIImage imageNamed:@"箭头left40x40"] style:UIBarButtonItemStylePlain target:self action:@selector(back)];    

    self.navigationItem.leftBarButtonItem = backItem;

    修改成只有文字的:

    UIBarButtonItem *backItem = [[UIBarButtonItem alloc] initWithTitle:@"返回" style:UIBarButtonItemStylePlain target:self action:@selector(back)];     self.navigationItem.leftBarButtonItem = backItem;

    自定义文字和图片的:

    UIButton *backButton = [UIButton buttonWithType:UIButtonTypeCustom];     backButton.frame = CGRectMake(0, 0, 30, 30);    

    [backButton setTitle:@"返回" forState:UIControlStateNormal];    

    [backButton setImage:[[UIImage imageNamed:@"箭头left40x40"] imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal] forState:UIControlStateNormal];    

    [backButton addTarget:self action:@selector(back) forControlEvents:UIControlEventTouchUpInside];     self.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithCustomView:backButton];

    注意:上面这3种方法都会使返回按钮偏右,不大美观;而且会使系统自带的右滑手势返回移除控制的功能失效,解决办法就是在后面添加一句代码,让导航控制器重新设置这个功能,解决代码如下:

    self.navigationController.interactivePopGestureRecognizer.delegate = nil;


    以上这些方法都是只是在一个视图中生效,但是大多数项目自定义返回按钮的需求都是全局一样的,所以下面说一下全局设置的方法:

    3、全局设置返回按钮

    网上找到大多是说自定义一个导航控制器,重写其push方法,通过拦截push方法来进行设置,我对其稍微改动了点,代码如下:

    -(void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated {

    if (self.childViewControllers.count > 0) {

    UIBarButtonItem *backItem = [[UIBarButtonItem alloc] initWithImage:[[UIImage imageNamed:@"箭头left40x40"] imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal] style:UIBarButtonItemStylePlain target:self action:@selector(back)];         backItem.imageInsets = UIEdgeInsetsMake(0, -10, 0, 0);         viewController.navigationItem.leftBarButtonItem = backItem; viewController.hidesBottomBarWhenPushed = YES;

    [super pushViewController:viewController animated:animated];

    self.interactivePopGestureRecognizer.delegate = nil;

    }  

    - (void)back {     [self popViewControllerAnimated:YES]; }

    这样改动的好处就是会更美观一点,点击返回响应区域也比较大;如果是原来的自定义按钮来设置,让箭头向左偏移一点,这样的话整个返回按钮的响应区域就会很小,而且会使箭头有部分无法响应返回。


    当然网上也有说明使用分类的方法进行处理,代码如下:

    - (BOOL)navigationBar:(UINavigationBar *)navigationBar shouldPushItem:(UINavigationItem *)item{

    UIBarButtonItem *back = [[UIBarButtonItem alloc] initWithTitle:@"" style:UIBarButtonItemStylePlain target:nil action:nil];

    item.backBarButtonItem = back;

    return YES;

    }  

    这种方法也是有缺陷的,这样会使第一个控制器也出现返回按钮。


    下面给出的最后一种方法,也就是我最后所使用的解决方法,代码如下:

    #import "UINavigationController+Addtion.h"

    #import "NSObject+Swizzling.h"

    @implementation UINavigationController (Addtion)

    + (void)load {    

    static dispatch_once_t onceToken;    

    dispatch_once(&onceToken, ^{        

    [self methodSwizzlingWithOriginalSelector:@selector(pushViewController:animated:) bySwizzledSelector:@selector(replacePushViewController:animated:)];    

    });

    }

    - (void)replacePushViewController:(UIViewController *)viewController animated:(BOOL)animated {    

    if (self.viewControllers.count > 0) {        

    UIBarButtonItem *backItem = [[UIBarButtonItem alloc] initWithImage:[[UIImage imageNamed:@"箭头left40x40"] imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal] style:UIBarButtonItemStylePlain target:self action:@selector(back)];         backItem.imageInsets = UIEdgeInsetsMake(0, -10, 0, 0);         viewController.navigationItem.leftBarButtonItem = backItem;         viewController.hidesBottomBarWhenPushed = YES;    

    }    

    [self replacePushViewController:viewController animated:animated];     self.interactivePopGestureRecognizer.delegate = nil;

    }

    - (void)back {    

    [self popViewControllerAnimated:YES];

    }

    @end

    这是添加一个NavicationController的分类,使用runtime进行方法交换,然后再执行原来的方法,并且右滑手势返回也可以使用。至于方法交换的代码下面给出:

    Class class = [self class];     //原有方法    

    Method originalMethod = class_getInstanceMethod(class, originalSelector);     //替换原有方法的新方法    

    Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);     

    BOOL didAddMethod = class_addMethod(class,originalSelector,                                         method_getImplementation(swizzledMethod),                                         method_getTypeEncoding(swizzledMethod));    

    if (didAddMethod) {

    class_replaceMethod(class,swizzledSelector,                             method_getImplementation(originalMethod),                             method_getTypeEncoding(originalMethod));    

    } else {       

    method_exchangeImplementations(originalMethod, swizzledMethod);    

    }


    总结:

    这些自定义返回按钮的方式,各有利弊,实际根据自己的项目需求使用,总有比较适合自己的方式。

    PS:以上就是我所知道的自定义按钮的方式,如果有不足或者说的不对的地方,希望大家多多指正,可以评论指出或者私信我。

    demo地址

    方法交换分类

    UINavigationController分类

    参考文章:

    iOS关于设置一个自定义的导航控制器返回按钮

    ios自定义图片替换系统导航栏返回按钮样式

    iOS返回按钮自定义

    相关文章

      网友评论

        本文标题:iOS关于自定义返回按钮

        本文链接:https://www.haomeiwen.com/subject/noehfftx.html