美文网首页导航栏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