UITabBarController 虽然很强大,基本上也能满足常见的需求,但是用起来总没那么畅快。有时候,总有一些变态的需求,需要自定义。
之前也看过一些别人自定义的TabBarController,但是都不尽理想,准确的说,很多自定义的都是继承自UITabBarController的即是半自定义的。根本谈不上真正意义上的自定义。
今天就分析一个我认为比较好的自定义TabBarController.
一,先来看一下TabBarController的基本结构
一般说来TabBarController,下面有一个TabBar, TabBar是一个视图,继承自UIView.
TabBar这个视图里面又有多个选项TabBarItem,这些个Item,是一个可以点击,并且可以切换到对应的控制器页面的按键,同时,这些按键还有自己本身的属性设置。所以Item应该继承自UIControl. 所以结构就很清楚了:分3个部分。
- TabBarController (继承自UIViewController)
- TabBar (继承自UIView)
- TabBarItem (继承自UIControl)
二,分别来封装和分析各个部分
1. 先来看TabBarItem
tabBarItem,直观的显示,就是我们看到那几个点击来回切换的按键。当点击每一个选项的时候,选项就会有相应的变化,同时切换到对应的控制器界面。
有哪些属性和变化呢?
a. 本身基本属性:有选中/未选中图片、选中/未选中文字及颜色、选中/未选中背景色、有宽高大小位置偏移(供微调用);
b. 徽标显示:徽标值及其颜色大小、徽标图片、背景色
#import <UIKit/UIKit.h> @interface RDVTabBarItem : UIControl /** * itemHeight is an optional property. When set it is used instead of tabBar's height. */ @property CGFloat itemHeight; #pragma mark - Title configuration /** * The title displayed by the tab bar item. */ @property (nonatomic, copy) NSString *title; /** * The offset for the rectangle around the tab bar item's title. */ @property (nonatomic) UIOffset titlePositionAdjustment; /** * For title's text attributes see * https://developer.apple.com/library/ios/documentation/uikit/reference/NSString_UIKit_Additions/Reference/Reference.html */ /** * The title attributes dictionary used for tab bar item's unselected state. */ @property (copy) NSDictionary *unselectedTitleAttributes; /** * The title attributes dictionary used for tab bar item's selected state. */ @property (copy) NSDictionary *selectedTitleAttributes; #pragma mark - Image configuration /** * The offset for the rectangle around the tab bar item's image. */ @property (nonatomic) UIOffset imagePositionAdjustment; /** * The image used for tab bar item's selected state. */ - (UIImage *)finishedSelectedImage; /** * The image used for tab bar item's unselected state. */ - (UIImage *)finishedUnselectedImage; /** * Sets the tab bar item's selected and unselected images. */ - (void)setFinishedSelectedImage:(UIImage *)selectedImage withFinishedUnselectedImage:(UIImage *)unselectedImage; #pragma mark - Background configuration /** * The background image used for tab bar item's selected state. */ - (UIImage *)backgroundSelectedImage; /** * The background image used for tab bar item's unselected state. */ - (UIImage *)backgroundUnselectedImage; /** * Sets the tab bar item's selected and unselected background images. */ - (void)setBackgroundSelectedImage:(UIImage *)selectedImage withUnselectedImage:(UIImage *)unselectedImage; #pragma mark - Badge configuration /** * Text that is displayed in the upper-right corner of the item with a surrounding background. */ @property (nonatomic, copy) NSString *badgeValue; /** * Image used for background of badge. */ @property (strong) UIImage *badgeBackgroundImage; /** * Color used for badge's background. */ @property (strong) UIColor *badgeBackgroundColor; /** * Color used for badge's text. */ @property (strong) UIColor *badgeTextColor; /** * The offset for the rectangle around the tab bar item's badge. */ @property (nonatomic) UIOffset badgePositionAdjustment; /** * Font used for badge's text. */ @property (nonatomic) UIFont *badgeTextFont; @end
2. 再来看TabBar
tabBar是一个视图,自然有视图的基本属性,里面有多个tabBarItem,所以必然有items个数的属性;有事件响应,所以有代理抛出接口
#import <UIKit/UIKit.h> @class RDVTabBar, RDVTabBarItem; @protocol RDVTabBarDelegate <NSObject> /** * Asks the delegate if the specified tab bar item should be selected. */ - (BOOL)tabBar:(RDVTabBar *)tabBar shouldSelectItemAtIndex:(NSInteger)index; /** * Tells the delegate that the specified tab bar item is now selected. */ - (void)tabBar:(RDVTabBar *)tabBar didSelectItemAtIndex:(NSInteger)index; @end @interface RDVTabBar : UIView /** * The tab bar’s delegate object. */ @property (nonatomic, weak) id <RDVTabBarDelegate> delegate; /** * The items displayed on the tab bar. */ @property (nonatomic, copy) NSArray *items; /** * The currently selected item on the tab bar. */ @property (nonatomic, weak) RDVTabBarItem *selectedItem; /** * backgroundView stays behind tabBar's items. If you want to add additional views, * add them as subviews of backgroundView. */ @property (nonatomic, readonly) UIView *backgroundView; /* * contentEdgeInsets can be used to center the items in the middle of the tabBar. */ @property UIEdgeInsets contentEdgeInsets; /** * Sets the height of tab bar. */ - (void)setHeight:(CGFloat)height; /** * Returns the minimum height of tab bar's items. */ - (CGFloat)minimumContentHeight; /* * Enable or disable tabBar translucency. Default is NO. */ @property (nonatomic, getter=isTranslucent) BOOL translucent; @end
3. 最后来看TabBarController
TabBarController 里面有TabBar, 有默认选项,tabBarItem对应的View Controllers及其之间的关联。还有一些代理方法。故而
#import <UIKit/UIKit.h> #import "RDVTabBar.h" @protocol RDVTabBarControllerDelegate; @interface RDVTabBarController : UIViewController <RDVTabBarDelegate> /** * The tab bar controller’s delegate object. */ @property (nonatomic, weak) id<RDVTabBarControllerDelegate> delegate; /** * An array of the root view controllers displayed by the tab bar interface. */ @property (nonatomic, copy) IBOutletCollection(UIViewController) NSArray *viewControllers; /** * The tab bar view associated with this controller. (read-only) */ @property (nonatomic, readonly) RDVTabBar *tabBar; /** * The view controller associated with the currently selected tab item. */ @property (nonatomic, weak) UIViewController *selectedViewController; /** * The index of the view controller associated with the currently selected tab item. */ @property (nonatomic) NSUInteger selectedIndex; /** * A Boolean value that determines whether the tab bar is hidden. */ @property (nonatomic, getter=isTabBarHidden) BOOL tabBarHidden; /** * Changes the visibility of the tab bar. */ - (void)setTabBarHidden:(BOOL)hidden animated:(BOOL)animated; @end @protocol RDVTabBarControllerDelegate <NSObject> @optional /** * Asks the delegate whether the specified view controller should be made active. */ - (BOOL)tabBarController:(RDVTabBarController *)tabBarController shouldSelectViewController:(UIViewController *)viewController; /** * Tells the delegate that the user selected an item in the tab bar. */ - (void)tabBarController:(RDVTabBarController *)tabBarController didSelectViewController:(UIViewController *)viewController; @end @interface UIViewController (RDVTabBarControllerItem) /** * The tab bar item that represents the view controller when added to a tab bar controller. */ @property(nonatomic, setter = rdv_setTabBarItem:) RDVTabBarItem *rdv_tabBarItem; /** * The nearest ancestor in the view controller hierarchy that is a tab bar controller. (read-only) */ @property(nonatomic, readonly) RDVTabBarController *rdv_tabBarController; @end
重点分析一下, 当点击TabBarItem的时候,如何关联控制器的。
其实就是通过导航控制器来切换到对应编号的视图控制器。请看核心代码
- (BOOL)tabBar:(RDVTabBar *)tabBar shouldSelectItemAtIndex:(NSInteger)index { if ([[self delegate] respondsToSelector:@selector(tabBarController:shouldSelectViewController:)]) { if (![[self delegate] tabBarController:self shouldSelectViewController:[self viewControllers][index]]) { return NO; } } if ([self selectedViewController] == [self viewControllers][index]) { if ([[self selectedViewController] isKindOfClass:[UINavigationController class]]) { UINavigationController *selectedController = (UINavigationController *)[self selectedViewController]; if ([selectedController topViewController] != [selectedController viewControllers][0]) { [selectedController popToRootViewControllerAnimated:YES]; } } return NO; } return YES; }
三,代码部分
限于篇幅,不直接贴出代码,
直接获取源码Demo:https://github.com/robbdimitrov/RDVTabBarController