iOS 實(shí)現(xiàn)UIScrollView分頁滑動寬度自定義

前言

App中最常用 輪播圖, 關(guān)于它的實(shí)現(xiàn)有很多方法 如 :Anination, UIScrollView, UICollectionView 動畫是另一種思路, UICollectionView 繼承于 UIScrollView. 作者今天就用 UIScrollView 講一下 , 分頁效果下 滑動寬度小于屏幕寬度 露出上下頁內(nèi)容, 或 滑動視圖之間 間隙問題 . 如圖 :

分頁效果下 滑動寬度小于屏幕寬度 露出上下頁內(nèi)容
滑動視圖之間 間隙

一 思路整理

有些文章寫過 實(shí)現(xiàn)方法, 是把 pagingEnabled 設(shè)為NO, 自己寫出 分頁效果. 這么寫也可以. 作者更喜歡使用 系統(tǒng)的方法. 實(shí)現(xiàn)該效果, 有以下核心代碼 :

  1. pagingEnabled 設(shè)為YES, 用系統(tǒng)方法實(shí)現(xiàn).
  2. clipsToBounds 設(shè)為NO, 為了顯現(xiàn)上下頁內(nèi)容
  3. 分頁滑動寬度 系統(tǒng)默認(rèn)為 UIScrollViewwidth.
  4. 算法公式 : **(2 * i+ 1) *b + i * a ** => 說明 : b : 圖片間距一半, a : 圖片寬

二 封裝控件 RollView .h



#import <UIKit/UIKit.h>

/** 設(shè)置代理 */
@protocol RollViewDelegate <NSObject>

-(void)didSelectPicWithIndexPath:(NSInteger)index;
@end


@interface RollView : UIView

@property (nonatomic, assign) id<RollViewDelegate> delegate;

/**
 初始化
 
 @param frame 設(shè)置View大小
 @param distance 設(shè)置Scroll距離View兩側(cè)距離
 @param gap 設(shè)置Scroll內(nèi)部 圖片間距
 @return 初始化返回值
 */
- (instancetype)initWithFrame:(CGRect)frame withDistanceForScroll:(float)distance withGap:(float)gap;

/** 滾動視圖數(shù)據(jù) */
-(void)rollView:(NSArray *)dataArr;

@end

三 封裝控件 RollView .m


#import "RollView.h"

@interface RollView ()<UIScrollViewDelegate>

@property (nonatomic, strong) UIScrollView *scrollView;

@property (nonatomic, strong) NSArray *rollDataArr;   // 圖片數(shù)據(jù)

@property (nonatomic, assign) float halfGap;   // 圖片間距的一半

@end

@implementation RollView

- (instancetype)initWithFrame:(CGRect)frame withDistanceForScroll:(float)distance withGap:(float)gap
{
    
    self = [super initWithFrame:frame];
    if (self) {
        
        self.halfGap = gap / 2;
        
        /** 設(shè)置 UIScrollView */
        self.scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(distance, 0, self.frame.size.width - 2 * distance, self.frame.size.height)];
        [self addSubview:self.scrollView];
        self.scrollView.pagingEnabled = YES;
        self.scrollView.delegate = self;
        
        self.scrollView.clipsToBounds = NO;
        
        /** 添加手勢 */
        UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(tapAction:)];
        tap.numberOfTapsRequired = 1;
        tap.numberOfTouchesRequired = 1;
        [self.scrollView addGestureRecognizer:tap];
        self.scrollView.showsHorizontalScrollIndicator = NO;
        
        /** 數(shù)據(jù)初始化 */
        self.rollDataArr = [NSArray array];
        
    }
    
    
    return self;
}

