iOS-UI搭建之【滾動標簽欄】

導讀:

下面這個視圖(多視圖滑動點擊切換)在很多App都有用到.我對這個View進行了封裝,外界只需要調用一個接口,就能實現這個效果.相當好用的一個輪子,github源碼分享https://github.com/HelloYeah/HYTabbarView.
大家checkout時順手點個星星,與人為樂骗露,自得其樂.

HYTabbarView效果圖如下
1.gif
HYTabbarView可靈活配置一屏寬顯示多少個標題,以及標題欄的高度,具體看項目需求
#define HYTabbarViewHeight 49    //頂部標簽條的高度
#define HYColumn 4      //一屏幕寬顯示4個標題

實現思路詳解

  • 界面分析:分為上下部分,頂部UIScrollView,底部UICollectionView.再實現兩部分的聯動即可實現 (底部視圖相對復雜,占用內存大,底部用UICollectionView實現會比用UIScrollView性能好很多)
  • 每一個標題對應一個View視圖,View視圖交由相應的控制器來管理,代碼結構十分清晰.做到不同View上的業(yè)務邏輯高聚合.也不會產生耦合性
  • 上下兩部分的聯動,這里是同過KVO實現的,監(jiān)聽當前的selectedIndex,底部視圖滾動時,修改selectedIndex的值.在KVO監(jiān)聽的回調方法里讓標題居中.
  • 其他細節(jié)相對簡單,大家不看代碼都知道如何處理,比如:點擊頂部標題,設置按鈕選中,切換到對應的CollectionCell等
代碼片段:

1.外界傳個控制器和一個標題,添加一個欄目

//外界傳個控制器,添加一個欄目
- (void)addSubItemWithViewController:(UIViewController *)viewController{
    
    UIButton * btn = [UIButton buttonWithType:UIButtonTypeCustom];
    [self.tabbar addSubview:btn];
    [self setupBtn:btn withTitle:viewController.title];
    [btn addTarget:self action:@selector(itemSelected:) forControlEvents:UIControlEventTouchUpInside];
    [self.subViewControllers addObject:viewController];
}   

2.KVO監(jiān)聽當前選中View的序號值

//viewDidLoad中添加觀察者
[self addObserver:self forKeyPath:@"selectedIndex" options:NSKeyValueObservingOptionOld |NSKeyValueObservingOptionNew context:@"scrollToNextItem"];

  //讓標題按鈕居中算法
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
    
    if (context == @"scrollToNextItem") {
        self.prevSelectedIndex = [change[@"old"] integerValue];
        if (self.prevSelectedIndex == self.selectedIndex) {
            return;
        }

        //設置按鈕選中
        [self itemSelectedIndex:self.selectedIndex];
        UIButton * btn = self.titles[self.selectedIndex];
    
        //讓選中按鈕居中
        NSInteger  min = HYColumn  / 2 ;
        if (_selectedIndex <= min) {
            [UIView animateWithDuration:0.25 animations:^{
                _tabbar.contentOffset = CGPointMake(0, 0);
            }];
        }else if (_selectedIndex >= self.titles.count - min) {
            UIButton * tempBtn = self.titles[self.titles.count - min - 1];
            CGFloat btnX = (HYColumn % 2 ) ? tempBtn.center.x : (tempBtn.center.x + btn.frame.size.width * 0.5) ;
            CGFloat offsetX = _tabbar.center.x - btnX;
            [UIView animateWithDuration:0.25 animations:^{
                _tabbar.contentOffset = CGPointMake(- offsetX, 0);
            }];     
        }else if (_selectedIndex > min && _selectedIndex < self.titles.count - min && self.titles.count > HYColumn ) {
            CGFloat btnX  = (HYColumn % 2 ) ? btn.center.x : (btn.center.x - btn.frame.size.width * 0.5) ;
            CGFloat offsetX = _tabbar.center.x - btnX;
            [UIView animateWithDuration:0.25 animations:^{
                _tabbar.contentOffset = CGPointMake( - offsetX, 0);
            }];
        }
    } else {
        [super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
    }
}

控制器代碼如下

使用方法類似系統的UITabbarController,外界只需直接傳入控制器.

- (void)viewDidLoad {
   
   [super viewDidLoad];

   [self.view addSubview:self.tabbarView];
}
//懶加載
- (HYTabbarView *)tabbarView{

   if (!_tabbarView) {
       _tabbarView = ({
           
           HYTabbarView * tabbar = [[HYTabbarView alloc]initWithFrame:CGRectMake(0, 30, [UIScreen mainScreen].bounds.size.width, 600)];
           
           for (NSInteger i = 0; i< 10; i ++) {
               UIViewController * vc = [[UIViewController alloc]init];
               vc.title = [NSString stringWithFormat:@"第%ld個",i+1];
               [tabbar addSubItemWithViewController:vc];
           }
           tabbar;
       });
   }
   return _tabbarView;
}

