转自: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. 在linux上使用ssh登录服务器,Linux权限

    本文是作者原创,版权归作者所有.若要转载,请注明出处 ssh为Secure Shell(安全外壳协议)的缩写. 很多ftp.pop和telnet在本质上都是不安全的. 我们使用的Xshell6就是基于 ...

  2. js正则匹配的出链接地址

    content为需要匹配的值 var b=/<a([\s]+|[\s]+[^<>]+[\s]+)href=(\"([^<>"\']*)\"| ...

  3. tap事件封装

    <!DOCTYPE html> <html lang="zh"> <head>     <meta charset="UTF-8 ...

  4. 通过javascript 执行环境理解她

    古往今来最难的学的武功(javascript)算其一. 欲练此功必先自宫,愿少侠习的此功,笑傲江湖. 你将了解 执行栈(Execution stack) 执行上下文(Execution Context ...

  5. 在C\C++中char 、short 、int各占多少个字节

    在C\C++中char .short .int各占多少个字节 : #include <bits/stdc++.h> using namespace std; int main() { co ...

  6. ArcGIS API For Javascript :如何动态生成 token 加载权限分配的地图服务?

    一.需求 项目中我们通常会遇到为外协团队.合作友商提供地图服务的需求,因此对地图服务的权限需要做出分配. 二.现状 主流的办法是用用户和角色来控制,通常使用代理方式和用户名密码的方式来实现. 三.思路 ...

  7. 学习记录:《C++设计模式——李建忠主讲》5.“对象性能”模式

    对象性能模式:面向对象很好地解决了抽象地问题,但是必不可免地要付出一定地代价.对于通常情况来讲,面向对象地成本大都可以忽略不计,但某些情况,面向对象所带来地成本必须谨慎处理. 典型模式:单件模式(Si ...

  8. 源码分析RocketMQ消息轨迹

    目录 1.发送消息轨迹流程 1.1 DefaultMQProducer构造函数 1.2 SendMessageTraceHookImpl钩子函数 1.3 TraceDispatcher实现原理 2. ...

  9. 为React绑定事件,并修改state中的值

    import React from 'react' export default class ClickS extends React.Component { constructor () { sup ...

  10. ansible start canal

    - name: Start canal server shell: source /etc/profile && nohup /opt/canal/bin/startup.sh