自定义UIScrollView翻页宽度

预览

ScrollView.gif

前言

好奇心日报的有的文章页面是这样展示的,点击其中一个就可以进入单个的文章详情页面,于是想了一下该怎么实现。

思路

一开始想的是用一个ScrollView去实现,后来发现一个ScrollView实现的话最后一页会有问题,也没有想到解决方案,于是就用两个ScrollView来实现

屏幕快照 2017-02-21 下午3.32.44.png

屏幕快照 2017-02-21 下午3.32.55.png

第一张图蓝色标记的是底下的小的ScrollView,它的作用是实现翻页效果,第二张图蓝色标记的是上层的大的ScrollView,用来展示内容,用小的UIScrollView来带动大的ScrollView滑动,大的ScrollView不参与用户滑动事件

实现

假设展示4页内容

一些宏定义

#define kScreenWidth ([UIScreen mainScreen].bounds.size.width)
#define kScreenHeight ([UIScreen mainScreen].bounds.size.height)
#define kContentWidth 250
#define kContentHeight 400
#define kPadding 20

SmallScrollView

需要注意的是,小的ScrollView最后一页的宽度需要计算一下

self.smallScrollView = ({
        UIScrollView *scroll = [[UIScrollView alloc] init];
        scroll.delegate = self;
        scroll.showsHorizontalScrollIndicator = NO;
        scroll.pagingEnabled = YES;
        [self addSubview:scroll];
        [scroll makeConstraints:^(MASConstraintMaker *make) {
            make.edges.equalTo(self).offset(UIEdgeInsetsMake(0, kPadding / 2, 0, kScreenWidth - kPadding - kContentWidth - kPadding / 2));
        }];
        
        UIView *content = [[UIView alloc] init];
        [scroll addSubview:content];
        [content makeConstraints:^(MASConstraintMaker *make) {
            make.edges.equalTo(scroll);
            make.height.equalTo(scroll);
        }];
        
        UIView *lastView;
        
        for (NSInteger i=0; i<4; i++) {
            UIView *view = [[UIView alloc] init];
            [content addSubview:view];
            if (i == 0) {
                [view makeConstraints:^(MASConstraintMaker *make) {
                    make.left.equalTo(kPadding / 2);
                    make.top.bottom.equalTo(content);
                    make.width.equalTo(kContentWidth);
                    make.height.equalTo(kContentHeight);
                }];
            } else if (i == 3) {
                [view makeConstraints:^(MASConstraintMaker *make) {
                    make.left.equalTo(lastView.right).offset(kPadding);
                    make.top.bottom.equalTo(content);
                    make.width.equalTo(kContentWidth - (kScreenWidth - kPadding * 2 - kContentWidth));
                    make.height.equalTo(kContentHeight);
                }];
            } else {
                [view makeConstraints:^(MASConstraintMaker *make) {
                    make.left.equalTo(lastView.right).offset(kPadding);
                    make.top.bottom.equalTo(content);
                    make.width.equalTo(kContentWidth);
                    make.height.equalTo(kContentHeight);
                }];
            }
            lastView = view;
        }
        [content makeConstraints:^(MASConstraintMaker *make) {
            make.right.equalTo(lastView.right).offset(kPadding / 2);
        }];
        
        UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(smallScrollTap:)];
        [scroll addGestureRecognizer:tap];
        
        scroll;
    });

BigScrollView

