实现横向tableView功能,仿新闻首页标签栏和内容均可滚动

要实现横向滚动的像新闻首页这样的scrollView,用最笨的方法就是scrollView上添加不同的view,没有优化,没有缓存。

假如标签栏有很多标签,并且大部分标签展示的页面的布局类似,这就很有必要采用缓存机制,类似tableView功能。

假如标签栏有很多标签,但是所有标签展示的页面的布局几乎都不一样,那么采用缓存机制也达不到优化的作用,因为缓存池里几乎装下了所有页面。

先看一下如下效果图:


1499A456-6270-44FD-B4AD-1D8EA1FB8EF6.png

一.后面一种简单的假设

这种情况采用类似懒加载的原理,scrollerView里最多只有三个view,滑动时创建新的view,及时将远离的view移除掉。这里的view采用的是子控制器的view。在初始化自定义控件时不能创建好子控制器传入,这样就会一直占用内存,我采用了传入类名的字符串方式,动态的去创建子类。

关键代码如下:

- (void)scrollViewDidEndScrollingAnimation:(UIScrollView *)scrollView
{
    //当前的索引
    NSInteger index = scrollView.contentOffset.x / scrollView.width;
    
    if (self.controllerArray.count <= index) {
        return;
    }
    
    if (![self.childVCDic objectForKey:IntgerToStr(index)]) {
        [self createChildVCWithIndex:index];
    }
    for (int i = 0; i < self.childVCDic.allKeys.count; i++) {
        NSString *indexKey = self.childVCDic.allKeys[i];
        NSInteger lastIndex = [indexKey integerValue];
        ChildBaseController *vc = [self.childVCDic objectForKey:indexKey];

        if (labs(lastIndex-index) >= 2) {
            [vc.view removeFromSuperview];
//            [vc removeFromParentViewController];
            [self.childVCDic removeObjectForKey:indexKey];
            i--;
        } else if (lastIndex == index) {
            
            [vc downloadData];
        }
    }
}

在滑动内容view或点击标签按钮时都会掉用上面的方法,通过掉用下面的方法创建新的viewCell.

- (void)createChildVCWithIndex:(NSInteger)index
{
    ChildBaseController *vc = [[NSClassFromString(self.controllerArray[index]) alloc] init];
    vc.view.x = index * SCREEN_WIDTH;
    vc.view.y = 0;
    vc.view.height = self.contentView.height;
    vc.index = index;
    [self.contentView addSubview:vc.view];
//    [self addChildViewController:vc];
    [self.childVCDic setObject:vc forKey:IntgerToStr(index)];
}

通过代码if (labs(lastIndex-index) >= 2)判断不是当前界面位置左或右的控制器视图都将其引用全部删除。

集成这个控件相对简单,主控制器须继承ScrollerChildsController类,代码如下:

//RootViewController.h
@interface RootViewController : ScrollerChildsController
@end

//RootViewController.m
@implementation RootViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor grayColor];
    
    self.title = @"ScrollerChildVCs";
    [self.navigationController.navigationBar setBarTintColor:[UIColor colorWithRed:0.800 green:0.600 blue:0.800 alpha:0.5]];

    self.titleCanScroll = YES;
    
    NSMutableArray *titleArr = [NSMutableArray array];
    for (int i = 0; i < 20; i++) {
        [titleArr addObject:[NSString stringWithFormat:@"title_%d", i]];
    }
    self.titleArray = titleArr;
    
    NSMutableArray *vcArr = [NSMutableArray array];
    for (int i = 0; i < self.titleArray.count; i++) {
        if (i%2 == 0) {
            [vcArr addObject:@"OneDetailController"];
        } else {
            [vcArr addObject:@"TwoDetailController"];
        }
    }
    self.controllerArray = vcArr;
}

@end

所有子控制器也需要继承ChildBaseController,然后重载父类方法,如下:

- (void)downloadData
{
    self.label.text = [NSString stringWithFormat:@"这是OneDetailController\n第 %ld 页", (long)self.index];
}

二.第一种假设采用复用机制

需要有一个缓存池,熟悉tableview缓存机制的可以很好理解。

基本原理:

滑动scrollview时,先判断是否需要加载新的cell,若需要,则先去缓存池取cell,如果取不到就创建cell。存取cell时有一个标识reuseIdentifier,即id,布局一样的界面就加载一样的cell,具有一样的id,加载cell时就通过传入的id匹配上缓存池中的cell的id就返回复用。


- (void)scrollViewDidEndScrollingAnimation:(UIScrollView *)scrollView
{
    // 当前的索引
    NSInteger index = scrollView.contentOffset.x / scrollView.width;
    
    if (self.titleArray.count <= index) {
        return;
    }
    
    newIndex = index;
    
    if ([self.delegate respondsToSelector:@selector(scrollView:cellForIndex:)]) {
        [self manageData];
        
        LXScrollViewCell *cell = [self.delegate scrollView:self cellForIndex:index];
        [self refreshCellFrame:cell withIndex:index];
    }
}

