iOS實(shí)現(xiàn)UIScrollView的無(wú)限輪播功能(原理)詳解
前言
同學(xué)們?cè)趯?xiě)需求的時(shí)候肯定會(huì)時(shí)常用到UIScrollView。
UIScrollView控件是什么?
(1)移動(dòng)設(shè)備的屏幕⼤大⼩小是極其有限的,因此直接展⽰示在⽤用戶眼前的內(nèi)容也相當(dāng)有限
(2)當(dāng)展⽰示的內(nèi)容較多,超出⼀一個(gè)屏幕時(shí),⽤用戶可通過(guò)滾動(dòng)⼿手勢(shì)來(lái)查看屏幕以外的內(nèi)容
(3)普通的UIView不具備滾動(dòng)功能,不能顯⽰示過(guò)多的內(nèi)容
(4)UIScrollView是一個(gè)能夠滾動(dòng)的視圖控件,可以⽤用來(lái)展⽰示⼤大量的內(nèi)容,并且可以通過(guò)滾 動(dòng)查看所有的內(nèi)容
(5) 舉例:手機(jī)上的“設(shè)置”、其他⽰示例程序
而說(shuō)到UIScrollView,大家最先想到的肯定就是它上面的無(wú)限輪播功能。蘋(píng)果在UIScrollView上并沒(méi)有提供相應(yīng)的方法讓大家實(shí)現(xiàn)輪播,所以就需要通過(guò)代碼進(jìn)行處理來(lái)實(shí)現(xiàn)。
先上圖
無(wú)限輪播效果圖.gif
我先給大家講講其實(shí)現(xiàn)的原理:
我們假設(shè)用幾張圖片實(shí)現(xiàn)輪播效果。首先,我們需要打開(kāi)UIScrollView的分頁(yè)滑動(dòng)
/// 分頁(yè)滑動(dòng) _scrollView.scrollEnabled = YES;
它方便的幫助我們實(shí)現(xiàn)了輪播的效果,然后就需要我們來(lái)實(shí)現(xiàn)“無(wú)限的”輪播。接下來(lái),我們就需要擺放圖片了,在擺放圖片時(shí)需要注意,我們需要在第一張圖片的位置擺放最后一張圖片(可能有點(diǎn)懵哈,不過(guò)不要著急慢慢往下看),然后我們依次擺放圖片(從第一張到最后一張),最后在所有圖片的尾部我們?cè)俜派系谝粡垐D片。這樣我們就多放了兩張圖片(分別在首尾多放了一張圖)。我把對(duì)應(yīng)的方法寫(xiě)一下:
/// 將圖片放置在UIScrollView上 -(void)setupImage { /// 在UIScrollView的最前面添加一張圖片 UIImageView *firstImageView = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, kScreenWidth, self.scrollView.frame.size.height)]; /// 圖片名是最后一張圖片 firstImageView.image = [UIImage imageNamed:self.imageNameList.lastObject]; [self.scrollView addSubview:firstImageView]; /// 添加圖片 for (NSInteger index = 0; index < self.imageNameList.count; index ++) { /// UIScrollView上的每一張圖片 UIImageView *imageView = [[UIImageView alloc] initWithFrame:CGRectMake((index + 1) * kScreenWidth, 0, kScreenWidth, self.scrollView.frame.size.height)]; imageView.image = [UIImage imageNamed:self.imageNameList[index]]; [self.scrollView addSubview:imageView]; self.scrollView.contentSize = CGSizeMake((index + 2) * self.scrollView.bounds.size.width, 0); } /// 在UIScrollView的最后面添加一張圖片 UIImageView *lastImageView = [[UIImageView alloc] initWithFrame:CGRectMake((self.imageNameList.count + 1) * kScreenWidth, 0, kScreenWidth, self.scrollView.frame.size.height)]; /// 圖片名是第一張圖片 lastImageView.image = [UIImage imageNamed:self.imageNameList.firstObject]; [self.scrollView addSubview:lastImageView]; /// 設(shè)置UIScrollView的偏移量 self.scrollView.contentSize = CGSizeMake((self.imageNameList.count + 2) * self.scrollView.bounds.size.width, 0); /// 設(shè)置UIScrollView的起始偏移距離(將第一張圖片跳過(guò)) self.scrollView.contentOffset = CGPointMake(kScreenWidth, 0); /// 圖片總數(shù) self.pageControl.numberOfPages = self.imageNameList.count; self.pageControl.currentPage = 0; }
其實(shí),如果大家看到這里,應(yīng)該就會(huì)大致明白無(wú)線輪播的實(shí)現(xiàn)原理了。接下來(lái)就是最后一步,在UIScrollView的代理方法里面寫(xiě)邏輯:判斷UIScrollView的偏移量,當(dāng)其滑動(dòng)到首位時(shí)(顯示的是最后一張圖片),滑動(dòng)停止,就把偏移量修改最后面圖片的位置上(倒數(shù)第二張)。同理,當(dāng)UIScrollView滑動(dòng)到最后時(shí)(顯示的是第一張圖片),滑動(dòng)停止,就把偏移量修改到第一張圖片的位置上(正數(shù)第二張)。
#pragma mark - UIScrollViewDelegate -(void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView { /// 當(dāng)UIScrollView滑動(dòng)到第一位停止時(shí),將UIScrollView的偏移位置改變 if (scrollView.contentOffset.x == 0) { scrollView.contentOffset = CGPointMake(self.imageNameList.count * kScreenWidth, 0); self.pageControl.currentPage = self.imageNameList.count; /// 當(dāng)UIScrollView滑動(dòng)到最后一位停止時(shí),將UIScrollView的偏移位置改變 } else if (scrollView.contentOffset.x == (self.imageNameList.count + 1)* kScreenWidth) { scrollView.contentOffset = CGPointMake(kScreenWidth, 0); self.pageControl.currentPage = 0; } else { self.pageControl.currentPage = scrollView.contentOffset.x / kScreenWidth - 1; } }
ok,原理其實(shí)就是這樣。在首尾多加兩張圖片當(dāng)做占位符,然后當(dāng)UIScrollView滑動(dòng)到占位符的位置時(shí),改變UIScrollView的偏移量,簡(jiǎn)單且方便。下面就是全部代碼:
#import "ViewController.h" #define kScreenWidth [UIScreen mainScreen].bounds.size.width @interface ViewController () <UIScrollViewDelegate> /// 滑動(dòng)控制器 @property (nonatomic, strong) UIScrollView *scrollView; /// 圖片數(shù)組 @property (nonatomic, strong) NSArray<NSString *> *imageNameList; /// 頁(yè)碼控制器 @property (nonatomic, strong) UIPageControl *pageControl; @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; // 設(shè)置圖片名的數(shù)組 self.imageNameList = @[@"image0", @"image1", @"image2", @"image3"]; // 添加圖片 [self setupImage]; } /// 將圖片放置在UIScrollView上 -(void)setupImage { /// 在UIScrollView的最前面添加一張圖片 UIImageView *firstImageView = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, kScreenWidth, self.scrollView.frame.size.height)]; /// 圖片名是最后一張圖片 firstImageView.image = [UIImage imageNamed:self.imageNameList.lastObject]; [self.scrollView addSubview:firstImageView]; /// 添加圖片 for (NSInteger index = 0; index < self.imageNameList.count; index ++) { /// UIScrollView上的每一張圖片 UIImageView *imageView = [[UIImageView alloc] initWithFrame:CGRectMake((index + 1) * kScreenWidth, 0, kScreenWidth, self.scrollView.frame.size.height)]; imageView.image = [UIImage imageNamed:self.imageNameList[index]]; [self.scrollView addSubview:imageView]; self.scrollView.contentSize = CGSizeMake((index + 2) * self.scrollView.bounds.size.width, 0); } /// 在UIScrollView的最后面添加一張圖片 UIImageView *lastImageView = [[UIImageView alloc] initWithFrame:CGRectMake((self.imageNameList.count + 1) * kScreenWidth, 0, kScreenWidth, self.scrollView.frame.size.height)]; /// 圖片名是第一張圖片 lastImageView.image = [UIImage imageNamed:self.imageNameList.firstObject]; [self.scrollView addSubview:lastImageView]; /// 設(shè)置UIScrollView的偏移量 self.scrollView.contentSize = CGSizeMake((self.imageNameList.count + 2) * self.scrollView.bounds.size.width, 0); /// 設(shè)置UIScrollView的起始偏移距離(將第一張圖片跳過(guò)) self.scrollView.contentOffset = CGPointMake(kScreenWidth, 0); /// 圖片總數(shù) self.pageControl.numberOfPages = self.imageNameList.count; self.pageControl.currentPage = 0; } #pragma mark - UIScrollViewDelegate -(void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView { /// 當(dāng)UIScrollView滑動(dòng)到第一位停止時(shí),將UIScrollView的偏移位置改變 if (scrollView.contentOffset.x == 0) { scrollView.contentOffset = CGPointMake(self.imageNameList.count * kScreenWidth, 0); self.pageControl.currentPage = self.imageNameList.count; /// 當(dāng)UIScrollView滑動(dòng)到最后一位停止時(shí),將UIScrollView的偏移位置改變 } else if (scrollView.contentOffset.x == (self.imageNameList.count + 1)* kScreenWidth) { scrollView.contentOffset = CGPointMake(kScreenWidth, 0); self.pageControl.currentPage = 0; } else { self.pageControl.currentPage = scrollView.contentOffset.x / kScreenWidth - 1; } } #pragma mark - Get方法 -(UIScrollView *)scrollView { if (!_scrollView) { _scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, 0, kScreenWidth, 200)]; _scrollView.pagingEnabled = YES; _scrollView.clipsToBounds = NO; _scrollView.scrollEnabled = YES; _scrollView.delegate = self; _scrollView.bounces = NO; _scrollView.showsHorizontalScrollIndicator = NO; _scrollView.showsVerticalScrollIndicator = NO; [self.view addSubview:_scrollView]; } return _scrollView; } -(UIPageControl *)pageControl { if (!_pageControl) { _pageControl = [[UIPageControl alloc] initWithFrame:CGRectMake(0, 150, kScreenWidth, 50)]; _pageControl.pageIndicatorTintColor = [UIColor blackColor]; _pageControl.currentPageIndicatorTintColor = [UIColor grayColor]; [self.view addSubview:_pageControl]; } return _pageControl; } @end
好了,如果大家使用的是swift語(yǔ)言,還可以參考這篇文章:http://www.dbjr.com.cn/article/148191.htm
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
iOS App開(kāi)發(fā)中UISearchBar搜索欄組件的基本用法整理
iOS開(kāi)發(fā)組件中自帶的UISearchBar提供了很多基礎(chǔ)和好用的搜索欄UI功能,下面就來(lái)總結(jié)一下iOS App開(kāi)發(fā)中UISearchBar搜索欄組件的基本用法整理,需要的朋友可以參考下2016-05-05ios使用OC寫(xiě)算法之遞歸實(shí)現(xiàn)八皇后
本篇文章主要介紹了ios使用OC寫(xiě)算法之遞歸實(shí)現(xiàn)八皇后,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-08-08iOS應(yīng)用程序之間的幾種跳轉(zhuǎn)情況詳解
這篇文章給大家詳細(xì)介紹了iOS應(yīng)用程序之間跳轉(zhuǎn)的幾種情況,包括跳轉(zhuǎn)到另一個(gè)程序的主界面、跳轉(zhuǎn)到另一個(gè)程序的指定界面以及如何從目標(biāo)程序的非主頁(yè)界面回到當(dāng)前(跳轉(zhuǎn)前)程序呢?有需要的朋友們可以下面來(lái)一起看看。2016-09-09iOS UIScrollView和控制器返回手勢(shì)沖突解決方法
這篇文章主要介紹了iOS UIScrollView和控制器返回手勢(shì)沖突解決方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02