iOS-UITabBarController詳細(xì)總結(jié)

一、UITabBarController以其相關(guān)控件之間的關(guān)系

@interface UITabBarController : UIViewController
@property(nonatomic,readonly) UITabBar *tabBar;

@interface UITabBar : UIView
@property(nullable, nonatomic, copy) NSArray<UITabBarItem *> *items;

@interface UITabBarItem : UIBarItem 
@property(nullable, nonatomic,strong) UIImage *selectedImage;
@property(nullable, nonatomic, copy) NSString *badgeValue;
  • UIBarItem
    一個可以放置在Bar之上的所有小控件類的抽象類,可以設(shè)置標(biāo)題,圖片等
  • UITabBarItem
    繼承UIBarItem岂丘,增加了selected一也、unselected時不同狀態(tài)以及badgeValue等屬性,相當(dāng)于放在TabBar上的特殊“button
  • UITabBar
    NavigaitonBar就是底部的選擇欄 主要對UITabBarItem進行管理 并負(fù)責(zé)展示底部選擇欄的外觀背景
  • UITabBarController
    包含了viewcontrollers风纠、tabBar等

關(guān)系綜述

1蚯斯、UITabBarController繼承UIViewController薄风,是一個Container饵较;
2、UITabBarController擁有一個readonlyTabBar遭赂,TabBar擁有一到多個TabBarItem循诉;
3、每一個TabBarItem需要關(guān)聯(lián)一個UIViewController撇他;
這里可以參考IOS-UINavigationController詳解關(guān)于UINavigationController等相關(guān)控件之間的關(guān)系綜述茄猫,UITabBarControllerUINavigationController相似。

2困肩、UITabBarController及其相關(guān)控件的屬性和方法

1.UITabBarItem

UITabBarItem時一個抽象類划纽,主要負(fù)責(zé)設(shè)置底部每個Item的文字和圖片等屬性。

typedef NS_ENUM(NSInteger, UITabBarSystemItem) {
    UITabBarSystemItemMore,
    UITabBarSystemItemFavorites,
    UITabBarSystemItemFeatured,
    UITabBarSystemItemTopRated,
    UITabBarSystemItemRecents,
    UITabBarSystemItemContacts,
    UITabBarSystemItemHistory,
    UITabBarSystemItemBookmarks,
    UITabBarSystemItemSearch,
    UITabBarSystemItemDownloads,
    UITabBarSystemItemMostRecent,
    UITabBarSystemItemMostViewed,
};
NS_CLASS_AVAILABLE_IOS(2_0) @interface UITabBarItem : UIBarItem 
//初始化方法
- (instancetype)init;
- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder;
- (instancetype)initWithTitle:(nullable NSString *)title image:(nullable UIImage *)image tag:(NSInteger)tag;
- (instancetype)initWithTitle:(nullable NSString *)title image:(nullable UIImage *)image selectedImage:(nullable UIImage *)selectedImage NS_AVAILABLE_IOS(7_0);
- (instancetype)initWithTabBarSystemItem:(UITabBarSystemItem)systemItem tag:(NSInteger)tag;

//選中的圖片
@property(nullable, nonatomic,strong) UIImage *selectedImage NS_AVAILABLE_IOS(7_0);
//角標(biāo)
@property(nullable, nonatomic, copy) NSString *badgeValue;

//_____________________________________________IOS7廢除的方法(忽略)______________________________________________________
- (void)setFinishedSelectedImage:(nullable UIImage *)selectedImage withFinishedUnselectedImage:(nullable UIImage *)unselectedImage NS_DEPRECATED_IOS(5_0,7_0);
- (nullable UIImage *)finishedSelectedImage NS_DEPRECATED_IOS(5_0,7_0);
- (nullable UIImage *)finishedUnselectedImage NS_DEPRECATED_IOS(5_0,7_0);
//___________________________________________________________________________________________________

//title的偏移量
@property (nonatomic, readwrite, assign) UIOffset titlePositionAdjustment;
//_____________________________________________IOS10新增的屬性和方法______________________________________________________
//角標(biāo)顏色
@property (nonatomic, readwrite, copy, nullable) UIColor *badgeColor;
//角標(biāo)設(shè)置富文本
- (void)setBadgeTextAttributes:(nullable NSDictionary<NSString *,id> *)textAttributes forState:(UIControlState)state;
- (nullable NSDictionary<NSString *,id> *)badgeTextAttributesForState:(UIControlState)state;
//___________________________________________________________________________________________________
@end