#pragma mark - 視圖數(shù)據(jù)
-(void)rollView:(NSArray *)dataArr{

    self.rollDataArr = dataArr;
    
    
    //循環(huán)創(chuàng)建添加輪播圖片, 前后各添加一張
    for (int i = 0; i < self.rollDataArr.count + 2; i++) {
        
        for (UIView *underView in self.scrollView.subviews) {
            
            if (underView.tag == 400 + i) {
                [underView removeFromSuperview];
            }
        }
        
        UIImageView *picImageView = [[UIImageView alloc] init];
        picImageView.userInteractionEnabled = YES;
        picImageView.tag = 400 + i ;

        /**  說明
         *   1. 設(shè)置完 ScrollView的width, 那么分頁的寬也為 width.
         *   2. 圖片寬為a 間距為 gap, 那么 圖片應(yīng)該在ScrollView上居中, 距離ScrollView左右間距為halfGap.
         *   與 ScrollView的width關(guān)系為 width = halfGap + a + halfGap.
         *   3. distance : Scroll距離 底層視圖View兩側(cè)距離.  
         *   假設(shè) 要露出上下頁內(nèi)容大小為 m ,   distance = m + halfGap
         *
         *  圖片位置對應(yīng)關(guān)系 :
         *  0 ->  1 * halfGap ;
         *  1 ->  3 * halfGap + a ;
         *  2 ->  5 * halfGap + 2 * a ;
              .
              .
         *  i   -> (2 * i +1) *  halfGap + i *(width - 2 * halfGap )
         */


        picImageView.frame = CGRectMake((2 * i + 1) * self.halfGap + i * (self.scrollView.frame.size.width - 2 * self.halfGap), 0, (self.scrollView.frame.size.width - 2 * self.halfGap), self.frame.size.height);
        
        //設(shè)置圖片
        if (i == 0) {
    
            picImageView.image = [UIImage imageNamed:[NSString stringWithFormat:@"%@", self.rollDataArr[self.rollDataArr.count - 1]]];
            
        }else if (i == self.rollDataArr.count+1) {
            
            picImageView.image = [UIImage imageNamed:[NSString stringWithFormat:@"%@", self.rollDataArr[0]]];
        }else {
            
            picImageView.image = [UIImage imageNamed:[NSString stringWithFormat:@"%@", self.rollDataArr[i - 1]]];
        }
        
        [self.scrollView addSubview:picImageView];
    }
    //設(shè)置輪播圖當(dāng)前的顯示區(qū)域
    self.scrollView.contentOffset = CGPointMake(self.scrollView.frame.size.width, 0);
    self.scrollView.contentSize = CGSizeMake(self.scrollView.frame.size.width * (self.rollDataArr.count + 2), 0);
    
}

