联动菜单的实现方案探索

前言

开发中,经常用到分页滚动菜单的功能点,底部页面滚动,顶部的菜单标题也会随着页面的滚动位置随之进行切换,这样的效果实际上在项目中常常能用上。为了以最快的速度去实现该功能,这里对这样的滚动菜单做了一个封装,简单说一下实现的过程。

分析

简单来看,由于滚动菜单点击切换并且切换对应的页面可以去控制菜单的item有个联动的过程。所以封装的时候也主要对外暴露几个重要属性内容:

  • 菜单item点击回调
  • 菜单item的设置属性
  • 标题数组的设置

实现

这样的滚动菜单无疑是布局以及点击跳转跟联动逻辑的结合。

布局

由于菜单的布局受菜单的item的数目约束,考虑到屏幕的宽度与菜单展示的总长度关系,这里对菜单的展示做了个适配,即如果没有超出屏幕菜单的item的间隔等距。如果超出屏幕则按照固定的距离添加对应的菜单上。

- (void)setTitleArray:(NSArray *)titleArray{
    
    _titleArray = titleArray;
    
    
    [self setUI];
    
    self.isBlock = YES;
    
    NSInteger btnOffset = 0;
    
    //判断要添加的item是否超出屏幕,如果没有,等分
    BOOL isMore = [self isMoreScreenWidth];

    if (isMore) {
        
        
        for (int i = 0; i < titleArray.count; i++) {
            
            UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
            [btn setTitle:self.titleArray[i] forState:UIControlStateNormal];
            [btn setTitleColor:LCColorRGB(74, 74, 74) forState:UIControlStateNormal];
            [btn setTitleColor:LCColorRGB(173, 135, 72) forState:UIControlStateSelected];
            btn.titleLabel.font = [UIFont systemFontOfSize:15];
            btn.tag = i;
            
            [btn sizeToFit];
            float originX =  i? 37+btnOffset:18;
            btn.frame = CGRectMake(originX, 0, btn.frame.size.width, 48);
            
            btnOffset = CGRectGetMaxX(btn.frame);
            
            btn.titleLabel.textAlignment = NSTextAlignmentLeft;
            [btn addTarget:self action:@selector(changeSelectedState:) forControlEvents:UIControlEventTouchUpInside];
            
            btn.titleLabel.font = [UIFont systemFontOfSize:14];
            
            [self.scrollView addSubview:btn];
            [self.buttonsArray addObject:btn];
            
            //默认选择第一个
            if (i == 0) {
                
                btn.selected = YES;
                btn.titleLabel.font = [UIFont systemFontOfSize:15];
                self.currentSelectBtn = btn;
                
                self.bottomBarView.frame = CGRectMake(btn.frame.origin.x, self.scrollView.frame.size.height-2, btn.frame.size.width, 2);
                [self.scrollView addSubview:self.bottomBarView];
                
            }
            
            
        }

        
        
    }else{
        
        //计算等分之后的间隙大小
        CGFloat interValWidth = (SCREEN_WIDTH - self.totalTitleWidth) / (self.titleArray.count + 1);
        
        
        for (int i = 0; i < titleArray.count; i++) {
            
            UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
            [btn setTitle:self.titleArray[i] forState:UIControlStateNormal];
            [btn setTitleColor:LCColorRGB(74, 74, 74) forState:UIControlStateNormal];
            [btn setTitleColor:LCColorRGB(173, 135, 72) forState:UIControlStateSelected];
            btn.titleLabel.font = [UIFont systemFontOfSize:15];
            btn.tag = i;
            
            [btn sizeToFit];
            float originX =  i? interValWidth+btnOffset:interValWidth;
            btn.frame = CGRectMake(originX, 0, btn.frame.size.width, 48);
            
            btnOffset = CGRectGetMaxX(btn.frame);
            
            
            btn.titleLabel.textAlignment = NSTextAlignmentLeft;
            [btn addTarget:self action:@selector(changeSelectedState:) forControlEvents:UIControlEventTouchUpInside];
            
            btn.titleLabel.font = [UIFont systemFontOfSize:14];
            
            [self.scrollView addSubview:btn];
            [self.buttonsArray addObject:btn];
            
            //默认选择第一个
            if (i == 0) {
                
                btn.selected = YES;
                btn.titleLabel.font = [UIFont systemFontOfSize:15];
                self.currentSelectBtn = btn;
                
                self.bottomBarView.frame = CGRectMake(btn.frame.origin.x, self.scrollView.frame.size.height-2, btn.frame.size.width, 2);
                [self.scrollView addSubview:self.bottomBarView];
                
            }
            
            
        }

        
        
    }
    
    
    
    //更新scrollView的内容区域
    self.scrollView.contentSize = CGSizeMake(btnOffset+18, self.scrollView.frame.size.height);
    

}

