UINavigationController的那些事

为了便于控制器的管理,iOS提供了2个比较特殊的控制器

UINavigationController
UITabBarControlle
  • 利用UINavigationController,可以轻松地管理多个控制器,轻松完成控制器之间的切换,典型例子就是系统自带的“设置”应用
Snip20160522_1.png

UINavigationController的简单使用

  • UINavigationController的使用步骤

  • 初始化UINavigationController

  • 设置UIWindow的rootViewController为UINavigationController

  • 根据具体情况,通过push方法添加对应个数的子控制器

管理方式

  • UINavigationController以栈的形式保存子控制器
@property(nonatomic,copy) NSArray*viewControllers;
@property(nonatomic,readonly) NSArray*childViewControllers;
  • 使用push方法能将某个控制器压入栈
- (void)pushViewController:(UIViewController*)viewController animated:(BOOL)animated;
  • 使用pop方法可以移除控制器
    • 将栈顶的控制器移除
- (UIViewController*)popViewControllerAnimated:(BOOL)animated;
  • 回到指定的子控制器
-(NSArray*)popToViewController:(UIViewController
  • 回到根控制器(栈底控制器)
- (NSArray*)popToRootViewControllerAnimated:(BOOL)animated;
Untitled.gif

如何修改导航栏的内容

  • 导航栏的内容由栈顶控制器的navigationItem属性决定
    • UINavigationItem有以下属性影响着导航栏的内容
左上角的返回按钮
@property(nonatomic,retain) UIBarButtonItem*backBarButtonItem;
中间的标题视图
@property(nonatomic,retain) UIView          *titleView;
中间的标题文字
@property(nonatomic,copy)   NSString        *title;
左上角的视图
@property(nonatomic,retain) UIBarButtonItem*leftBarButtonItem;
UIBarButtonItem*rightBarButtonItem  右上角的视图
@property(nonatomic,retain) UIBarButtonItem*rightBarButtonItem;

注意:

  • 导航条上面的子控件位置由系统决定,我们自己只能决定控件的尺寸
// 导航条上面的子控件位置由系统决定,我们自己只能决定控件的尺寸
    UIView *view = [[UIView alloc] initWithFrame:CGRectMake(0, 2000, 200, 35)];
 
    view.backgroundColor = [UIColor redColor];

   // titleView
    self.navigationItem.titleView = view;
  • 在iOS7之后,默认会导航条上按钮的图片渲染成蓝色
 // 获取图片
    UIImage *image = [UIImage imageNamed:@"navigationbar_friendsearch"];
    // 告诉这个图片不要渲染
    // 返回一个没有渲染的图片给你
    image = [image imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal];
    
    UIBarButtonItem *item1 = [[UIBarButtonItem alloc] initWithImage:image style:UIBarButtonItemStyleDone target:nil action:nil];
    
    
    // 显示多张图片,不同状态,用按钮
    UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
    [btn setImage:[UIImage imageNamed:@"navigationbar_friendsearch"] forState:UIControlStateNormal];
    
    [btn setImage:[UIImage imageNamed:@"navigationbar_friendsearch_highlighted"] forState:UIControlStateHighlighted];
    
    // 按钮自适应,根据图片计算尺寸
    [btn sizeToFit];
    
    
    UIBarButtonItem *item2 = [[UIBarButtonItem alloc] initWithCustomView:btn];

// 右边内容
    self.navigationItem.rightBarButtonItems = @[item,item1,item2];
  • 在iOS7之后,默认会给导航控制器里所有UIScrollView顶部都会添加额外的滚动区域64
// 设置不需要添加额外滚动区域
    self.automaticallyAdjustsScrollViewInsets = NO;
  • 设置导航条透明
// UIBarMetricsDefault:必须传入这个家伙
    // 传入一个空的UIImage
    [self.navigationController.navigationBar setBackgroundImage:[[UIImage alloc] init] forBarMetrics:UIBarMetricsDefault];
  • 设置导航条阴影背景
[self.navigationController.navigationBar setShadowImage:[[UIImage alloc] init]];

实际开发中UINavigationController使用

  • 一般自定义UINavigationController

1.拿到全局的UINavigationBar设置文字属性跟背景图片

// 在自定义的UINavigatonController的.m文件中实现
+ (void)load
{
    // 获取整个应用的navigationBar
    UINavigationBar *navigationBar = [UINavigationBar appearanceWhenContainedIn:self, nil];
    
    // 设置导航条标题
    NSMutableDictionary *titleAttr = [NSMutableDictionary dictionary];
    titleAttr[NSFontAttributeName] = [UIFont systemFontOfSize:20];
    [navigationBar setTitleTextAttributes:titleAttr];
    
    // 设置导航条背景图片:一定要是UIBarMetricsDefault
    // iOS9之前:UIBarMetricsDefault,导航控制器跟控制器的view尺寸会减少64
    // iOS9没有了.
    // iOS8和iOS9适配:
    [navigationBar setBackgroundImage:[UIImage imageNamed:@"navigationbarBackgroundWhite"] forBarMetrics:UIBarMetricsDefault];
    
}

2.默认一个导航控制器都是有滑动返回功能,当我们自定义返回按钮的时候,手势就会消失

  • 解决方案
    • 如果直界面设置,貌似可以手势滑动了,但是会出现假死
self.navigationController.interactivePopGestureRecognizer = NO;
  • 真正采取的方案 遵守代理实现代理方法
self.navigationController.interactivePopGestureRecognizer.delegate = self;

#pragma mark - UIGestureRecognizerDelegate
/**
 *  这个代理方法的作用:决定pop手势是否有效
 *
 *  @return YES:手势有效, NO:手势无效
 */
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch
{
// 注意如果是在自定义的UINaVigationController中实现该方法就写成
    return self.childViewControllers.count > 1;
// 如果是在childViewControllers实现该方法,写成
  return self.navigationController.childViewControllers.count > 1;
}

  • 统一设置程序的控制器的返回按钮(在自定义的UINavigationController重写push方法)
/**
 *  重写这个方法的目的:为了拦截整个push过程,拿到所有push进来的子控制器
 *
 *  @param viewController 当前push进来的子控制器
 */
- (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated
{
    //    if (viewController != 栈底控制器) {
    if (self.viewControllers.count > 0) {
        // 当push这个子控制器时, 隐藏底部的工具条
        viewController.hidesBottomBarWhenPushed = YES;
        
        UIButton *backButton = [UIButton buttonWithType:UIButtonTypeCustom];
        [backButton setImage:[UIImage imageNamed:@"navigationButtonReturn"] forState:UIControlStateNormal];
        [backButton setImage:[UIImage imageNamed:@"navigationButtonReturnClick"] forState:UIControlStateHighlighted];
        [backButton setTitle:@"返回" forState:UIControlStateNormal];
        [backButton setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
        [backButton setTitleColor:[UIColor redColor] forState:UIControlStateHighlighted];
        backButton.titleLabel.font = [UIFont systemFontOfSize:16];
        [backButton sizeToFit];
        [backButton addTarget:self action:@selector(back) forControlEvents:UIControlEventTouchUpInside];
        backButton.contentEdgeInsets = UIEdgeInsetsMake(0, - 20, 0, 0);
        viewController.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithCustomView:backButton];
    }
    
    // 将viewController压入栈中
    [super pushViewController:viewController animated:animated];
}

- (void)back
{
    [self popViewControllerAnimated:YES];
}
  • 设置全局滑动


你可能感兴趣的:(UINavigationController的那些事)