转自:http://blog.csdn.net/xn4545945/article/details/35994863

一、自定义的思路

iOS中的TabBarController确实已经很强大了,大部分主流iOS应用都会采用。但是往往也不能满足全部的需求,因此需要自定义TabBar,自定义需要对系统的TabBar工作方式有很好的理解,自定义需要勇气。

自定义TabBar的原则:尽量利用系统自带TabBar,只改需要改的地方。

二、自定义TabBar的总体过程
1.先把自带的TabBar条给取消了
2.自己做一个view,上面放几个按钮,设定按钮的点击事件.并设置selectIndex。
3.关联各个子viewController,覆盖相关事件。

三、细节很重要

1. 让自己创建的按钮关联到viewController:
•用tabbar的selectedIndex属性.设置这个属性就行了.
2. 取消系统的高亮:
•可以自定义一个按钮.重写里面的setHighhighted方法,什么也不做就行了.(如果调用super就相当于没写)
3. 关于几个按钮只选中一个的方法:
•设置一个属性,记录上一个选中的按钮.
•点击当前按钮时,把上一个按钮设置为未选中,并把当前按钮设置为选中,最后把当前按钮赋值给上一个按钮.
四、初步自定义
直接上代码,详见注释。
 

XNTabBarController.h

  1. #import <UIKit/UIKit.h>

    @interface

    @end