self.bigScrollView = ({
        UIScrollView *scroll = [[UIScrollView alloc] init];
        scroll.clipsToBounds = NO;
        scroll.showsHorizontalScrollIndicator = NO;
        [self addSubview:scroll];
        [scroll makeConstraints:^(MASConstraintMaker *make) {
            make.edges.equalTo(self).offset(UIEdgeInsetsMake(0, kPadding / 2, 0, kPadding / 2));
        }];
        
        UIView *content = [[UIView alloc] init];
        [scroll addSubview:content];
        [content makeConstraints:^(MASConstraintMaker *make) {
            make.edges.equalTo(scroll);
            make.height.equalTo(scroll);
        }];
        
        UIView *lastView;
        
        for (NSInteger i=0; i<4; i++) {
            UIView *view = [[UIView alloc] init];
            view.backgroundColor = [self randomColor];
            [content addSubview:view];
            if (i == 0) {
                [view makeConstraints:^(MASConstraintMaker *make) {
                    make.left.equalTo(kPadding / 2);
                    make.top.equalTo(content);
                    make.width.equalTo(kContentWidth);
                    make.height.equalTo(kContentHeight);
                }];
            } else {
                [view makeConstraints:^(MASConstraintMaker *make) {
                    make.left.equalTo(lastView.right).offset(kPadding);
                    make.top.equalTo(content);
                    make.width.equalTo(kContentWidth);
                    make.height.equalTo(kContentHeight);
                }];
            }
            lastView = view;
            
//            UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
//            [btn setBackgroundColor:[UIColor lightGrayColor]];
//            [btn setTitle:@"Button" forState:UIControlStateNormal];
//            [btn addTarget:self action:@selector(btnClick:) forControlEvents:UIControlEventTouchUpInside];
//            [view addSubview:btn];
//            [btn makeConstraints:^(MASConstraintMaker *make) {
//                make.centerX.equalTo(view);
//                make.width.equalTo(100);
//                make.height.equalTo(50);
//                make.bottom.equalTo(-50);
//            }];
        }
        [content makeConstraints:^(MASConstraintMaker *make) {
            make.right.equalTo(lastView.right).offset(kPadding / 2);
        }];
        
        scroll;
    });

其它

前面说到大的ScrollView不参与用户事件,那该怎样做到呢?

- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    return self.smallScrollView;
}

就是这么简单。。。所有事件都让底下的小的ScrollView去处理,实现ScrollView的代理方法,进而带动大的ScrollView滑动

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    if (scrollView == self.smallScrollView) {
        [self.bigScrollView setContentOffset:scrollView.contentOffset animated:NO];
    }
}

关于滑动的就处理完了

点击事件

下面就是处理用户点击单个页面了,给小的ScrollView添加单击手势,然后将点转化到大的ScrollView即完成了单击手势的检测,假如页面上有button,判断点是否在button内进而处理

UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(smallScrollTap:)];
        [scroll addGestureRecognizer:tap];
- (void)smallScrollTap:(UITapGestureRecognizer *)sender{
    [self.bigSubViews removeAllObjects];
    [self listSubviewsOfView:self.bigScrollView];
    CGPoint bigPoint = [sender locationInView:self.bigScrollView];
    for (UIView *view in self.bigSubViews.reverseObjectEnumerator) {
        if ([view isKindOfClass:[UIButton class]]) {
            CGPoint buttonPoint = [view convertPoint:bigPoint fromView:self.bigScrollView];
            if (CGRectContainsPoint(view.bounds, buttonPoint)) {
                [(UIButton *)view sendActionsForControlEvents:UIControlEventTouchUpInside];
            }
        }
    }
}

- (void)listSubviewsOfView:(UIView *)view {
    NSArray *subviews = [view subviews];
    if ([subviews count] == 0) return;
    for (UIView *subview in subviews) {
        [self.bigSubViews addObject:subview];
        [self listSubviewsOfView:subview];
    }
}

不足

我也不清楚别人是怎么实现这种ScrollView效果的,这个方法我也不清楚是不是最好的,只是我能想到的方法,当然有些许不足,比如都交给小的ScrollView处理,展示的内容如果有button,button在normall,highlighted,selected状态设置了不同的样式,该实现是没有办法展示的,因为我只是发送了一个事件过去。所以如果你有更好的方法,不妨发表一下!

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,116评论 25 707
  • 为了能预览GitHub上的页面,GitHub推出了 GitHub Pages 功能,以下是设置GitHub Pag...
    chiang24阅读 1,484评论 0 0
  • 一个走在伪文艺道路上,混吃骗喝,爱健身,爱美食,爱摄影的编程妞带你一起领略做饭这件小事 不知道从什么时候开始喜欢上...
    编程妞阅读 313评论 2 1
  • 儿子现在已有四岁三个月有余,在我的记忆中只有一次,在2015年九月份去济南学习,分开一晚;除此之外我们娘俩没有分开...
    Sunny仔仔阅读 382评论 1 0
  • 天晴的时候,放一只纸鸢,鸢在天上飞,人在地上跑。他们之间,有一根连线,叫做思念。
    童真的眼睛阅读 174评论 0 0