前言:
就在上个月也就是年末的时候,打开了以前的一个项目,忽然发现导航栏返回按钮几乎看不到了,一直忙另外一个项目,直到这几天才处理一下,总结一下遇到的一些问题和处理方法。
问题截图如下:
原来没有任何问题,但是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:以上就是我所知道的自定义按钮的方式,如果有不足或者说的不对的地方,希望大家多多指正,可以评论指出或者私信我。
参考文章: