iOS-自定義帶抽屜效果的tabBar

demo地址:gitHub

一废酷、先來個效果

tabBar.gif

二、代碼示例

1.抽屜頁作為根視圖:

@interface DrawerViewController ()
{
    UITapGestureRecognizer *tapGesture;
}

//創(chuàng)建左邊的抽屜
@property (nonatomic, strong) LeftViewController *leftViewController;

//創(chuàng)建右邊的標(biāo)簽控制器
@property (nonatomic, strong) MTabBarViewController *mainViewController;
//抽屜是否顯示的標(biāo)示

@property (nonatomic, assign) BOOL isOpen;
@end

@implementation DrawerViewController

- (void)dealloc
{
    //移除通知
    [[NSNotificationCenter defaultCenter]removeObserver:self name:@"buttonTap" object:nil];
}
- (void)viewDidLoad {
    [super viewDidLoad];
    
    [self createTabBarController];
    [self createLeftVc];
    
    self.view.backgroundColor = [UIColor grayColor];
    
    //添加通知抹缕,監(jiān)聽TabBar的點(diǎn)擊事件 隱藏左邊抽屜
    [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(buttonTap) name:@"buttonTap" object:nil];
    
}
- (void)buttonTap
{
    //抽屜展開時則隱藏
    if (self.isOpen == YES) {
        [self openOrHidden];
    }
    
}
//創(chuàng)建左邊抽屜
- (void)createLeftVc
{
    self.leftViewController = [[LeftViewController alloc]init];
    
    //抽屜控制器添加到父控制器中
    [self addChildViewController:self.leftViewController];
    self.leftViewController.view.frame = LeftViewStartFrame();
    [self.view addSubview:self.leftViewController.view];
    [self.leftViewController didMoveToParentViewController:self];
    
}
//創(chuàng)建右邊的標(biāo)簽控制器
- (void)createTabBarController{
    //
    NSArray *classNames = @[@"ProductViewController",@"MessageViewController",@"OrderViewController"];
    
    //保存viewControllers
    NSMutableArray *viewControllers = [NSMutableArray array];
    [classNames enumerateObjectsUsingBlock:^(NSString *  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        //先把字符串轉(zhuǎn)化為類名
        Class class = NSClassFromString(obj);
        
        //創(chuàng)建ViewController
        UIViewController *vc = [[class alloc]init];
        UINavigationController *aNav = [[UINavigationController alloc]initWithRootViewController:vc];
        
        [viewControllers addObject:aNav];
        
    }];
    //TabBarController 創(chuàng)建
    _mainViewController = [[MTabBarViewController alloc]initWithViewControllers:viewControllers];
    _mainViewController.view.backgroundColor = [UIColor brownColor];
    
    //添加標(biāo)簽控制器到父控制器
    [self addChildViewController:self.mainViewController];
    self.mainViewController.view.frame = BOUNDS;
    [self.view addSubview:self.mainViewController.view];
    [self.mainViewController didMoveToParentViewController:self];
}

//標(biāo)簽控制器顯示的vc的根控制器的view往右邊移動
//tabBar -> Nav (ViewControllers[selectedIndex]) -> Nav.rootViewController.view
- (void)tabBar_Nav_RootViewController_viewMoveRight
{
    //標(biāo)簽控制器當(dāng)中 當(dāng)前顯示的控制器
    UINavigationController *nav = self.mainViewController.viewControllers[self.mainViewController.selectedIndex];
    //取出導(dǎo)航控制器的根控制器
    UIViewController *rootVc = nav.childViewControllers[0];
    rootVc.view.frame = RightContentEndFrame();
}
- (void)tabBar_Nav_RootViewController_viewMoveLeft
{
    //標(biāo)簽控制器當(dāng)中 當(dāng)前顯示的控制器
    UINavigationController *nav = self.mainViewController.viewControllers[self.mainViewController.selectedIndex];
    //取出導(dǎo)航控制器的根控制器
    UIViewController *rootVc = nav.childViewControllers[0];
    rootVc.view.frame = RigntContentStartFrame();
}