#pragma mark - UIScrollViewDelegate 方法
-(void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView {
       
    NSInteger curIndex = scrollView.contentOffset.x  / self.scrollView.frame.size.width;
    
    if (curIndex == self.rollDataArr.count + 1) {
        
        scrollView.contentOffset = CGPointMake(self.scrollView.frame.size.width, 0);
    }else if (curIndex == 0){
        
        scrollView.contentOffset = CGPointMake(self.scrollView.frame.size.width * self.rollDataArr.count, 0);
    }
    
}

#pragma mark - 輕拍手勢的方法
-(void)tapAction:(UITapGestureRecognizer *)tap{
    
    if ([self.rollDataArr isKindOfClass:[NSArray class]] && (self.rollDataArr.count > 0)) {
        
        [_delegate didSelectPicWithIndexPath:(self.scrollView.contentOffset.x / self.scrollView.frame.size.width)];
    }else{
        
        [_delegate didSelectPicWithIndexPath:-1];
    }
    
}

四 調(diào)用


#import "ViewController.h"
#import "RollView.h"

@interface ViewController ()<RollViewDelegate>

@property (nonatomic, strong) RollView *rollView;

@end

-(void)creatPicRollView{
    
    
    
    self.rollView = [[RollView alloc] initWithFrame:CGRectMake(0, 50, self.view.frame.size.width, 150) withDistanceForScroll:12.0f withGap:8.0f];
   
    /** 全屏寬滑動 視圖之間間隙,  將 Distance 設(shè)置為 -12.0f */
   // self.rollView = [[RollView alloc] initWithFrame:CGRectMake(0, 50, self.view.frame.size.width, 150) withDistanceForScroll: -12.0f withGap:8.0f];
   // self.rollView.backgroundColor = [UIColor blackColor];
    
    self.rollView.delegate = self;
   
    [self.view addSubview:self.rollView];
    
    
    NSArray *arr = @[@"1.jpg",
                     @"2.jpg",
                     @"3.jpg"];
    
    [self.rollView rollView:arr];
}

#pragma mark - 滾動視圖協(xié)議
-(void)didSelectPicWithIndexPath:(NSInteger)index{
    
    if (index != -1) {
        
        NSLog(@"%ld", (long)index);
    }
    
}

五 效果

分頁效果下 滑動寬度小于屏幕寬度 露出上下頁內(nèi)容
**滑動視圖之間有間隙**

以 上 !

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末镜遣,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子偶洋,更是在濱河造成了極大的恐慌映穗,老刑警劉巖诬乞,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡懂更,警方通過查閱死者的電腦和手機(jī)锌订,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進(jìn)店門竹握,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人辆飘,你說我怎么就攤上這事啦辐∥酱” “怎么了?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵芹关,是天一觀的道長续挟。 經(jīng)常有香客問我,道長侥衬,這世上最難降的妖魔是什么诗祸? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮轴总,結(jié)果婚禮上直颅,老公的妹妹穿的比我還像新娘。我一直安慰自己怀樟,他們只是感情好功偿,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著往堡,像睡著了一般脖含。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上投蝉,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天养葵,我揣著相機(jī)與錄音,去河邊找鬼瘩缆。 笑死关拒,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的庸娱。 我是一名探鬼主播着绊,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼熟尉!你這毒婦竟也來了归露?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤斤儿,失蹤者是張志新(化名)和其女友劉穎剧包,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體往果,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡疆液,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了陕贮。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片堕油。...
    茶點(diǎn)故事閱讀 37,997評論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出掉缺,到底是詐尸還是另有隱情卜录,我是刑警寧澤,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布眶明,位于F島的核電站艰毒,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏赘来。R本人自食惡果不足惜现喳,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一凯傲、第九天 我趴在偏房一處隱蔽的房頂上張望犬辰。 院中可真熱鬧,春花似錦冰单、人聲如沸幌缝。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽涵卵。三九已至,卻和暖如春荒叼,著一層夾襖步出監(jiān)牢的瞬間轿偎,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工被廓, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留坏晦,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓嫁乘,卻偏偏與公主長得像昆婿,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子蜓斧,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評論 2 345

推薦閱讀更多精彩內(nèi)容

  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫仓蛆、插件、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,022評論 4 62
  • 前段時(shí)間挎春,有個(gè)朋友讓我?guī)兔ψ鲆粋€(gè)圖片在UIScrollView滑動翻頁的效果看疙,只不過圖片寬度只有屏幕的一半,屏幕左...
    iOS_凱閱讀 12,585評論 11 33
  • 樟樹正香 和風(fēng)正好 卿本佳人 何事自擾 樟樹自香 和風(fēng)自好 我心自憂 與卿何擾
    卿云歌兮閱讀 213評論 0 1
  • 聽一個(gè)人的歌 細(xì)細(xì)的聽它里面的曲調(diào) 或高或低或輕快或低沉 跟著這些跳動的音符 體會它隱藏的那些情愫 感覺這樣就能 ...
    大樹與洞閱讀 265評論 0 3
  • 周末上午閑來無事,跟室友瞎侃帮碰,從兼職的工資好低相味,到G室友太懶了從來都不打掃寢室,最后不知道怎么回事就聊到了找不到男...
    酌飲秋黃閱讀 542評論 6 6