- (BOOL)isMoreScreenWidth{

    CGFloat totalWidth = 0;
    
    totalWidth += 18;
    
    for (int i = 0; i<self.titleArray.count; i++) {
        
        UILabel *label = [UILabel new];
        label.font = [UIFont systemFontOfSize:15];
        label.text = self.titleArray[i];
        
        [label sizeToFit];
        totalWidth += label.frame.size.width;
        totalWidth += 22;
        
        self.totalTitleWidth += label.frame.size.width;
    }
    
    
    if (totalWidth < SCREEN_WIDTH - 18) {
        
        return NO;
        
    }
    
    return YES;
    
}

点击item回调

为了保证点击菜单的item能够进行外部业务逻辑的处理,为此使用了Block进行事件的回调。

- (void)changeSelectedState:(UIButton *)button{
    
    self.currentSelectBtn.selected = NO;
    self.currentSelectBtn.titleLabel.font = [UIFont systemFontOfSize:14];
    
    self.currentSelectBtn = button;
    
    self.currentSelectBtn.selected = YES;
    self.currentSelectBtn.titleLabel.font = [UIFont systemFontOfSize:15];
    
    [UIView animateWithDuration:0.2 animations:^{
        
        if (button.tag == 0) {
            
            self.bottomBarView.frame = CGRectMake(self.currentSelectBtn.frame.origin.x, self.scrollView.frame.size.height - 2, self.currentSelectBtn.frame.size.width, 2);
            [self.scrollView scrollRectToVisible:CGRectMake(0, 0, self.scrollView.frame.size.width, self.scrollView.frame.size.height) animated:YES];
            
        }else{
            
            UIButton *preButton = self.buttonsArray[button.tag - 1];
            
            float offsetX = CGRectGetMinX(preButton.frame) - 18;
            
            [self.scrollView scrollRectToVisible:CGRectMake(offsetX, 0, self.scrollView.frame.size.width, button.frame.size.height) animated:YES];
            
            self.bottomBarView.frame = CGRectMake(self.currentSelectBtn.frame.origin.x, self.scrollView.frame.size.height-2, self.currentSelectBtn.frame.size.width, 2);
        }
        
//        self.scrollView.contentOffset = CGPointMake(SCREEN_WIDTH *button.tag, 0);
        
        if(self.pageSelectBlock && self.isBlock){
        
            NSLog(@"current seleted menu is %ld",button.tag);
            self.currentPage = button.tag;  //更新当前的curPage
            self.pageSelectBlock(button.tag);
            
        }
        
        //默认将传递打开
        self.isBlock = YES;
        
    }];

}

对外暴露设置当前菜单item的属性

为了保证外部能够控制当前的菜单的item的选中位置,所以提供了一个修改当前菜单的index的属性。

- (void)setCurrentPage:(NSInteger)currentPage{
    
    //防止重复设置
    if (_currentPage == currentPage) {
        
        return;
    }
    
    _currentPage = currentPage;

    if (self.titleArray.count == 0) {
        return;
    }
    
    
    self.isBlock = NO;
    //改变当前的按钮状态以及偏移对应的菜单
    UIButton *currentBtn = self.buttonsArray[currentPage];
    [self changeSelectedState:currentBtn];
    
}

注意点

这里需要注意的技术点在于边缘菜单item点击问题,如果被遮挡部分的item菜单点击的话,我们需要将当前的菜单露出来以保证能够被用户看到具体的菜单内容,这里可以通过scrollView 的scrollRectToVisible方法进行调整并滚动到可见位置来保证菜单的滚动效果以及显示特性。

具体Demo可以看这个:
https://github.com/cclbj/LCHangMenuDemo

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,711评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,932评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,770评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,799评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,697评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,069评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,535评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,200评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,353评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,290评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,331评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,020评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,610评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,694评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,927评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,330评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,904评论 2 341

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,265评论 25 707
  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,721评论 1 92
  • 我们不辞而别得很默契 在你的成长过程中,你有没有遇到过这样一些人?你和他们的交汇,像是两颗流星划出的线,在某个点相...
    宝宝不笑阅读 353评论 0 0
  • 01 迷迷糊糊张开了双眼,便看到头顶摇摇欲坠,发出微光的灯泡,才意识到,我被关在一共笼子里。‘’呜....‘’旁边...
    生途记录阅读 215评论 0 0
  • 看《返老还童》 Benjiemin . Button他有着一般人的内心,但是非一般人的经历。本剧是围绕不同视角的人...
    实施阅读 811评论 0 1