//顯示左邊抽屜
- (void)open
{
    [UIView animateWithDuration:0.48 animations:^{
        self.leftViewController.view.frame = LeftViewEndFrame();
        [self tabBar_Nav_RootViewController_viewMoveRight];
    } completion:nil];
    
    //添加點(diǎn)擊手勢澈蟆,點(diǎn)擊某些區(qū)域的隱藏抽屜
    tapGesture = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(tap:)];
    tapGesture.numberOfTapsRequired = 1;
    tapGesture.numberOfTouchesRequired = 1;
    [self.view addGestureRecognizer:tapGesture];
}
- (void)tap:(UITapGestureRecognizer *)gesturer
{
    //獲取點(diǎn)擊的位置
    CGPoint point = [gesturer locationInView:self.view];
    if (CGRectContainsPoint(self.leftViewController.view.frame, point) == YES) {
        return;
    }
    
    [self hidden];
    self.isOpen = NO;
    //移除手勢
    [self.view removeGestureRecognizer:tapGesture];
}
//隱藏左邊抽屜
- (void)hidden
{
    [UIView animateWithDuration:0.48 animations:^{
        self.leftViewController.view.frame = LeftViewStartFrame();
        [self tabBar_Nav_RootViewController_viewMoveLeft];
    } completion:nil];
    
}

- (void)openOrHidden
{
    //當(dāng)前如果是隱藏,則顯示
    if (self.isOpen == NO) {
        [self open];
    }
    
    //當(dāng)前如果是顯示的卓研,則隱藏
    if (self.isOpen == YES) {
        [self hidden];
    }
    
    //改變隱藏標(biāo)記
    self.isOpen = !self.isOpen;
}
@end

2.標(biāo)簽視圖 修改方法 - (void)selectBtn:(UIButton *)sender中的切換效果可以實(shí)現(xiàn)不同的切換動畫與效果趴俘。

@implementation MTabBar
- (instancetype)initWithTitles:(NSArray *)titles imageNames:(NSArray *)imageNames
{
    self = [super initWithFrame:TabBarFrame()];
    if (self) {
        
        //標(biāo)題數(shù)組不為空,圖片名字個數(shù) = 標(biāo)題個數(shù)
        self.buttonBack = [[UIView alloc]initWithFrame:CGRM(0, 0, BUTTON_W, 64)];
        self.buttonBack.backgroundColor = BUTTON_BACK_COLOR;
        [self addSubview:self.buttonBack];
        
        self.backgroundColor = TABBAR_BACK_COLOR;
        
        if ([titles count] && [titles count] == [imageNames count]) {
            [titles enumerateObjectsUsingBlock:^(NSString *  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
                UIButton *button = [[UIButton alloc]initWithFrame:CGRM(BUTTON_W*idx, 0, BUTTON_W, 64)];
                button.tag = 1000 + idx;
                [self addSubview:button];
                //默認(rèn)選中第一個
                if (idx == 0) {
                    button.selected = YES;
                    self.selectedButton = button;
                }
                
                [button addSubview:MakeLabel(CGRM(0, 30, BUTTON_W, 34), obj)];
//                // 圖片寬 高分別為 44 24
                CGFloat x = (BUTTON_W - 44)/2;
                [button addSubview:MakeImageView(CGRM(x, 5, 44, 24),[imageNames objectAtIndex:idx])];
                //添加點(diǎn)擊方法
                [button addTarget:self action:@selector(selectBtn:) forControlEvents:UIControlEventTouchUpInside];
            }];
        }
    }
    return self;
}
//button 點(diǎn)擊方法
- (void)selectBtn:(UIButton *)sender
{
    //讓抽屜隱藏鉴分,發(fā)出通知
    [[NSNotificationCenter defaultCenter]postNotificationName:@"buttonTap" object:nil];
    
    //選中的button 已經(jīng)是選中狀態(tài) 不用處理
    if (self.selectedButton == sender) {
        return;
    }
    //改變之前選中button的狀態(tài)  為非選中狀態(tài)
    self.selectedButton.selected = NO;
    
    //改變當(dāng)前選中button的狀態(tài)
    sender.selected = YES;
    self.selectedButton = sender;
    //通知標(biāo)簽控制器顯示當(dāng)前button對應(yīng)的viewController
    if (self.callBack) {
        self.callBack(sender.tag - 1000);
    }
    [UIView animateWithDuration:0.5 animations:^{
        self.buttonBack.center = CGPointMake(BUTTON_W/2+(sender.tag - 1000) * BUTTON_W, 32);
    } completion:nil];
}
UILabel *MakeLabel(CGRect frame, NSString *title)
{
    UILabel *label = [[UILabel alloc]initWithFrame:frame];
//    label.userInteractionEnabled = YES;
    label.backgroundColor = [UIColor clearColor];
    label.textColor = [UIColor whiteColor];
    label.textAlignment = NSTextAlignmentCenter;
    label.text = title;
    return label;
}
//根據(jù) frame和 imageName 創(chuàng)建UIImageView
UIImageView *MakeImageView(CGRect frame, NSString *imageName)
{
    UIImageView *imageV = [[UIImageView alloc]initWithFrame:frame];
//    imageV.userInteractionEnabled = YES;
    imageV.backgroundColor = [UIColor clearColor];
    imageV.image = [UIImage imageNamed:imageName];
    return imageV;
}
@end