總結(jié)

  1. 通過初始化方法可以設(shè)置title锌畸,image勇劣,selectedImage等展示的元素。
  2. badgeValue屬性可以在Item的右上角顯示一個數(shù)字角標(biāo)潭枣。
  3. titlePositionAdjustment設(shè)置文字的偏移量
  4. iOS10之后比默,可以通過badgeColor,setBadgeTextAttributes等設(shè)置角標(biāo)的背景色盆犁,富文本的角標(biāo)數(shù)值命咐。
  5. 當(dāng)UITabBar上的UITabBarItem>=6個時,底部的UITabBar左側(cè)會顯示一個More的Item谐岁。

2.UITabBar

一個UITabBarController只有一個TabBar醋奠。

NS_CLASS_AVAILABLE_IOS(2_0) @interface UITabBar : UIView
//代理
@property(nullable, nonatomic, weak) id<UITabBarDelegate> delegate;
//get/set UITabBarItems 默認(rèn)是nil 改變時沒有動畫效果 按順序展示
@property(nullable, nonatomic, copy) NSArray<UITabBarItem *> *items;
//選中的item
@property(nullable, nonatomic, weak) UITabBarItem *selectedItem;
//設(shè)置Items
- (void)setItems:(nullable NSArray<UITabBarItem *> *)items animated:(BOOL)animated;

- (void)beginCustomizingItems:(NSArray<UITabBarItem *> *)items;
- (BOOL)endCustomizingAnimated:(BOOL)animated;
#if UIKIT_DEFINE_AS_PROPERTIES
@property(nonatomic, readonly, getter=isCustomizing) BOOL customizing;
#else
- (BOOL)isCustomizing;
#endif

//ios7之后,
@property(null_resettable, nonatomic, strong) UIColor *tintColor ;
@property(nullable, nonatomic, strong) UIColor *barTintColor ;

//未選中的Item的顏色 IOS10可用
@property (nonatomic, readwrite, copy, nullable) UIColor *unselectedItemTintColor;

//ios8后廢除 使用tintColor
@property(nullable, nonatomic, strong) UIColor *selectedImageTintColor(5_0,8_0);

//背景圖片
@property(nullable, nonatomic, strong) UIImage *backgroundImage;
@property(nullable, nonatomic, strong) UIImage *selectionIndicatorImage;
@property(nullable, nonatomic, strong) UIImage *shadowImage;

//item位置
@property(nonatomic) UITabBarItemPositioning itemPositioning;
//item寬度
@property(nonatomic) CGFloat itemWidth;
//item間隙
@property(nonatomic) CGFloat itemSpacing;
//樣式
@property(nonatomic) UIBarStyle barStyle;
//半透明
@property(nonatomic,getter=isTranslucent) BOOL translucent NS_AVAILABLE_IOS(7_0);
@end

以下是tabBar的代理方法,操作TabBar時的回調(diào)伊佃,主要是對UITabBarItem的操作

@protocol UITabBarDelegate<NSObject>
@optional
/**
 用戶選中某個UITabBarItem
 */
- (void)tabBar:(UITabBar *)tabBar didSelectItem:(UITabBarItem *)item;
//此四個代理方法是當(dāng)Items>=6個時钝域,當(dāng)進入More頁面時,開始或結(jié)束Item編輯狀態(tài)的相關(guān)回調(diào)
- (void)tabBar:(UITabBar *)tabBar willBeginCustomizingItems:(NSArray<UITabBarItem *> *)items;                    
- (void)tabBar:(UITabBar *)tabBar didBeginCustomizingItems:(NSArray<UITabBarItem *> *)items; 
- (void)tabBar:(UITabBar *)tabBar willEndCustomizingItems:(NSArray<UITabBarItem *> *)items changed:(BOOL)changed; 
- (void)tabBar:(UITabBar *)tabBar didEndCustomizingItems:(NSArray<UITabBarItem *> *)items changed:(BOOL)changed;
@end

3.UITabBarController

NS_CLASS_AVAILABLE_IOS(2_0) @interface UITabBarController : UIViewController <UITabBarDelegate, NSCoding>
//視圖控制器數(shù)組锭魔,這個屬性被賦值時,customizableViewControllers屬性的值與之一樣路呜。
@property(nullable, nonatomic,copy) NSArray<__kindof UIViewController *> *viewControllers;
- (void)setViewControllers:(NSArray<__kindof UIViewController *> * __nullable)viewControllers animated:(BOOL)animated;

