iOS 自定义 UIScrollerView的PageSize大小

前言: UIScrollerView 设置PageEnable,
每页滚动的偏移量是 UIScrollerView的frame.width的宽度,使用pageEnable控制page每次偏移量只能通过控制 frame解决

需求:

gif5新文件.gif

方案1:

使用UIScrollerView 通过代理控制一次滑动的滚动距离实现pageSize效果
根据滑动状态来调整 仿 pageEnable的效果

这里就不多介绍了 直接上代码:(查看了一些资料大部分的实现都基本是基于这段代码)

#pragma mark <UIScrollViewDelegate>
- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView {
    _startY = scrollView.contentOffset.y;
}
- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate {
    if (decelerate) return;
    [selfdealPageEnableWithScrollView:scrollView];
}
- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView {
    [selfdealPageEnableWithScrollView:scrollView];
}
#pragma mark 处理scrollView翻页效果
- (void)dealPageEnableWithScrollView:(UIScrollView *)scrollView{
    static CGFloat halfH;
    halfH = halfH ? halfH : scrollView.bounds.size.height;
    NSLog(@"-----------------%zi%zi%zi",scrollView.contentOffset.y > (_otherPageStartY - halfH),_startY < scrollView.contentOffset.y,_startY < (_otherPageStartY - halfH));
    NSLog(@"*****************%zi%zi%zi",scrollView.contentOffset.y < _otherPageStartY,_startY > scrollView.contentOffset.y,_startY >= _otherPageStartY);
    if (scrollView.contentOffset.y > (_otherPageStartY - halfH +60) &&
        _startY < scrollView.contentOffset.y &&
        _startY < _otherPageStartY
//        _isSecondPage) {//此处还可以用一个BOOL类型来记录是否是处在第一页,代替后面两个判断
        [UIViewanimateWithDuration:0.5animations:^{
            [scrollView setContentOffset:CGPointMake(0,_otherPageStartY)];
        } completion:^(BOOL finished) {
            NSLog(@"进入第二页------->%f",scrollView.contentOffset.y);
        }];
    }
    else if (scrollView.contentOffset.y < (_otherPageStartY -30) &&
             _startY > scrollView.contentOffset.y &&
             _startY >= (_otherPageStartY - halfH)
//        !_isSecondPage){
        [UIViewanimateWithDuration:0.5animations:^{
            [scrollView setContentOffset:CGPointZero];
        } completion:^(BOOL finished) {
            NSLog(@"进入第一页------->%f",scrollView.contentOffset.y);
        }];
    }

缺点:

效果死板, 每页切换的时候,流畅度不够, 甚至可以说略微的卡顿 , 只支持缓慢拖动, 当手势轻扫的时候甚至不能识别

方案2:

使用UICollectionView
可能很多人的第一想法就是使用 UICollectionView, 一波操作下来问题还是挺明显

  1. 创建UICollectionView的视图
  2. 主要属性设置: 将frame.width值 小于屏幕的宽
collectionView.pageEnable = YES 
collectionView.clipsToBounds = NO

优点:

即是有很多的item也能正常显示出来, 并且不会有打的内存资源消耗

缺点:

cell会被重用和销毁, 一个pageSize中只能显示两个cell, 但是相较于中间放大的这种需求明显是不合适的,不管网那个方向滑动 都有有一个是消失的,
暂时没有想到解决方案, 有兴趣的可以自己去研究下
这里有个链接有兴趣的可以看看

Paging a overflowing collection view

方案3:

使用UIScroller + pageEnable 实现(个人感觉比较简单方便,推荐)

主要利用系统pageEnable滑动一页,该一页其实就是当前scroller的frame 宽或者高 是frame不是contentSize 不要搞错

实现逻辑:
  1. 中间放大两侧缩小显示逻辑:
  • 将UIScrollerView的frame小于屏宽,具体小于多少取决于到时候左右要显示的多少
  • 设置UISCrollerView的pageEnable = YES
    clipsToBounds = NO;//不裁剪
  • 关于子视图间距 Doem中将最大设置成1 那么最小的就小于1 当视图缩小的时候本身就会产生间距, 该间距取决于视图缩放程度,自己把握吧
  1. 无限滚动实现逻辑:
    主要利用scrolle, 将contentSzie设置成很大, 但是其中子视图的创建和item数组有关,通过在滚动过程中不断调整 对应item的frame 值, 来实现无限滚动

主要类介绍

  1. ZPScrollerScaleViewConfig pageSize大小, 缩放大小, 子视图间距大小等配置信息
  2. ZPScrollerScaleView 实现无限滚动和滚动缩放的主要类,所有功能都是在这个类中实现完成
  • 三个重要属性:
    items: 子视图数组, 将要时间轮播的视图数组. 要求大于2个
    defalutIndex:视图展示的收个视图位于数组的下标
    currentIndex:当前界面居中显示的下标值(readonly)

ZPScrollerScaleViewConfig:
配置属性:

1. pageSize   //自定义的pagesize大小
2. ItemMaingin //子视图间距
3. scaleMin //最小缩放比
4. scaleMax //最大缩放比 建议设置为1 , 当大于1的时候 子视图会产生缩放导致失真

ZPScrollerScaleView

  • 设置默认显示 当默认值较小和较大的时候制造循环轮播条件
/**当默认值较小和较大的时候制造循环轮播条件*/
- (void)configDefult:(NSInteger)defultIndex{
    NSInteger currentIndex = 0;
    NSInteger needMoveIndex = 0;
    NSInteger currentIndex2 = 0;
    NSInteger needMoveIndex2 = 0;
    CGFloat shouldOffset = 0;
    if(_defalutIndex <= 1){ //将最大和第二大的视图调整位置
         currentIndex = 0;
         needMoveIndex = self.items.count -1;
         currentIndex2 = self.items.count -1;
         needMoveIndex2 = self.items.count -2;
        shouldOffset = -self.config.pageSize.width;
    }else if(_defalutIndex >= self.items.count -2){//将最小和第二小的视图调整位置
         currentIndex = self.items.count -1;
         needMoveIndex = 0;
         currentIndex2 = 0;
         needMoveIndex2 = 1;
        shouldOffset = self.config.pageSize.width;

    }
    
    UIView * currentView = [self.contentView viewWithTag:BaseTag + currentIndex];
    UIView * needMoveView = [self.contentView viewWithTag:BaseTag + needMoveIndex];
    needMoveView.transform = CGAffineTransformMakeScale(self.config.scaleMin, self.config.scaleMin);
    needMoveView.center = CGPointMake(currentView.center.x + shouldOffset, needMoveView.center.y);
   
    
    UIView * currentView2 = [self.contentView viewWithTag:BaseTag + currentIndex2];
    UIView * needMoveView2 = [self.contentView viewWithTag:BaseTag + needMoveIndex2];
    needMoveView2.transform = CGAffineTransformMakeScale(self.config.scaleMin, self.config.scaleMin);
    needMoveView2.center = CGPointMake(currentView2.center.x + shouldOffset, needMoveView2.center.y);
}
  • 添加需要参与轮播的子View
- (void)setItems:(NSArray<UIView *> *)items{
    _items = items;
    
    CGSize pageSize = self.config.pageSize;
    
    //将视图摆在中间, 并且消除求余误差
    CGFloat centerIetm = self.contentView.contentSize.width * 0.5;
    NSInteger pageIndex = centerIetm/pageSize.width;
    NSInteger pageOffsetIndex = pageIndex % self.items.count;
    centerIetm = centerIetm - pageOffsetIndex * pageSize.width;
    
    for(int i =0; i < items.count;i++){
        UIView * view = items[I];
        view.tag = BaseTag + I;
        view.frame = CGRectMake(i * pageSize.width + self.config.ItemMaingin + centerIetm, 0, pageSize.width-self.config.ItemMaingin*2, pageSize.height);
        [_contentView addSubview:view];
        view.transform = CGAffineTransformMakeScale(self.config.scaleMin, self.config.scaleMin);
    }
    //这是默认显示
    UIView * view = [self.contentView viewWithTag:self.defalutIndex+BaseTag];
    view.transform = CGAffineTransformMakeScale(self.config.scaleMax, self.config.scaleMax);
    [self.contentView setContentOffset:CGPointMake(centerIetm + pageSize.width*self.defalutIndex, 0)];
    [self configDefult:self.defalutIndex];
}

无限滚动实现代码:

- (void)itemViewStartAnimationWithContentOffset:(CGFloat)contentOffsetX{
    
    CGFloat pageSizeW = [UIScreen mainScreen].bounds.size.width - self.pageMagin*2;
    NSInteger pageIndex = contentOffsetX/pageSizeW;
    
    CGFloat scrale = (contentOffsetX/pageSizeW - pageIndex);
    if(scrale >= 0.99){
        [self exchangeItemViewPosition]; //完成一次显示, 调整子视图位置
    }
    if(scrale <= 0){
        return;
    }
    
    //视图(左)
    NSInteger currentIndex = (pageIndex%self.items.count);
    NSInteger nextIndex = ((currentIndex+1)>(self.items.count-1)?0:(currentIndex+1));;
    
    //视图缩放值(左)
    CGFloat scraleCurrent = self.config.scaleMax - (self.config.scaleMax-self.config.scaleMin)*scrale;
    CGFloat scraleNext = (self.config.scaleMax-self.config.scaleMin) *scrale +self.config.scaleMin;
    
    if (contentOffsetX < _lastContentOffset ){ //向右
        currentIndex = currentIndex +1 >= self.items.count?0:currentIndex +1;
        nextIndex = ((currentIndex-1)<0?self.items.count-1:(currentIndex-1));
        scraleCurrent = (self.config.scaleMax-self.config.scaleMin) *scrale +self.config.scaleMin;
        scraleNext = self.config.scaleMax - (self.config.scaleMax-self.config.scaleMin)*scrale;
    }
    
    
    UIView * subViewCurrent = [self.contentView viewWithTag:currentIndex+BaseTag];
    subViewCurrent.transform = CGAffineTransformMakeScale( scraleCurrent,  scraleCurrent);
    UIView * subViewNext = [self.contentView viewWithTag:nextIndex+BaseTag];
    subViewNext.transform = CGAffineTransformMakeScale(scraleNext,  scraleNext);
    
    
}
  1. 当scrollerView快速滚动的时候, 此时获取contentOffset是不连续的, 如过根据偏移量连续去处理滚动比例, 会导致缩放比例乃至视图显示不正确,此时采用_lastOffsetX 记录上一次的滚动, 与当前偏移量比较并+1加到当前偏移量,每次+1都会执行一次滚动计算,耗点性能但是丝般流畅,通过调试工具测试对性能影响不大.
- (void)scrollViewDidScroll:(UIScrollView *)scrollView{
    
    //首次显示
    if (_lastOffsetX == 0) {
        [self itemViewStartAnimationWithContentOffset:scrollView.contentOffset.x];
        _lastOffsetX = scrollView.contentOffset.x;
        return;
    }
    if(_lastOffsetX  > scrollView.contentOffset.x){ //向左滑动
        
        for (CGFloat i = _lastOffsetX; i >= scrollView.contentOffset.x; i--) {
            [self itemViewStartAnimationWithContentOffset:i];
        }
        
    }else{//向右滑动
        for (CGFloat i = _lastOffsetX; i < scrollView.contentOffset.x; i++) {
            [self itemViewStartAnimationWithContentOffset:i];
        }
    }
    _lastOffsetX = scrollView.contentOffset.x;
}
  1. 判断滚动一页完成, 调整子视图位置
- (void)exchangeItemViewPosition{
    self.pageIndex = self.contentView.contentOffset.x/([UIScreen mainScreen].bounds.size.width  - self.pageMagin*2);
    CGSize pageSize = self.config.pageSize;
    self.contentView.contentSize  = CGSizeMake(self.contentView.contentSize.width + pageSize.width, pageSize.height);
    if (self.contentView.contentOffset.x < _lastContentOffset ){
        //向右
        NSInteger currentIndex = (self.pageIndex%self.items.count);
        NSInteger needMoveIndex = currentIndex-2 < 0?self.items.count+(currentIndex-2):currentIndex-2;
        UIView * subView = [self.contentView viewWithTag:BaseTag + needMoveIndex];
        subView.transform = CGAffineTransformMakeScale(self.config.scaleMin, self.config.scaleMin);
        subView.center = CGPointMake((self.pageIndex-2) * pageSize.width+pageSize.width/2, subView.center.y);

    }else{
        //向左
        NSInteger currentIndex = (self.pageIndex%self.items.count);
        NSInteger needMoveIndex = currentIndex+2 > self.items.count-1?(currentIndex+2 -self.items.count):currentIndex+2;
        UIView * subView = [self.contentView viewWithTag:BaseTag + needMoveIndex];
        subView.transform = CGAffineTransformMakeScale(self.config.scaleMin, self.config.scaleMin);
        subView.center = CGPointMake((self.pageIndex+2) * pageSize.width+pageSize.width/2, subView.center.y);

    }
}

案例Dome:

https://github.com/ZPP506/ZPScrollerScaleView

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

推荐阅读更多精彩内容