摘要:runtime、kvo、selectedIndex
在开发公司的一个项目时,遇到一个说正常也正常,说奇葩也奇葩的需求,需求如下:1、首页tab有好几个其他tab的入口,点击该入口进入指定的tab(比如“秘籍"Tab)
2、进入“秘籍”tab需要先登录才能进入使用,
2.1、如果用户已经登录,直接进入“秘籍”tab使用相关功能
2.2、如果当用户未登录时,是需要先到“秘籍”tab所在页面显示空白提示信息,比如“您还未登录,请先登录”,然后展示登录页面
2.2.1、如果正常登录或注册成功,显示“秘籍”tab功能
2.2.2、如果用户在登录页面没有登录,点击了登录页面的返回按钮,则要返回到“秘籍”之前的那个tab里,比如如果从首页的某个按钮进入“秘籍”tab,则返回首页,如果从“我的”tab点击底部的tab按钮就如“秘籍”tab,则回到“我的”tab
刚拿到这个需求时想着直接写个自定义的UITabBarController,然后设置代理,在shouldSelectViewController:和didSelectViewController:做下记录进入“秘籍”tab之前的tab,然后当点击登录页面返回按钮时直接选中记录的tab就可以了,但在实际的代码里却发现当从首页的按钮入口进入“秘籍”tab时,使用的是tabVC.selectedIndex=xx,这种方式并不会走UITabBarController的代理回调,为了解决这个办法,我写了一个UITabBarController的分类,使用runtime修改了setSelectIndex:这个方法,然后为了方便起见,也放弃了在shouldSelectViewController:和didSelectViewController:中操作的方式,因为该方法可以同时让我拿到old和new两个tab对应的内容,而使用kvo监听selectedViewController的变化,但kvo也面临使用setSelectIndex收不到通知的问题,解决办法是在这个分类里手动发送kvo的变化,具体分类代码如下:
// ZQQTabBarController.h
@interface ZQQTabBarController :UITabBarController
@end
// ZQQTabBarController.m
@implementation ZQQTabBarController
- (void)viewDidLoad {
[self addObserver:selfforKeyPath:@"selectedViewController"options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOldcontext:nil];
}
- (void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)object change:(NSDictionary*)change context:(void*)context{
if([keyPathisEqualToString:@"selectedViewController"]) {
if(nil != change[@"old"]) {//记录前一个tab}
}
}
// UITabBarController+ZQQTabBarController.h
@interface UITabBarController (ZQQTabBarController)
@end
// UITabBarController+ZQQTabBarController.m
@implementation UITabBarController (ZQQTabBarController)
+ (void)load
{
MethodoriginalMethod =class_getInstanceMethod(self,@selector(setSelectedIndex:));
MethodswizzledMethod =class_getInstanceMethod(self,@selector(zqq_setSelectedIndex:));
method_exchangeImplementations(originalMethod, swizzledMethod);
}
- (void)zqq_setSelectedIndex:(NSInteger)index
{
if(index >=self.childViewControllers.count|| index <0) {
return;
}
if(self.delegate==nil) {//如果没有代理,直接调用修改index的方法
[self willChangeValueForKey:@"selectedViewController"];
[self zqq_setSelectedIndex:index];
[self didChangeValueForKey:@"selectedViewController"];
}else{
//即将被选中的controller
UIViewController *willBeSelectController =self.childViewControllers[index];
//如果可以被选中,则继续下一步操作
if([self zqq_canSelectController:willBeSelectController]) {
//手动发送节将修改通知
[self willChangeValueForKey:@"selectedViewController"];
//修改index
[self zqq_setSelectedIndex:index];
//如果delegate实现了这个代理方法,调用该代理方法
if([self.delegate respondsToSelector:@selector(tabBarController:didSelectViewController:)]) {
[self.delegate tabBarController:selfdidSelectViewController:willBeSelectController];
}
//发送已经修改内容通知
[self didChangeValueForKey:@"selectedViewController"];
}
}
}
/**
判断即将被选中的controller是否可以被选中
@paramwillBeSelectController即将被选中的controller
@return是否可以被选中
*/
- (BOOL)zqq_canSelectController:(UIViewController*)willBeSelectController
{
//如果delegate实现了判断是否可选中的方法,用代理的方法判断
if([self.delegate respondsToSelector:@selector(tabBarController:shouldSelectViewController:)]) {
return[self.delegate tabBarController:selfshouldSelectViewController:willBeSelectController];
}else{
//如果delegate没有实现是否允许选中controller的方法,则默认为可以选中
returnYES;
}
}
@end