多控制器切换在项目中应用十分广泛 , 市面上的大部分app都能见到其身影 .
首先我认为 , 多控制器的切换 , 应尽量避免一次性加载所有的控制器页面 , 造成内存飙升 , 应尽量满足用到时加载 , 不用时不加载 .
当然也不乏有例外 , 有的时候为了更好的用户体验 , 而页面又较少时 , 可以一次性加载完毕 , 给用户造成一种切换页面加载迅速,不卡顿的假象 , 具体还是看项目需求.
之前我做多控制器切换, 大多在横向加了UIScrollView , 设置UIViewController的ChildViewControllers , 通过改变UIScrollView的偏移量来实现切换 , 如果只能按钮点击切换 , 就禁用调滚动 scrollEnabled = NO , 实际上是比较low的 ,今天突然想到 , 既然是切换 , 可不可以用UITabbarViewController来实现 , 这样有系统自己帮我们管理生命周期 和切换
市面上的APP大体不外乎这三种
仅仅是 多控制器切换 , 就可能采取非常多的策略
仅仅是点击按钮切换页面 , 不可滑动切换
此种情况 , 我个人觉得巧用tabbarController实现再好不过了可以通过按钮点击切换页面 , 也可滑动切换
此种情况 , 个人建议还是用scrollView实现 , 前提是切换按钮并不是很多的情况 , 不超过4个吧 差不多顶部按钮特别多 , 又要支持滑动动画的情况 ,这种情况 直接用JXCategoryView第三方就行了
这里介绍一个第一种写法.比较简单,
#import "BBTabbarVC.h"
#import "UIImage+Color.h"
#import "BBHomeVC.h"
#import "BBJiHuaPaiHangVC.h"
#import "BBShouCangVC.h"
#define TOPBTNARR @[@"全部计划",@"计划排行",@"我的收藏"]
@interface BBTabbarVC ()
@property (nonatomic ,strong)NSMutableArray *btnArr;
@end
@implementation BBTabbarVC
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
[UITabBar appearance].translucent = YES;
//切换栏
[self setUpSelectedBar];
//设置子控制器
[self configChildControllers];
}
-(void)setUpSelectedBar {
UIView *tabbarView = [[UIView alloc] init];
tabbarView.frame = CGRectMake(0, kScreenHeight-kTabbarHeight, kScreenWidth, kTabbarHeight);
tabbarView.backgroundColor = HIGHGREENCOLOR;
[self.view addSubview:tabbarView];
self.btnArr = [NSMutableArray array];
for (int i = 0; i < TOPBTNARR.count; i++) {
UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
[btn setTitleColor:WHITECOLOE forState:UIControlStateNormal];
[btn setTitleColor:SHENGREENCOLOR forState:UIControlStateSelected];
btn.titleLabel.font = FONTSYS(16);
btn.tag = 10+i;
[btn setTitle:TOPBTNARR[i] forState:UIControlStateNormal];
[btn setBackgroundImage:[UIImage imageWithColor:GREENCOLOR] forState:UIControlStateNormal];
[btn setBackgroundImage:[UIImage imageWithColor:HIGHGREENCOLOR] forState:UIControlStateSelected];
[btn addTarget:self action:@selector(selectVC:) forControlEvents:UIControlEventTouchUpInside];
if (i == 0) {
btn.selected = YES;
}
btn.frame = CGRectMake(tabbarView.czh_width/TOPBTNARR.count*i, 0, tabbarView.czh_width/TOPBTNARR.count, tabbarView.czh_height);
[tabbarView addSubview:btn];
[self.btnArr addObject:btn];
}
}
//设置子控制器
//设置子控制器的好处在于 , 我们不需要去关心各个控制器的生命周期 , 控制器在用到时才会加载 , 且只加载一次 .
- (void)configChildControllers {
NSArray *arr1 = @[@"BBHomeVC",@"BBJiHuaPaiHangVC",@"BBShouCangVC"];
for (NSInteger index = 0; index <arr1.count; index ++) {
Class VcClass = NSClassFromString(arr1[index]);
UIViewController *viewController = [[VcClass alloc]init];
UINavigationController *nav = [[UINavigationController alloc] initWithRootViewController:viewController];
[self addChildViewController:nav];
}
// 隐藏tabbar ,或者直接移除
self.tabBar.hidden = YES;
[self.tabBar removeFromSuperview];
}
//切换控制器
- (void)selectVC:(UIButton *)btn
{
for (UIButton *button in self.btnArr) {
button.selected = NO;
}
btn.selected = YES;
self.selectedIndex = btn.tag-10;
}
这里是有一个问题,,就是出现二级界面的时候自定义的tabbar不会隐藏,,
所以要在.m中写一个隐藏的方法
-(BOOL)hidesBottomBarWhenPushed {
//这里能做一些隐藏的动画
if (self.tabbarView.isHidden) {
self.tabbarView.hidden = NO;
self.tabBar.hidden = YES;
[UIView animateWithDuration:0.3 animations:^{
self.tabbarView.frame = CGRectMake(0, kScreenHeight-kTabbarHeight, kScreenWidth, kTabbarHeight);
}];
} else {
self.tabbarView.hidden = YES;
self.tabBar.hidden = YES;
[UIView animateWithDuration:0.3 animations:^{
self.tabbarView.frame = CGRectMake(0, kScreenHeight, kScreenWidth, kTabbarHeight);
}];
}
return YES;
}
使用
在要隐藏tabbar 的界面调用
-(void)viewWillAppear:(BOOL)animated {
[super viewWillAppear:animated];
[self.tabBarController hidesBottomBarWhenPushed];
}
-(void)viewWillDisappear:(BOOL)animated {
[super viewWillDisappear:animated];
[self.tabBarController hidesBottomBarWhenPushed];
}