XNTabBarController.m

  1. //
    //  XNTabBarController.m
    //
    //
    //  Created by neng on 14-6-19.
    //  Copyright (c) 2014年 neng. All rights reserved.
    //

    #import "XNTabBarController.h"
    #import "Common.h"
    #import "XNTabBarButton.h"

    @interface
    /**

  2. *  设置之前选中的按钮
  3. */
    @propertynonatomicUIButton
    @end

    @implementation

    void
    super];

  4. //    NSLog(@"%s",__func__);
    //    NSLog(@"%@",self.view.subviews); //能打印出所有子视图,和其frame

    self);

  5. self.frame
    self removeFromSuperview
  6. UIView]];
  7. = rect;
  8. = [UIColor];
  9. self addSubview

    forint; i < ; i++) {

  10. XNTabBarButton]];
  11. NSString:, i +];
  12. NSString:, i +];
  13. :[UIImage:imageName]:UIControlStateNormal];
  14. :[UIImage:imageNameSel]:UIControlStateSelected];
  15. i.size / ;
  16. = CGRectMake(x, , myView.size / , myView.size);
  17. :btn];
  18. = i;
  19. :self:@selector forControlEvents
  20. if == i) {
  21. = YES
    self = btn;
  22. /**
  23. *  自定义TabBar的按钮点击事件
  24. */
    voidUIButton
  25. self.selectedNO
  26. = YES
  27. self = button;
  28. self = button;
  29. @end

XNTabBarButton.h

  1. #import <UIKit/UIKit.h>

    @interface

    @end

XNTabBarButton.m

  1. #import "XNTabBarButton.h"

    @implementation
    /**什么也不做就可以取消系统按钮的高亮状态*/
    voidBOOL
    //    [super setHighlighted:highlighted];

    @end

五、代码重构

重构的目的是把代码放到他最该到的地方去. 提高可读写与可拓展性。
对控件的重构要保证可重用性. 做到封装做其他应用时,可以直接拿过去用的地步.
tips :
 
1、关于init与initWithFrame:
•在对象初始化调用init时,会调用initWithFrame方法.
•Init与initWithFrame都会被调用.
•建议自定义控件不要重写init方法,需要初始化时重写initWithFrame方法.
•好处:其他人调用无论是调用init,还是调用initWithFrame都会调用initWithFrame方法.
2、关于控件的布局代码:
•建议写在layoutSubviews方法中.
•不要忘记写super方法
•将设置x,y,frame等写在这里面.
3、将自定义的Tabbar添加为系统TabBar的子视图,这样TabBar的切换自动隐藏/滑动功能就不用自己做了.
(hidebottombaronpush)
重构后的代码如下
将自定义的TabBar单独建立,并将代码移过去。
设置代理方法,工具栏按钮被选中,记录从哪里跳转到哪里. 

XNTabBar.h

  1. #import <UIKit/UIKit.h>
    @class

    @protocol
    /**

  2. *  工具栏按钮被选中, 记录从哪里跳转到哪里. (方便以后做相应特效)
  3. */
    void tabBarXNTabBar:(NSInteger) from:(NSInteger)to;
  4. @end

    @interface
    @propertynonatomicid
    /**

  5. *  使用特定图片来创建按钮, 这样做的好处就是可扩展性. 拿到别的项目里面去也能换图片直接用
  6. *
  7. *  @param image         普通状态下的图片
  8. *  @param selectedImage 选中状态下的图片
  9. */
    voidUIImage:(UIImage
    @end

XNTabBar.m

  1. //
    //  XNTabBar.m
    //
    //  Created by neng on 14-6-19.
    //  Copyright (c) 2014年 neng. All rights reserved.
    //

    #import "XNTabBar.h"
    #import "XNTabBarButton.h"

    @interface
    /**

  2. *  设置之前选中的按钮
  3. */
    @propertynonatomicUIButton
    @end

    @implementation

    /**

  4. *  在这个方法里写控件初始化的东西, 调用init方法时会调用
  5. */
    //- (id)initWithFrame:(CGRect)frame {
    //  if (self = [super initWithFrame:frame]) {
    //      //添加按钮
    //      for (int i = 0; i < 5; i++) { //取消掉特定的数字
    //          //UIButton *btn = [[UIButton alloc] init];
    //          XNTabBarButton *btn = [[XNTabBarButton alloc] init];
    //
    //          NSString *imageName = [NSString stringWithFormat:@"TabBar%d", i + 1];
    //          NSString *imageNameSel = [NSString stringWithFormat:@"TabBar%dSel", i + 1];
    //
    //          [btn setImage:[UIImage imageNamed:imageName] forState:UIControlStateNormal];
    //          [btn setImage:[UIImage imageNamed:imageNameSel] forState:UIControlStateSelected];
    //
    //          [self addSubview:btn];
    //
    //          btn.tag = i; //设置按钮的标记, 方便来索引当前的按钮,并跳转到相应的视图
    //
    //          //带参数的监听方法记得加"冒号"
    //          [btn addTarget:self action:@selector(clickBtn:) forControlEvents:UIControlEventTouchUpInside];
    //
    //          if (0 == i) {
    //              [self clickBtn:btn];
    //          }
    //      }
    //  }
    //  return self;
    //}

    voidUIImage:(UIImage
    UIButton]];

  6. :image:UIControlStateNormal];
  7. :selectedImage:UIControlStateSelected];
  8. self:btn];
  9. :self:@selector forControlEvents
  10. ifself.count) {
  11. self:btn];
  12. /**专门用来布局子视图, 别忘了调用super方法*/
    void
    super];
  13. intself.count
    forint; i < count; i++) {
  14. UIButtonself[i];
  15. = i;
  16. iself.size / count;
  17. ;
  18. self.size / count;
  19. self.size;
  20. = CGRectMake(x, y, width, height);
  21. /**
  22. *  自定义TabBar的按钮点击事件
  23. */
    voidUIButton
  24. self.selectedNO
  25. = YES
  26. self = button;
  27. ifself respondsToSelector@selector
    self tabBarself:self.tag:button];
  28. @end

