UITabBarController的selectedIndex不走delegate和kvo

摘要: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

你可能感兴趣的:(UITabBarController的selectedIndex不走delegate和kvo)