您的位置:首页 > 移动开发 > IOS开发

iOS7上leftBarButtonItem无法实现滑动返回的完美解决方案

2014-06-11 10:05 281 查看
今天遇到了在iOS7上使用leftBarButtonItem却无法响应滑动返回事件的问题,一番谷歌,最后终于解决了,在这里把解决方案分享给大家。

在iOS7之前的系统,如果要自定义返回按钮,直接设置backBarButtonItem是不行的,有两种方式,一种是用leftBarButtonItem替代;一种是:

[plain] view
plaincopy





UIImage *backButtonImage = [[UIImage imageNamed:@"Graphics/Shared/navigation_back_button.png"] resizableImageWithCapInsets:UIEdgeInsetsMake(0, 0, 0, 0)];

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

[[UIBarButtonItem appearance] setBackButtonTitlePositionAdjustment:UIOffsetMake(0, backButtonImage.size.height*2) forBarMetrics:UIBarMetricsDefault];

但第二种方法只能自定义返回按钮的背景和文字,如果要自定义上面的图标则不行。所以使用leftBarButtonItem是最佳方式。

但是在iOS7系统中,自带了可以通过右滑返回上一级页面的手势,如果仅仅修改leftBarButtonItem是无法响应这个手势的。可以在pushViewController之后加入如下代码:

[plain] view
plaincopy





[_currentNav pushViewController:viewController animated:YES];

//开启iOS7的滑动返回效果

if ([_currentNav respondsToSelector:@selector(interactivePopGestureRecognizer)]) {

_currentNav.interactivePopGestureRecognizer.delegate = nil;

}

这样即可完美解决iOS7上无法滑动的问题。

当然网上还有一种解决方法是在设置leftBarButtonItem之后加上:

[plain] view
plaincopy





self.navigationItem.backBarButtonItem = backButton;

self.navigationController.interactivePopGestureRecognizer.delegate = self;

但这两行代码必须放在viewWillAppear中,相比上一种方式要修改的代码更多,因此建议使用上一种方式。

响应了iOS7的滑动返回手势之后,我们还会发现一个问题,iOS7的滑动返回不会调用我们给leftBarButtonItem设置的回调,那么返回的事件如何捕捉呢?可以通过以下代码:

[plain] view
plaincopy





- (void)viewWillDisappear: (BOOL)animated

{

[super viewWillDisappear: animated];

if (![[self.navigationController viewControllers] containsObject: self])

{

// the view has been removed from the navigation stack, back is probably the cause

// this will be slow with a large stack however.

}

}

至此,我们就完美解决了iOS7上使用leftBarButtonItem的滑动返回问题。

有朋友提出以上方式在多次滑动之后会导致界面假死,这里再给出一种解决方案:

在所有除一级页面之外的页面的viewDidAppear和viewWillDisappear中加入以下代码:

[objc] view
plaincopy





- (void)viewWillDisappear:(BOOL)animated {

[super viewWillDisappear:animated];

//代理置空,否则会闪退

if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) {

self.navigationController.interactivePopGestureRecognizer.delegate = nil;

}

}

[objc] view
plaincopy





- (void)viewDidAppear:(BOOL)animated {

[super viewDidAppear:animated];

//开启iOS7的滑动返回效果

if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) {

//只有在二级页面生效

if ([self.navigationController.viewControllers count] == 2) {

self.navigationController.interactivePopGestureRecognizer.delegate = self;

}

}

}

在uinavigationcontroller的delegate中实现以下方法:

[objc] view
plaincopy





- (void)navigationController:(UINavigationController *)navigationController didShowViewController:(UIViewController *)viewController animated:(BOOL)animated {

//开启滑动手势

if ([navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) {

navigationController.interactivePopGestureRecognizer.enabled = YES;

}

}

在pushviewcontroller之前加入以下代码:

[objc] view
plaincopy





//在切换界面的过程中禁止滑动手势,避免界面卡死

if ([_currentNav respondsToSelector:@selector(interactivePopGestureRecognizer)]) {

_currentNav.interactivePopGestureRecognizer.enabled = NO;

}

[_currentNav pushViewController:viewController animated:YES];

即可在实现滑动返回的同时,避免界面卡死的问题。

如果大家觉得对自己有帮助的话,还希望能帮顶一下,谢谢:)

个人博客:http://blog.csdn.net/zhaoxy2850
本文地址:/article/1607017.html
转载请注明出处,谢谢!
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