登录 立即注册
金钱:

Code4App-iOS开发-iOS 开源代码库-iOS代码实例搜索-iOS特效示例-iOS代码例子下载-Code4App.com

iOS关于自定义返回按钮

[复制链接]
来自: 独孤红雨 分类: iOS精品源码 上传时间: 2018-3-12 11:25:13
Tag:

项目介绍:

前言:

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

问题截图如下:

图片描述

原来没有任何问题,但是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返回按钮自定义

相关源码推荐:

我来说两句
*滑动验证:
所有评论(5)
逆鳞九霄 2018-3-12 14:27:49
虽不明,但觉厉!
回复
littleRed 2018-3-12 14:27:57
code4app好的代码demo真的很多,谢谢啦~
回复
inta加加 2018-3-12 14:28:02
虽不明,但觉厉...
回复
phoiu 2018-3-12 14:28:05
感谢分享,楼主V5~
回复
kengsir 2018-3-12 14:28:13
code4app好的代码demo真的很多,谢谢啦~
回复
知新1990 2018-3-12 14:29:07
帮帮顶顶!!
回复
天天bug 2018-3-12 14:29:21
感谢分享,楼主V5~
回复
code4app热心网友 2018-3-12 15:54:27
楼主威武啊,Code4App有你更给力!
回复
code4app热心网友 2018-3-13 08:37:09
学习学习!
回复
提取码:  下载次数:21 状态:已购或VIP 售价:0(原价:15)金钱 下载权限:初级码农 
790 1 21
联系我们
首页/微信公众账号投稿

帖子代码编辑/版权问题

QQ:435399051,742864542

如何获得代码达人称号?

代码贡献英雄榜
用户名 下载数
通过邮件订阅最新 Code4App 信息
上一条 /4 下一条
联系我们
关闭
合作电话:
13802416937
Email:
435399051@qq.com
商务市场合作/投稿
问题反馈及帮助
联系我们

广告投放| Github|申请友链|手机版|Code4App ( 粤ICP备15117877号-1 )

快速回复 返回顶部 返回列表