//當(dāng)前選中的視圖控制器
@property(nullable, nonatomic, assign) __kindof UIViewController *selectedViewController; 
//與selectedViewController對應(yīng)
@property(nonatomic) NSUInteger selectedIndex;

//當(dāng)ViewController的數(shù)量>=6迷捧,TabBar會出現(xiàn)一個moreNavigationController管理多余的viewcontroller。readonly屬性
@property(nonatomic, readonly) UINavigationController *moreNavigationController;

//當(dāng)viewcontroller>=6時,moreNavigationController右上方會有個edit按鈕胀葱,支持通過拖拽修改ViewController的順序漠秋,如若要屏蔽該功能,customizableViewControllers設(shè)置為nil即可抵屿。
@property(nullable, nonatomic, copy) NSArray<__kindof UIViewController *> *customizableViewControllers;

//只讀屬性庆锦,為了配置UITabBarItem,應(yīng)該去修改ViewControllers屬性轧葛。
@property(nonatomic,readonly) UITabBar *tabBar;

//協(xié)議
@property(nullable, nonatomic,weak) id<UITabBarControllerDelegate> delegate;

@end
//should選中viewController  return YES 可以本選中搂抒, NO不可以被選中
- (BOOL)tabBarController:(UITabBarController *)tabBarController shouldSelectViewController:(UIViewController *)viewController;

// 選中viewController后執(zhí)行的Action
- (void)tabBarController:(UITabBarController *)tabBarController didSelectViewController:(UIViewController *)viewController;

// 將要處于編輯狀態(tài)(即點擊MoreNavigationController的edit按鈕)
- (void)tabBarController:(UITabBarController *)tabBarController willBeginCustomizingViewControllers:(NSArray<__kindof UIViewController *> *)viewControllers;

// MoreNavigationController will結(jié)束編輯狀態(tài)
- (void)tabBarController:(UITabBarController *)tabBarController willEndCustomizingViewControllers:(NSArray<__kindof UIViewController *> *)viewControllers changed:(BOOL)changed;

//MoreNavigationController did結(jié)束編輯狀態(tài)
- (void)tabBarController:(UITabBarController *)tabBarController didEndCustomizingViewControllers:(NSArray<__kindof UIViewController *> *)viewControllers changed:(BOOL)changed;

//UITabBarController支持的界面方向
- (UIInterfaceOrientationMask)tabBarControllerSupportedInterfaceOrientations:(UITabBarController *)tabBarController;

//對于將要展示的tabBarController 優(yōu)先選擇屏幕方向
- (UIInterfaceOrientation)tabBarControllerPreferredInterfaceOrientationForPresentation:(UITabBarController *)tabBarController;

//自定義轉(zhuǎn)場動畫
- (nullable id <UIViewControllerInteractiveTransitioning>)tabBarController:(UITabBarController *)tabBarController
                      interactionControllerForAnimationController: (id <UIViewControllerAnimatedTransitioning>)animationController;
- (nullable id <UIViewControllerAnimatedTransitioning>)tabBarController:(UITabBarController *)tabBarController
            animationControllerForTransitionFromViewController:(UIViewController *)fromVC
                                              toViewController:(UIViewController *)toVC;

@end
@interface UIViewController (UITabBarControllerItem)
@property(null_resettable, nonatomic, strong) UITabBarItem *tabBarItem;
@property(nullable, nonatomic, readonly, strong) UITabBarController *tabBarController;
@end

3艇搀、實際開發(fā)中的相關(guān)問題

1. UITabBar的背景顏色

1.直接設(shè)置背景顏色
//    self.tabBar.backgroundColor = [UIColor orangeColor];
//    [[UITabBar appearance] setBackgroundColor:[UIColor orangeColor]];
    [[UITabBar appearance]setBarTintColor:[UIColor orangeColor]];
    [UITabBar appearance].translucent = NO

注意:
1.前兩種設(shè)置背景顏色的方法是無效的。
2.tabBar是一個readonly屬性求晶,這里不能使用TabBarController.tabBar.barTintColor設(shè)置背景色焰雕。
前兩種方法是無效的。
3.這里設(shè)置tabBar的半透明屬性translucent設(shè)置為NO芳杏,默認(rèn)為YES,若保留半透明效果矩屁,設(shè)置的顏色會與正常的顏色有色差;

2.添加一個有顏色的View
    UIView * view = [UIView new];
    view.backgroundColor = [UIColor orangeColor];
    view.frame = self.tabBar.bounds;
    [[UITabBar appearance] insertSubview:view atIndex:0];