3.標(biāo)簽控制器

@interface MTabBarViewController ()

@end

@implementation MTabBarViewController

- (instancetype)initWithViewControllers:(NSArray *)viewControllers
{
    self = [super init];
    if (self) {
        
        _viewControllers = viewControllers;
        //遍歷數(shù)組哮幢,添加子控制器
        [_viewControllers enumerateObjectsUsingBlock:^(UIViewController *  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
            [self addChild:obj];
        }];
        /*
         @[@"商品.png",@"消息.png",@"訂單.png"]
         @[@"商品瀏覽",@"我的消息",@"我的訂單"]
         */
        _tabBar = [[MTabBar alloc]initWithTitles:@[@"商品瀏覽",@"我的消息",@"我的訂單"] imageNames:@[@"商品.png",@"消息.png",@"訂單.png"]];
        [self.view addSubview:self.tabBar];
        __weak typeof(self) weakSelf = self;
        self.tabBar.callBack = ^(NSInteger index){
            weakSelf.selectedIndex = index;
        };
        //默認(rèn)選中第0個
        self.selectedIndex = 0;
    }
    return self;
}
- (void)setSelectedIndex:(NSInteger)selectedIndex
{
    //取出當(dāng)前控制器 oldVc
    UIViewController *oldVc = self.viewControllers[self.selectedIndex];
    //取出將要顯示的 控制器 newVc
    UIViewController *newVc = self.viewControllers[selectedIndex];
    //動畫 向左邊移動
    newVc.view.frame = CGRectMake(S_W, 0, S_W, S_H);
    //改變 newVc 的視圖層次
    
    //把newVc.view 的視圖放在 self.tabBar 的下面
    [self.view insertSubview:newVc.view belowSubview:self.tabBar];
    [UIView animateWithDuration:0.5 animations:^{
        oldVc.view.frame = CGRectMake(-S_W, 0, S_W, S_H);
        newVc.view.frame = BOUNDS;
    }];
    _selectedIndex = selectedIndex;
}

//添加子控制器具體步驟
- (void)addChild:(UIViewController *)viewController
{
    [self addChildViewController:viewController];
    viewController.view.frame = BOUNDS;
    //將viewContoller.view 放在最底層
    [self.view insertSubview:viewController.view atIndex:0];
    [viewController didMoveToParentViewController:self];
    
//    self.view.subviews 數(shù)組 下標(biāo)越小带膀,視圖層次越在下面志珍,下標(biāo)越大,視圖層次越在上面
}
@end