原先的XNTabBarController.m经过修改后,注释了原先的代码。

  1. //
    //  XNTabBarController.m
    //
    //  Created by neng on 14-6-19.
    //  Copyright (c) 2014年 neng. All rights reserved.
    //

    #import "XNTabBarController.h"
    #import "XNTabBarButton.h"
    #import "XNTabBar.h"

    @interface
    /**

  2. *  设置之前选中的按钮
  3. */
    @propertynonatomicUIButton
    @end

    @implementation

    void
    super];

  4. //    NSLog(@"%s",__func__);
    //    NSLog(@"%@",self.view.subviews); //能打印出所有子视图,和其frame
    //  LogFun;
    //  LogSubviews(self.view);
  5. self.bounds
  6. self);
  7. XNTabBar]];
  8. = self
  9. = rect;
  10. self addSubview
  11. forint; i<self.count
  12. NSString:, i +];
  13. NSString:, i +];
  14. UIImage:imageName];
  15. UIImage:imageNameSel];
  16. :image:imageSel];
  17. //    //添加按钮
    //  for (int i = 0; i < 5; i++) {
    //      //UIButton *btn = [[UIButton alloc] init];
    //        XNTabBarButton *btn = [[XNTabBarButton alloc] init];
    //
    //      NSString *imageName = [NSString stringWithFormat:@"TabBar%d", i + 1];
    //      NSString *imageNameSel = [NSString stringWithFormat:@"TabBar%dSel", i + 1];
    //
    //      [btn setImage:[UIImage imageNamed:imageName] forState:UIControlStateNormal];
    //      [btn setImage:[UIImage imageNamed:imageNameSel] forState:UIControlStateSelected];
    //
    //      CGFloat x = i * myView.frame.size.width / 5;
    //      btn.frame = CGRectMake(x, 0, myView.frame.size.width / 5, myView.frame.size.height);
    //
    //      [myView addSubview:btn];
    //
    //        btn.tag = i;//设置按钮的标记, 方便来索引当前的按钮,并跳转到相应的视图
    //
    //      //带参数的监听方法记得加"冒号"
    //      [btn addTarget:self action:@selector(clickBtn:) forControlEvents:UIControlEventTouchUpInside];
    //
    //      //设置刚进入时,第一个按钮为选中状态
    //      if (0 == i) {
    //          btn.selected = YES;
    //          self.selectedBtn = btn;  //设置该按钮为选中的按钮
    //      }
    //  }

    /**永远别忘记设置代理*/
    voidXNTabBar:(NSInteger)from:(NSInteger)to {

  18. self = to;
  19. /**
  20. *  自定义TabBar的按钮点击事件
  21. */
    //- (void)clickBtn:(UIButton *)button {
    //  //1.先将之前选中的按钮设置为未选中
    //  self.selectedBtn.selected = NO;
    //  //2.再将当前按钮设置为选中
    //  button.selected = YES;
    //  //3.最后把当前按钮赋值为之前选中的按钮
    //  self.selectedBtn = button;
    //
    //    //4.跳转到相应的视图控制器. (通过selectIndex参数来设置选中了那个控制器)
    //    self.selectedIndex = button.tag;
    //}

    @end

自定义后的效果图:

例子源码下载 http://download.csdn.net/detail/xn4545945/7572263

转载请注明出处:http://blog.csdn.net/xn4545945

iOS 自定义TabBarController的更多相关文章

  1. iOS 自定义tabBarController(中间弧形)

    效果图 1.在继承自UITabBarController的自定义controller中调用以下方法(LZCustomTabbar为自定义的tabbar) - (void)viewDidAppear:( ...

  2. 【iOS自定义键盘及键盘切换】详解

    [iOS自定义键盘]详解 实现效果展示: 一.实现的协议方法代码 #import <UIKit/UIKit.h> //创建自定义键盘协议 @protocol XFG_KeyBoardDel ...

  3. iOS自定义的UISwitch按钮

    UISwitch开关控件 开关代替了点选框.开关是到目前为止用起来最简单的控件,不过仍然可以作一定程度的定制化. 一.创建 UISwitch* mySwitch = [[ UISwitchalloc] ...

  4. 如何实现 iOS 自定义状态栏

    给大家介绍如何实现 iOS 自定义状态栏 Sample Code: 01 UIWindow * statusWindow = [[UIWindow alloc] initWithFrame:[UIAp ...

  5. iOS自定义组与组之间的距离以及视图

    iOS自定义组与组之间的距离以及视图 //头视图高度 - (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(N ...

  6. iOS 自定义转场动画

    代码地址如下:http://www.demodashi.com/demo/12955.html 一.总效果 本文记录分享下自定义转场动画的实现方法,具体到动画效果:新浪微博图集浏览转场效果.手势过渡动 ...

  7. iOS 自定义转场动画浅谈

    代码地址如下:http://www.demodashi.com/demo/11612.html 路漫漫其修远兮,吾将上下而求索 前记 想研究自定义转场动画很久了,时间就像海绵,挤一挤还是有的,花了差不 ...

  8. iOS自定义转场动画实战讲解

    iOS自定义转场动画实战讲解   转场动画这事,说简单也简单,可以通过presentViewController:animated:completion:和dismissViewControllerA ...

  9. iOS 实现Tabbarcontroller中间自定义样式 最简单的方法

    先上图: 如果我们要实现中间按钮自定义样式,方法应该蛮多,这里介绍一种最简单的. 1.创建类继承:UITabBarController,如下代码都是写在该类的 .m文件里 2.定义最中间的自定义样式, ...

随机推荐

  1. IntelliJ IDEA 中设置左菜单字体, 编辑器字体和控制台的字体

    IntelliJ IDEA 中设置左菜单字体大小 File-Settings,然后选择appearance,下图右侧红色边框中的内容即设置菜单的字体和大小 ​ IntelliJ IDEA 中设置当前编 ...

  2. python机器学习——使用scikit-learn训练感知机模型

    这一篇我们将开始使用scikit-learn的API来实现模型并进行训练,这个包大大方便了我们的学习过程,其中包含了对常用算法的实现,并进行高度优化,以及含有数据预处理.调参和模型评估的很多方法. 我 ...

  3. python中的集合、元组和布尔

    #元组,元组跟列表一样,只不过列表可读可写,而元组一般用来只读,不修改#python中不允许修改元组的数据,也包括不能删除其中的元素. t1 = ('a','b','c','d','s','a') & ...

  4. Eclipse中修改tomcat的部署路径deploypath

    在eclipse上面部署web项目后,它没有将你的项目文件放到tomcat 的目录下面.而是放在了你的工作目录下面. 在tomcat上右键选择“stop” .停止eclipse内的Tomcat服务器 ...

  5. T-SQL Part X: UNION, EXCEPT and INTERSECT

    MSDN上关于EXCEPT和INTERSECT的文档.MSDN上关于UNION的文档. 值得注意的是,UNION其实有两种,一种是普通的UNION,另外一种是UNION ALL.加上EXCEPT和IN ...

  6. PHP 核心特性 - 匿名函数

    提出 在匿名函数出现之前,所有的函数都需要先命名才能使用 function increment($value) { return $value + 1; } array_map('increment' ...

  7. 分组取topN

    假设有这样一个文件,文件内容如下 class1 class2 class1 class1 class2 class2 class1 class2 class1 class2 要求按照班级分组取出每个班 ...

  8. (二)初识NumPy库(数组的操作和运算)

    本章主要介绍的是ndarray数组的操作和运算! 一. ndarray数组的操作: 操作是指对数组的索引和切片.索引是指获取数组中特定位置元素的过程:切片是指获取数组中元素子集的过程. 1.一维数组的 ...

  9. JVM浅谈

    **前言** 由于先前也遇到过一些性能问题,OOM算是其中的一大类了.因此也对jvm产生了一些兴趣.自己对jvm略做了些研究.后续继续补充. **从oom引申出去** 既然说到oom,首先需要知道oo ...

  10. 接口测试之-postman

    在使用postman进行接口测试的时候,对于有些接口字段需要时间戳加密,这个时候我们就遇到2个问题,其一是接口中的时间戳如何得到?其二就是对于现在常用的md5加密操作如何在postman中使用代码实现 ...