3.使用背景圖片
[[UITabBar appearance] setBackgroundImage:[UIImage imageNamed:@"tabBarBackgroundImage"]];
[UITabBar appearance].translucent = NO;

這里同樣需要設(shè)置translucent為NO

2. UITabBar的頂部的shadowImage

[[UITabBar appearance] setShadowImage:[UIImage new]];
[[UITabBar appearance] setBackgroundImage:[[UIImage alloc]init]];

UINavigationBar一樣,需同時設(shè)置ShadowImageBackgroundImage才能生效爵赵。

3.tabBarItem選中時的背景色

 //    [UITabBar appearance].selectionIndicatorImage = [self drawTabBarItemBackgroundImageWithSize:size];
self.tabBar.selectionIndicatorImage = [self drawTabBarItemBackgroundImageWithSize:size];

此處可直接獲取TabBarControllertabBar直接設(shè)置selectionIndicatorImage吝秕。
獲取某背景顏色的image的方法可參考IOS-UINavigationController詳解相關(guān)方法,注意圖片size要設(shè)置正確空幻。

4.修改tabBarItem的文字烁峭、圖片顏色

tabBarItem,默認(rèn)狀態(tài)下氛悬,選中狀態(tài)是藍色则剃,未選中狀態(tài)下是灰色。選中狀態(tài)的Item的文字和圖片顏色可直接通過tintColor屬性修改

self.tabBar.tintColor = [UIColor yellowColor];

現(xiàn)實中如捅,往往都是現(xiàn)實圖片實際的顏色棍现。我們可以通過自定義一個tabBarItem繼承UITabBarItem,在initWithCoder方法添加以下代碼完成現(xiàn)實圖片實際顏色的效果镜遣。

-(id)initWithTitle:(NSString *)title image:(UIImage *)image selectedImage:(UIImage *)selectedImage{
    if (self = [super initWithTitle:title image:image selectedImage:selectedImage]) {
        self.image = [self.image imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal];
        self.selectedImage = [self.selectedImage imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal];
        
        [self setTitleTextAttributes:@{NSForegroundColorAttributeName : [self mostColorWithImage:self.image]}
                            forState:UIControlStateNormal];
        
        [self setTitleTextAttributes:@{NSForegroundColorAttributeName : [self mostColorWithImage:self.selectedImage]}
                            forState:UIControlStateSelected];
    }
    return self;
}

-(UIColor*)mostColorWithImage:(UIImage *)image
{
#if __IPHONE_OS_VERSION_MAX_ALLOWED > __IPHONE_6_1
    int bitmapInfo = kCGBitmapByteOrderDefault | kCGImageAlphaPremultipliedLast;
#else
    int bitmapInfo = kCGImageAlphaPremultipliedLast;
#endif
    
    //第一步 先把圖片縮小 加快計算速度. 但越小結(jié)果誤差可能越大
    CGSize thumbSize=CGSizeMake(50, 50);
    
    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
    CGContextRef context = CGBitmapContextCreate(NULL,
                                                 thumbSize.width,
                                                 thumbSize.height,
                                                 8,//bits per component
                                                 thumbSize.width*4,
                                                 colorSpace,
                                                 bitmapInfo);
    
    CGRect drawRect = CGRectMake(0, 0, thumbSize.width, thumbSize.height);
    CGContextDrawImage(context, drawRect, image.CGImage);
    CGColorSpaceRelease(colorSpace);
    
    
    //第二步 取每個點的像素值
    unsigned char* data = CGBitmapContextGetData (context);
    
    if (data == NULL) return nil;
    
    NSCountedSet *cls = [NSCountedSet setWithCapacity:thumbSize.width*thumbSize.height];
    
    for (int x=0; x<thumbSize.width; x++) {
        for (int y=0; y<thumbSize.height; y++) {
            
            int offset = 4*(x*y);
            
            int red = data[offset];
            int green = data[offset+1];
            int blue = data[offset+2];
            int alpha =  data[offset+3];
            
            if (alpha != 255) continue;
            
            NSArray *clr=@[@(red),@(green),@(blue),@(alpha)];
            [cls addObject:clr];
            
        }
    }
    CGContextRelease(context);
    
    
    //第三步 找到出現(xiàn)次數(shù)最多的那個顏色
    NSEnumerator *enumerator = [cls objectEnumerator];
    NSArray *curColor = nil;
    
    NSArray *MaxColor=nil;
    NSUInteger MaxCount=0;
    
    while ( (curColor = [enumerator nextObject]) != nil )
    {
        NSUInteger tmpCount = [cls countForObject:curColor];
        
        if ( tmpCount < MaxCount ) continue;
        
        MaxCount=tmpCount;
        MaxColor=curColor;
    }
    
    return [UIColor colorWithRed:([MaxColor[0] intValue]/255.0f) green:([MaxColor[1] intValue]/255.0f) blue:([MaxColor[2] intValue]/255.0f) alpha:1.0 ];
}