4.視圖位置控制 修改對應(yīng)的視圖的frame可以實(shí)現(xiàn)不同的視圖效果

CGRect TabBarFrame()
{
    return CGRectMake(0, S_H-64, S_W, 64);
}

//左邊抽屜隱藏(開始)的位置
CGRect LeftViewStartFrame()
{
    return CGRectMake(-S_W*0.75, 67,S_W*0.75 , S_H-64-64-6);
}
//左邊抽屜顯示(結(jié)束)的位置
CGRect LeftViewEndFrame()
{
    return CGRectMake(0, 67, S_W*0.75, S_H-64-64-6);
}

//右邊內(nèi)容開始(抽屜隱藏時)的位置
CGRect RigntContentStartFrame()
{
    return CGRectMake(0, 0, S_W, S_H);
}

//右邊內(nèi)容結(jié)束(抽屜顯示時)的位置
CGRect RightContentEndFrame()
{
    return CGRectMake(S_W*0.75, 0, S_W, S_H);
}

CGRect CGRM(CGFloat x, CGFloat y,CGFloat w,CGFloat h)
{
    return CGRectMake(x, y, w, h);
}
@end

歡迎下載

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末垛叨,一起剝皮案震驚了整個濱河市伦糯,隨后出現(xiàn)的幾起案子柜某,更是在濱河造成了極大的恐慌,老刑警劉巖敛纲,帶你破解...
    沈念sama閱讀 217,542評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件喂击,死亡現(xiàn)場離奇詭異,居然都是意外死亡淤翔,警方通過查閱死者的電腦和手機(jī)翰绊,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,822評論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來旁壮,“玉大人监嗜,你說我怎么就攤上這事÷招常” “怎么了裁奇?”我有些...
    開封第一講書人閱讀 163,912評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長麦撵。 經(jīng)常有香客問我刽肠,道長,這世上最難降的妖魔是什么免胃? 我笑而不...
    開封第一講書人閱讀 58,449評論 1 293
  • 正文 為了忘掉前任音五,我火速辦了婚禮,結(jié)果婚禮上羔沙,老公的妹妹穿的比我還像新娘放仗。我一直安慰自己,他們只是感情好撬碟,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,500評論 6 392
  • 文/花漫 我一把揭開白布诞挨。 她就那樣靜靜地躺著,像睡著了一般呢蛤。 火紅的嫁衣襯著肌膚如雪惶傻。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,370評論 1 302
  • 那天其障,我揣著相機(jī)與錄音银室,去河邊找鬼。 笑死励翼,一個胖子當(dāng)著我的面吹牛蜈敢,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播汽抚,決...
    沈念sama閱讀 40,193評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼抓狭,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了造烁?” 一聲冷哼從身側(cè)響起否过,我...
    開封第一講書人閱讀 39,074評論 0 276
  • 序言:老撾萬榮一對情侶失蹤午笛,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后苗桂,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體药磺,經(jīng)...
    沈念sama閱讀 45,505評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,722評論 3 335
  • 正文 我和宋清朗相戀三年煤伟,在試婚紗的時候發(fā)現(xiàn)自己被綠了癌佩。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,841評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡便锨,死狀恐怖驼卖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情鸿秆,我是刑警寧澤酌畜,帶...
    沈念sama閱讀 35,569評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站卿叽,受9級特大地震影響桥胞,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜考婴,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,168評論 3 328
  • 文/蒙蒙 一贩虾、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧沥阱,春花似錦缎罢、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,783評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至崇棠,卻和暖如春咽袜,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背枕稀。 一陣腳步聲響...
    開封第一講書人閱讀 32,918評論 1 269
  • 我被黑心中介騙來泰國打工询刹, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人萎坷。 一個月前我還...
    沈念sama閱讀 47,962評論 2 370
  • 正文 我出身青樓凹联,卻偏偏與公主長得像,于是被迫代替她去往敵國和親哆档。 傳聞我的和親對象是個殘疾皇子蔽挠,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,781評論 2 354

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