这self.showChildVCArr数组里记录的是当前屏幕能看到的cell,当需加载新的cell时,得判断数组里全部cell是否还能在屏幕上显示(如果点击离当前界面对应的标签更远的标签,那么showChildVCArr里的所有cell都不会在屏幕中显示)。
- (void)manageData
{
    //判断滑动方向,因为self.showChildVCArr的数据是根据X坐标升序排列的
    if (lastIndex < newIndex) {
        for (int i = 0; i < self.showChildVCArr.count; i++) {
            LXScrollViewCell *cell = self.showChildVCArr[i];
            
            if (cell.x/SCREEN_WIDTH < newIndex-1) {
                //判断了此cell已经没有显示在屏幕上
                [self changeDataWithCell:cell];
                i--;
            } else {
                break;
            }
        }
    } else {
        for (int i = (int)self.showChildVCArr.count - 1; i >= 0 && self.showChildVCArr.count > 0; i--) {
            LXScrollViewCell *cell = self.showChildVCArr[i];
            
            if (cell.x/SCREEN_WIDTH > newIndex+1) {
                //判断了此cell已经没有显示在屏幕上
                [self changeDataWithCell:cell];
            } else {
                break;
            }
        }
    }
}

- (void)changeDataWithCell:(LXScrollViewCell *)cell
{
    [self.showChildVCArr removeObject:cell];
    if ([self.cacheChildVCDic objectForKey:cell.reuseIdentifier] == nil) {
        [self.cacheChildVCDic setObject:[NSMutableArray array] forKey:cell.reuseIdentifier];
    }
    
    //将cell添加到缓存池中
    [[self.cacheChildVCDic objectForKey:cell.reuseIdentifier] addObject:cell];
}

然后在业务控制器里的集成代码如下:

- (void)viewDidLoad {
    [super viewDidLoad];
    [self initSetup];

    NSMutableArray *titleArr = [NSMutableArray array];
    for (int i = 0; i < 20; i++) {
        [titleArr addObject:[NSString stringWithFormat:@"title_%d", i]];
    }
    self.titleArray = titleArr;

    LXScrollView *scrollerView = [[LXScrollView alloc] initWithFrame:CGRectMake(0, 64, SCREEN_WIDTH, SCREEN_HEIGHT - 64)];
    scrollerView.delegate = self;
    scrollerView.titleCanScroll = YES;
    scrollerView.titleArray = self.titleArray;
    [self.view addSubview:scrollerView];
    self.scrollerView = scrollerView;
}

- (LXScrollViewCell *)scrollView:(LXScrollView *)scrollView cellForIndex:(NSInteger)index
{
    if (index%4 == 0 ) {
        static NSString *cellId = @"OneScrollViewCell";
        OneScrollViewCell *viewCell = [scrollView dequeueReusableCellWithIdentifier:cellId];
        if (!viewCell) {
            viewCell = [[OneScrollViewCell alloc] initWithReuseIdentifier:cellId];
        }
        [viewCell refreshDataWithIndex:index];
        return viewCell;
        
    } else if (index%4 == 1) {
        static NSString *cellId = @"TwoScrollViewCell";
        TwoScrollViewCell *viewCell = [scrollView dequeueReusableCellWithIdentifier:cellId];
        if (!viewCell) {
            viewCell = [[TwoScrollViewCell alloc] initWithReuseIdentifier:cellId];
        }
        [viewCell refreshDataWithIndex:index];
        return viewCell;
    } else {
        static NSString *cellId = @"ThreeScrollViewCell";
        ThreeScrollViewCell *viewCell = [scrollView dequeueReusableCellWithIdentifier:cellId];
        if (!viewCell) {
            viewCell = [[ThreeScrollViewCell alloc] initWithReuseIdentifier:cellId];
        }
        [viewCell refreshDataWithIndex:index];
        return viewCell;
    }
}

- (void)scrollView:(LXScrollView *)tableView didSelectIndex:(NSInteger)index
{
    NSLog(@"点击了第 %ld 个标签", (long)index);
}

仿tableview的原理和协议方法做的复用,通过dequeueReusableCellWithIdentifier:实现缓存机制。

- (nullable __kindof LXScrollViewCell *)dequeueReusableCellWithIdentifier:(NSString *)identifier
{
    //先取显示中的cell
    if (lastIndex > newIndex) {//如果向右滑动
        for (int i = 0; i < self.showChildVCArr.count; i++) {
            LXScrollViewCell *cell = self.showChildVCArr[i];
            if (cell.x/SCREEN_WIDTH == newIndex) {//if成立说明这个cell还是显示在界面上的
                showing = YES;
                return cell;
            } else if (cell.x/SCREEN_WIDTH > newIndex){
                break;
            }
        }
    } else {
        for (int i = (int)self.showChildVCArr.count - 1; i >= 0; i--) {
            LXScrollViewCell *cell = self.showChildVCArr[i];
            if (cell.x/SCREEN_WIDTH == newIndex) {
                showing = YES;
                return cell;
            } else if (cell.x/SCREEN_WIDTH < newIndex){
                break;
            }
        }
    }
    
    //再取缓存中的cell
    NSArray *cellArr = [self.cacheChildVCDic objectForKey:identifier];
    if (cellArr.count > 0) {
        //随意取一个相同id的cell返回
        return [cellArr firstObject];
    }
    
    return nil;
}

总结:

可以根据自己项目需求选择使用上面两个demo的一个。布局能复用的采用此demo,请点击github地址下载。布局大部分不一样,不需要复用的采用此demo,请点击github地址下载。两个demo的标签栏都可以由属性titleCanScroll设置能否滚动。

若有bug或不足之处,敬请评论告知。


QQ:2239344645 我的github

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

推荐阅读更多精彩内容