iOS10之后新增了unselectedItemTintColor設(shè)置未選中狀態(tài)下的item顏色己肮。配合tintColor可以達到我們需要的大部分效果。

5.修改TabBar的高度

UITabBarController里重寫viewWillLayoutSubviews

- (void)viewWillLayoutSubviews {
    CGRect tabFrame = self.tabBar.frame;
    tabFrame.size.height = 59;
    tabFrame.origin.y = self.view.bounds.size.height - 59;
    self.tabBar.frame = tabFrame;
}

然后通過UITabBarItemimageInsetstitlePositionAdjustment屬性調(diào)整圖片和文字的位置

for (UITabBarItem * item in self.tabBar.items) {
        item.imageInsets = UIEdgeInsetsMake(8, 0, -8, 0);
    }
[[UITabBarItem appearance]setTitlePositionAdjustment:UIOffsetMake(0, -8)];

6. 自定義UITabBarController

具體內(nèi)容參考:仿閑魚自定義Tabbar(純代碼)

參考文章
iOS UI Tab開發(fā)
UITabBarController的使用
UITabBarController悲关、TabBar背景顏色設(shè)置谎僻、TabBarItem顏色處理

結(jié)束語:

此文是我在學(xué)習(xí)過程中的探索與總結(jié),不足之處還望大家見諒寓辱,并歡迎指正艘绍!
愿每一個人都能學(xué)有所成!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末秫筏,一起剝皮案震驚了整個濱河市诱鞠,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌这敬,老刑警劉巖航夺,帶你破解...
    沈念sama閱讀 206,723評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異崔涂,居然都是意外死亡阳掐,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,485評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來缭保,“玉大人汛闸,你說我怎么就攤上這事′潭恚” “怎么了蛉拙?”我有些...
    開封第一講書人閱讀 152,998評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長彻亲。 經(jīng)常有香客問我孕锄,道長,這世上最難降的妖魔是什么苞尝? 我笑而不...
    開封第一講書人閱讀 55,323評論 1 279
  • 正文 為了忘掉前任畸肆,我火速辦了婚禮,結(jié)果婚禮上宙址,老公的妹妹穿的比我還像新娘轴脐。我一直安慰自己,他們只是感情好抡砂,可當(dāng)我...
    茶點故事閱讀 64,355評論 5 374
  • 文/花漫 我一把揭開白布大咱。 她就那樣靜靜地躺著,像睡著了一般注益。 火紅的嫁衣襯著肌膚如雪碴巾。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,079評論 1 285
  • 那天丑搔,我揣著相機與錄音厦瓢,去河邊找鬼。 笑死啤月,一個胖子當(dāng)著我的面吹牛煮仇,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播谎仲,決...
    沈念sama閱讀 38,389評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼浙垫,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了郑诺?” 一聲冷哼從身側(cè)響起绞呈,我...
    開封第一講書人閱讀 37,019評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎间景,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體艺智,經(jīng)...
    沈念sama閱讀 43,519評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡倘要,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,971評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片封拧。...
    茶點故事閱讀 38,100評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡志鹃,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出泽西,到底是詐尸還是另有隱情曹铃,我是刑警寧澤,帶...
    沈念sama閱讀 33,738評論 4 324
  • 正文 年R本政府宣布捧杉,位于F島的核電站陕见,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏味抖。R本人自食惡果不足惜评甜,卻給世界環(huán)境...
    茶點故事閱讀 39,293評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望仔涩。 院中可真熱鬧忍坷,春花似錦、人聲如沸熔脂。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,289評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽霞揉。三九已至旬薯,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間零聚,已是汗流浹背袍暴。 一陣腳步聲響...
    開封第一講書人閱讀 31,517評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留隶症,地道東北人政模。 一個月前我還...
    沈念sama閱讀 45,547評論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像蚂会,于是被迫代替她去往敵國和親淋样。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,834評論 2 345

推薦閱讀更多精彩內(nèi)容