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閱讀 228,606評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,582評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,540評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,028評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,801評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,223評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,294評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,442評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,976評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,800評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,996評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,543評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,233評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,662評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,926評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,702評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,991評論 2 374

推薦閱讀更多精彩內容