后續(xù)還會進一步優(yōu)化代碼

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市血巍,隨后出現的幾起案子萧锉,更是在濱河造成了極大的恐慌,老刑警劉巖述寡,帶你破解...
    沈念sama閱讀 217,826評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件柿隙,死亡現場離奇詭異,居然都是意外死亡辨赐,警方通過查閱死者的電腦和手機优俘,發(fā)現死者居然都...
    沈念sama閱讀 92,968評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來掀序,“玉大人帆焕,你說我怎么就攤上這事〔还В” “怎么了叶雹?”我有些...
    開封第一講書人閱讀 164,234評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長换吧。 經常有香客問我折晦,道長,這世上最難降的妖魔是什么沾瓦? 我笑而不...
    開封第一講書人閱讀 58,562評論 1 293
  • 正文 為了忘掉前任满着,我火速辦了婚禮,結果婚禮上贯莺,老公的妹妹穿的比我還像新娘风喇。我一直安慰自己,他們只是感情好缕探,可當我...
    茶點故事閱讀 67,611評論 6 392
  • 文/花漫 我一把揭開白布魂莫。 她就那樣靜靜地躺著,像睡著了一般爹耗。 火紅的嫁衣襯著肌膚如雪耙考。 梳的紋絲不亂的頭發(fā)上谜喊,一...
    開封第一講書人閱讀 51,482評論 1 302
  • 那天,我揣著相機與錄音倦始,去河邊找鬼斗遏。 笑死,一個胖子當著我的面吹牛楣号,可吹牛的內容都是我干的最易。 我是一名探鬼主播怒坯,決...
    沈念sama閱讀 40,271評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼炫狱,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了剔猿?” 一聲冷哼從身側響起视译,我...
    開封第一講書人閱讀 39,166評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎归敬,沒想到半個月后酷含,有當地人在樹林里發(fā)現了一具尸體,經...
    沈念sama閱讀 45,608評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡汪茧,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,814評論 3 336
  • 正文 我和宋清朗相戀三年椅亚,在試婚紗的時候發(fā)現自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片舱污。...
    茶點故事閱讀 39,926評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡呀舔,死狀恐怖,靈堂內的尸體忽然破棺而出扩灯,到底是詐尸還是另有隱情媚赖,我是刑警寧澤,帶...
    沈念sama閱讀 35,644評論 5 346
  • 正文 年R本政府宣布珠插,位于F島的核電站惧磺,受9級特大地震影響,放射性物質發(fā)生泄漏捻撑。R本人自食惡果不足惜磨隘,卻給世界環(huán)境...
    茶點故事閱讀 41,249評論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望顾患。 院中可真熱鬧番捂,春花似錦、人聲如沸描验。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,866評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽膘流。三九已至絮缅,卻和暖如春鲁沥,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背耕魄。 一陣腳步聲響...
    開封第一講書人閱讀 32,991評論 1 269
  • 我被黑心中介騙來泰國打工画恰, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人吸奴。 一個月前我還...
    沈念sama閱讀 48,063評論 3 370
  • 正文 我出身青樓允扇,卻偏偏與公主長得像,于是被迫代替她去往敵國和親则奥。 傳聞我的和親對象是個殘疾皇子考润,可洞房花燭夜當晚...
    茶點故事閱讀 44,871評論 2 354

推薦閱讀更多精彩內容

  • 發(fā)現 關注 消息 iOS 第三方庫、插件读处、知名博客總結 作者大灰狼的小綿羊哥哥關注 2017.06.26 09:4...
    肇東周閱讀 12,103評論 4 62
  • 面朝著大海 就算沒有春暖花開 但也不會是 嚴冬酷暑 在億萬的人海中 微小的我 是多么多么的 微不足道啊 就算是這樣...
    心笑塵閱讀 165評論 0 0
  • 今晚讀《高績效教練》糊治,談到改變。聯系到公司和工作罚舱,頗有感觸井辜。 我所服務的公司是一家德國家族企業(yè),已有將近100...
    鳶紫閱讀 249評論 0 1
  • 今天12月13日南京迎來又一次舉城默哀以國家的名義向77年前的300000生靈致哀這是一座城的受難日也是一個國家的...
    FB閱讀 434評論 0 1
  • 文:吃貨不孤獨 本文原創(chuàng),歡迎轉載 我們公司里面有個小陳包个,負責我們相關的公司的各種雜項工作刷允,也喜歡看各種雞湯文章,...
    吃貨不孤獨閱讀 1,197評論 15 14